-
Notifications
You must be signed in to change notification settings - Fork 14
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Performance while watching lots of files #64
Comments
Sorry for the delay in responding. How many is "lots?" :) This is the first issue I've had reporting performance issues. Way under the hood I'm using Is your project open source? |
Hi @shannonmoeller, I need to count those partials but I think it's 200+. Our projects are not open source but the build framework that uses your gulp task is: https://github.com/biotope/biotope-build Last week I learned two things:
After debugging your code I don't think this is in any case related to your plugin, sorry for buggin you. But anyway, I think it would still be an interesting feature to cache partials. That would boost the overall performance dramatically. But I think this needs to happen directly inside handlebars package. If you have any ideas how we could get something started, let me know! :) |
You can use a cache of the helpers and partials like this: const gulp = require('gulp');
const hb = require('gulp-hb');
const rename = require('gulp-rename');
const hbStream = hb({ bustCache: false })
.helpers('helpers/**/*.js')
.partials('partials/**/*.js');
function markup() {
return gulp
.src('src/posts/**/*.hbs')
.pipe(hbStream)
.pipe(rename({ extname: '.html' }))
.pipe(gulp.dest('dest'));
}
function watchMarkup() {
return gulp.watch('src/posts/**/*.js', markup);
}
gulp
.task('markup', markup)
.task('watch-markup', watchMarkup); The problem is that if you change any of your partials or helpers you won't get the newest versions when the files are rebuilt by the watch task. I can see how switching between branches without restarting the watch task would be quite the hit to performance. There might be room to add a feature to |
I thought about that too, and now I understand the usage of the bustCache option! First I thought about using something like https://github.com/gulp-community/gulp-cached that only pipes changed files into handlebars registerPartial function that would need something like an add (if its a new template) or replace (if the template changed) logic. |
@shannonmoeller have you ever tried to add updated partials to the hbStream with bustCache set to false? Something like:
If that works we already have the incremental update mechanism :D |
It's possible. I've never tried. |
You might need to do: hbStream.partials('partials/partial1.hbs', { bustCache: true }); |
The real issue in solving this transparently is the dependency tree. If you only want to update changed partials, you likely also only want to update files that make use of that partial, etc. That tree doesn't exist right now. |
I will try that... and tell you if it worked out. That would be a really great performance booster!!! |
We just wrote at the same time :) Maybe we switch to gitter?! But first finalize the thought here.... What I saw during tests the registerPartial was the main performance killer. If that is gone, or reduced to just updating a single file, the process of compiling all root templates should be ok. Also knowing the dependency tree and updating only the relevant partials would be the perfect solution. But updating and re-using the hbStream would be a major first step! Thank you already for your feedback! |
Ok, I just tried some things. First I modified the hbStream to be initially filled only once and updated by the watch task:
This ends with an Second test was the same code with pump syntax. I don't see any errors but the partials are not updated.
This just runs but doesn't update the partials inside hbStream :( |
You're right. The stream is getting closed. Does this work? Edit: Deleted bad code example. |
Actually, that won't work because hbStream is now undefined in the watch task. Doh. Let me rethink that. :) |
Ok. I think I have a way to do this, but it's kind of annoying. const gulp = require('gulp');
// import rename, watch, and runSequence
const hb = require('gulp-hb');
const handlebarsWax = require('handlebars-wax');
// Use the same handlebars instance for everything
const handlebars = hb.handlebars;
// Create a new wax wrapper so we can use `.partials()`
const wax = handlebarsWax(handlebars, {
bustCache: true
});
gulp.task('hb', () => {
// Go ahead and recreate the stream every time,
// but use the shared handlebars instance
const hbStream = hb({ bustCache: false, handlebars })
.partials('partials/**/*.hbs');
return gulp
.src('pages/**/*.hbs')
.pipe(hbStream)
.pipe(rename({extname: ".html"}))
.pipe(gulp.dest(config.global.dev))
});
gulp.task('watch:hb', () => {
watch(files, config.watch, (vinyl) => {
// Register changed partials on the shared Handlebars
// instance with our wax wrapper
wax.partials(vinyl.path);
runSequence('hb');
});
}); This is just a hack to see if this actually solves the performance issues you're seeing. |
Also, one case this doesn't solve is the deletion of a partial. |
In your example I don't see any connection between wax and the hbStream, except they are using the same handlebars instance. The code that eats most performance is hb().partials() and that is recreated every time the task executes. As long as wax (or in the end handlebars itself) is not capable of adding/updating/removing single partials this won't work. By the way, removing of files can be found out by the vinyl state of the file given by the watch task. |
Exactly.
I'll need to look into how I'm compiling the templates.
Found out, yes, but there's no API at present to remove it from the internal Handlebars cache. That would need to be added. |
Hi @shannonmoeller, I just invited you to a minimal demo repository that reflects the idea behind this "caching hbs" challenge: https://github.com/janrembold/handlebars-cache Hopefully it helps to get it kickstarted somehow... |
Awesome. Thanks! |
I tried to debug into gulp-hb->wax->handlebars and got lost pretty quick :( So I tried another approach to simply proof what I want to achieve and this PoC works like a charm!!! This is not the gulp'ish way and a very simple demo but it really works great. Please have a look at this branch https://github.com/janrembold/handlebars-cache/tree/precompilation-tests |
The same works fine with handlebars-wax: https://github.com/janrembold/handlebars-cache/tree/wax |
Me again... I tried to refactor the gulp-hb demo (https://github.com/janrembold/handlebars-cache/tree/master) but that doesn't work. The bustCache option has no effect on the compiled template. The resulting template isn't updated. But honestly I think we don't need the gulp wrapper anymore for this kind of workflow. This is not a real pipeline workflow anymore. I don't see any positive effects from using gulp pipelines over direct node fs calls. I also think If I find the time and understand your unhookRequire construct, I would like to create a PR if this additional feature would be OK for you?! What do you think? |
Hi @shannonmoeller, I did lots or research and refactorings during the last weeks and ended up with a complete rewrite of the task based on plain handlebars. If you are curious you can have a look at the task here: https://github.com/biotope/biotope-build/blob/hb-static-performance/tasks/hb2.js In short, this task uses only a single handlebars instance that is initially filled with all partials and helpers. I also pre-collect and compile all templates and data files and update only single files during runtime with specific watch tasks. After those really tiny updates I compile all templates again because we never know which templates are affected by partial/data changes. In our large projects this is up to 30% faster than the "easy way" using But I'm really confused about some results that came up with this refactoring. As stated above in an earlier post I found out that the registerPartial function consumes up to 70% of the complete task in Within the new refactored plain handlebars task I see something completely different. The registering and compiling of all templates and partials takes only about 16% of the total task execution time. More than 80% of the time is consumed by compiling the static html strings. So I asked myself if you have found any ways to dramatically speed up hbs compilation in any way. All handlebars options, like Thanks in advance!!! |
Hi @shannonmoeller,
we experience really bad performance issues while watching lots of hbs templates and partials in some of our larger projects. Especially on windows systems the task runs minutes to complete.
I think this is caused by a fresh and complete rebuild of the hbStream on every task execution.
I was thinking about creating the stream only once on startup and updating only changed parts incrementally on file changes, but sadly haven't found a way to manage that.
I think this incremental update approach would boost the performance dramatically. Do you think this is possible somehow?
Thanks for any help in advance!!!
The text was updated successfully, but these errors were encountered: