-
Notifications
You must be signed in to change notification settings - Fork 1.2k
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
Support for asynchronous loading (and not packing everything in one file) #58
Comments
1: wrapping a whole file in a function block is ugly 2: node modules use synchronous requires 3: browserify's goal is to let code written for node run in the browser |
That's a quite common pattern in Javascript. This avoids poluting the global namespace, this allows to close some variables private to the module. That's called the module pattern.
That's not a problem: Once a module is loaded you can synchronously // this loads 'foo' and 'bar' before calling the function
define(['foo', 'bar'], function() {
// this synchronous require() works because 'foo' is already loaded:
var foo = require('foo');
}); See the example code below "Or even easier (just wrap the unmodified script with a header and footer)" So a node module using synchronous requires can be made asynchronous by just wrapping it in a
In the browser, that's why the ability to make them asynchronous at the same time would be awesome |
This issue completely misses the point of browserify. If you want that stuff go use requirejs. |
So converting nodejs modules to asynchronous modules is a quite obvious way to load them in the browser. Doing something else is completely missing the point. |
We accept pull requests. |
Great! Could you reopen the issue then ? |
@arnaud-ib Why don't you use requirejs ? |
They can by defined in a way that they can be asynchronously loaded (see http://wiki.commonjs.org/wiki/Modules/AsynchronousDefinition ). As I shown in the issue, converting a synchronous module to an asynchronous one is very easy.
Yes, this is a draft that now is maintained on the AMD project, which stands for Asynchronous Module Definition: From the page:
It is a way to reliably load multiple inter-dependent javascript files asynchronously.
requirejs only provides |
It's not part of Modules 1.x You also said:
Fact is, that modules in Node are only synchronous and there are no plans even in far future for asynchronous support . Browserify is closely tied to Node and share same approach. Running modules asynchronously will just slow down page load and will still need kind of file wrapping, I'm not sure what's the added value. |
Please read earlier answers :)
Why is this a problem ? All script loaders conform to this spec. Even node implements it. Try it yourself: // foo.js
define(function() {
return {yes: "node implements asynchronous module definition too"};
});
// bar.js
foo = require('./foo');
console.log(foo.yes);
What is node's approach ? To concatenate the world in a single file ? To do things synchronously ? Node's approch is to load scripts on demand. The let's load scripts synchronously approch is based on the fact that node can do it, and with guarantees on the order of execution of required scripts. BTW that's probably the only synchronous by default thing in node. Browserify is porting node modules to the web. In a web environment you can't load multiple scripts synchronously in a portable manner, and most importantly, you can't guarantee the order of execution. That means you can't guarantee that module A is executed before module B, and this is bad if module B depends on module A. That's why CommonJS for the web uses asynchronous loading. That's still follows node's approach to load scripts on demand, and to keep modules in a separate file. Browserify wraps and packs everything in a single file. Nothing avoid it from converting everything to asynchronously-loadable modules. That's just cleaner and follows node's philosophy.
So the page will actually appear to load faster to the user with asynchronous loading. This also means that you can load some modules only when needed, for example you can load your form validation module only when the user actually uses a form. Any optimisation tips you read these days will recommend you to load your scripts asynchronously.
With current browserify, you get the following problems:
|
All CommonJS-loaders-for-the-web are using the async-loading, approach, not the pack-everything one: |
I don't think the "async loading" vs "all in one file" approach is at all settled. Look at the discussion here: https://github.com/paulirish/html5-boilerplate/issues/28 In general, the browserify approach is great for what it tries to do. I use it not to emulate node.js modules, but rather to allow for me to package my node.js module (so I get code sharing) as something that is suitable to the browser. Given what you want, it seems that browserify isn't the project for you. You'd be better served by something like requireJS/yabble/etc. |
Doesn't anyone read my answers before replying ? This is what I want ! Using browserify to produce asynchronously defined modules, and using requireJS/yabble/etc to load the produced modules. Producing scripts and loading them are two different things. |
I apologize - I did read your answers, but your intent wasn't clear to me at all. I understand what you're looking for now. Personally, as a user of browserify, I'm much more in the camp of "single file 'melding'", as I find it much easier to code and deploy that way. I also haven't seen convincing evidence that async loading of scripts is faster than a single, gzipped file. So to answer your concerns:
|
Node never implemented it, there was just a brief while when Node (in unstable branch) allowed 'define' construct to require Modules, but it was working same as regular 'require' - synchronously. Shortly after it was added it was taken out, it's not longer there, and won't be put back. Check following links: https://groups.google.com/d/msg/nodejs-dev/wITGE8TLk2M/ewajnrAF5BkJ Asynchronous script loading that you are talking about can be optimal when we talk about few files and not tens of them, (see discussion that itay linked), thing is when we talk about modules we usually talk about tens of files. Currently I'm working on web application (written purely in JavaScript), which at the moment consists of 103 modules from 11 packages, most of those modules are needed to show initial page, I can't imagine loading them each by each asynchronously, it will significantly slow down page load. Basically I've been there before, few years ago, in development I used to load each file separately, but after some time I switch to joined file load as growing number of files made my page load slow. So conclusion is, that it might be good to separate JS code base into few separate files but it won't work great at modules level, where we may have hundreds of files, it may work fine for packages (that's why I said AMD is conceptually more about packages not modules) but I think best option would be if Browserify knowing what's most optimal will prepare number of files that will result in fastest load in Browser - and such option will definitely be appreciated |
That's exactly why you want to use require() (sync or async)
I must have checked out the revision where define() was implemented
Obviously. You can still concatenate some modules together to do less requests. But do not pack everything. Have a big "core.js" in which you put things that every page on you site depends on. And a few set of other files for things that are not always required. The thing is that you'll still be able to load at least some parts asynchronously, and the site will load faster. And during development you won't have to pack the files files, all modules will be loaded independently, and transparently. Dev will be easier.
You can load the core of the app in a On a website, when javascript is not absolutly required to render the page, you can easily load everything but the loader asynchronously, and have the page rendered quickly. |
@arnaud-lb |
As an observer, I am having a hard time understanding why @arnaud-lb is getting so much blow-back for this feature request. It's likely a minimally invasive change, given that Browserify already needs to track this dependency information, and offers significant benefits for someone who has a large set of shared dependencies between two sets of unrelated code. It could be optional, so it need not affect people who don't care about it. If @arnaud-lb is willing to do the coding work, and Browserify is willing to accept pull requests, can we just stop belabouring the point until there is code to look at? |
@noonat you're totally right. |
Browserify sounded great at first, but once I looked into it deeper, I decided it wasn't yet the tool for me. Here's why:
For these reasons, I completely agree with @arnaud-lb. If Browserify allowed me to do the things above, I would not hesitate in the least to use it over RequireJS or others. Initial page load time is very important for my apps, and there are many things that can be done using async loading techniques to really make an application snappy. I don't like the idea of stalling the page render so that one massive browserified script can get loaded. Also, consider that the iPhone 3 limits cached elements to 15K. If I use browserify to combine my app and all of its dependencies into one file, then that poor iphone 3 user out there is loading the entire thing every time. Anyway, I understand that the point of browserify is for synchronous loading. But I believe it would get a lot more interest and traction if it was extended to have async features. I hope any pull requests that add async features would be welcome. |
Many bikes were shed. |
For those who needs this feature; https://github.com/azer/onejs#multiple |
@substack can you please reopen this issue? I see agreement that this is:
I assume you closed this thinking that there was no traction. Well this is still a problem despite the fact that no one has fixed it (I'd hope this is obvious ; ). I'd also like to hear from @arnaud-lb why he went dark, if he's willing to revisit this issue, and/or how he worked around it in his projects. |
I don't care if there is |
What changed your mind since you originally reopened the issue after saying " |
Well, the fact is that node ecosystem is completely different from the Browser, maybe the Browserify "mission" is just nice, but doesn't work for the web. |
Here's mine https://github.com/epeli/browserify-externalize |
I've been working on my own solution as well: https://github.com/fresheneesz/asyncify Its purpose is to make commonJS code just work with browsers, as long as you have control over the server-side (which you probably do if you care about this kind of thing). Browsers can request a set of modules, and the server will package up those modules and all their nested dependencies that haven't been already requested by the browser (no load duplication) and sends em down. It uses detective (just like browserify) to find dependencies and will also use browser-builtins to serve browser-side versions of node.js builtins. Its pretty close to being in a working state, and I'll probably try to put up a working version sometime in the next month (i'm pretty busy at the moment). Let me know if anyone is interested in helping me speed up this process. |
Here's mine: https://github.com/webpack/webpack It uses CommonJS and AMD. CommonJS and AMD |
Although you should think twice about asynchronously loading anything from a remote url, etc., you can use bromote for that use case. Using this together with browserify However I agree 100% that this functionality should not be part of browserify itself. |
I'm using webpack because it offers a bundle-feature that allows me to split the entire code into separate bundles that can be loaded on demand. They can also be cached more efficiently. Don't get me wrong, I don't want to advertise another module in this issue. But I think that this feature is very useful. |
I just wanted to note that I've since tried webpack and have to say it handles asynchronous loading exceptionally well using the commonjs |
+1 to @arnaud-lb's ideas |
Packing everying in one file causes some problems:
This is a feature request to add support (as a browserify option) for using asynchronous loading instead of packing everything in a single file.
This would solve all the problems described above: dev becomes easier, bandwidth is saved, and pages load faster (because scripts are loaded asynchronously).
Most node modules are currently using synchronous loading, but it would be possible to convert a script like that:
To something like this:
Or even easier (just wrap the unmodified script with a header and footer):
Optimization
Such asynchronous modules can still be packed together in a single javascript file for reducing the number of HTTP requests in production.
This is done by just adding the module name as first argument of
define
(define('moduleName', ['dependency'], func)
) and concatenating all modules.The text was updated successfully, but these errors were encountered: