sourceMap use
Opened this issue · 20 comments
It's not clear how to generate source maps.
There is a single paragraph about it in http://lesscss.org/usage/#programmatic-usage.
Without even a simple example it's impossible to make it work easily.
Can you please provide an example?
Hi @DarkPark, on the first sight the docs look pretty straight forward. But some points possible need more explanation.
If you already have some non working code, stackoverflow will maybe a better place to ask your questions.
First of all you should notice the difference between the less.js in browser compiler and the node compiler. When your are using the in browser compiler you can not use sourcemaps at all, due to less/less.js#1541
The docs describe two ways to call less.render:
less.render(lessInput,optionObject,function(error,output){}
which should be used whenless = require('less');
after runningnpm install less
. The second parameter (output
) of the anonymous function in the preceding willalways return a string. If you have set the sourcemap option, your sourcemap will be append to thereturn an object.output
string.
An example:
less = require('less');
less.render('@c: red; p{c: @c;}',{sourceMap:{}},function(error,output)
{
console.log(typeof output);
console.log(output);
});
outputs:
object
{ css: 'p {\n c: red;\n}\n',
map: '{"version":3,"sources":["input"],"names":[],"mappings":"AAAS;EAAE,MAAA"}',
imports: [] }
- The
less
object is a promise object. This happens when you use the in browsers less.js or less-node (less = require(less/lib/less-node)
). With the promise object you can use: less.render(lessInput).then(function(output) {});`
In the above output
is a object; output.css
holds the compiled code and output.css
the sourcemap code when sourcemaps are enabled:
An example:
less = require('/usr/local/lib/node_modules/less/lib/less-node');
less.render('@c: red; p{c: @c;}',{sourceMap:{}})
.then(function(output) {
console.log('output.css: ' + output.css + "\n");
console.log('output.map: ' + output.map + "\n");
});
outputs:
output.css: p {
c: red;
}
output.map: {"version":3,"sources":["input"],"names":[],"mappings":"AAAS;EAAE,MAAA"}
Without the sourcemaps enabled, as explain above, you can run the same code in your browser:
<script src="https://code.jquery.com/jquery-2.1.3.min.js"></script>
<script>
less.render('@c: red; p{c: @c;}',{})
.then(function(output) {
console.log('output.css: ' + output.css + "\n");
console.log('output.map: ' + output.map + "\n");//undefined now
});
</script>
The docs also tell you that you can set the sourcemap option object in the .then(function(output,{sourceMap:{}}) function. I believe this is wrong. Options should be set in the render call. So you should use the following code:
less.render(lessInput,{sourceMap:{}})
.then(function(output) {
// output.css = string of css
// output.map = string of sourcemap
}
Finally when you are looking for a example of the use of less-node (the promise method) you can take a look at the source of the command line compiler.
thank you for this detailed answer
I would also appreciate some words regarding sourceMapBasepath
, sourceMapRootpath
, outputSourceFiles
as I was not able to make it to create separate map files and have to manually save them from output.map
.
I was not able to make it to create separate map files and have to manually save them from output.map
This is how it's supposed to be. The less.render
itself is not supposed to work with files at all (with a few exceptions like imports). So these particular options are actually more like lessc
options not really the less
library options (it's assumed that if you use less
library programmatically you do not want just some "javascript trigger" for the lessc
but rather need a complete control of the whole compilation process instead, including all those files to create).
In other words, you may need to consult the lessc
code (and possibly copy-paste some snippets from there if it's about just cloning its functionality).
but this line https://github.com/less/less.js/blob/v2.2.0/bin/lessc#L398 shows that it should actually write separate source map files by itself
it should actually write separate source map files by itself
Nope. The writeSourceMap
is called within the render
callback not by the render
itself. (That is, if you need the same you need to write the same within your render
callback too. The source code of the writeSourceMap
is at L344).
this whole conception is not clear for me
there is a code inside which does seems exactly what I need but you say it's not working and I need to rewrite it in my app
it's rather strange
there is a code inside which does seems exactly what I need but you say it's not working
This is the code of the lessc
- the command line compiler script. It's not something you get with less = require("less");
(It looks like you're thinking that the code at https://github.com/less/less.js/blob/v2.2.0/bin/lessc#L398 is the source code of the render
function. No, it's not, it's just the code using render
. In other words lessc
is just the same node
app using less
library as yours).
thank you for explanations
what should I do if...
not:
output.map: {"version":3,"sources":["input"],"names":[],"mappings":"AAAS;EAAE,MAAA"}
what I want:
output.map: {"version":3,"sources":["main.less"],"names":[],"mappings":"AAAS;EAAE,MAAA"}
because I want to use http://code.tutsplus.com/tutorials/working-with-less-and-the-chrome-devtools--net-36636, But the responsibility of compile less to css is my nodejs server.
Wow. All this discussion would not have to be if there was a simple explanation of what the different properties in the sourceMap
object do and a simple example for one of the most obvious cases was given.
if there was a simple explanation of what the different properties in the sourceMap object do
Ready to make a PR?
Always ready to create PRs but I still don't fully understand how it works.
I'll create a PR when I figured it out.
@pixelass The options are actually options that are defined by Mozilla's source-map
library, AFAIK. However, it looks like Mozilla's source map API may have changed, so I'm not certain if / how Less's is working. Except maybe it's pulling in a custom build from lib/source-map
with the older API?
@matthew-dean thx. That's a good hint. I'll look into it.
The options are actually options that are defined by Mozilla's source-map library
No, most of these options are Less own options (most of them don't actually affect anything in the sourcemap itself at all, thus have nothing to do with Mozilla's source map API). There're a few more similar options (e.g. sourceMapOutputFilename
) not applicable at the lessc
command line but required to be set in programmatic usage (and then yet again, nothing but lessc
code itself can help you on learning how to use such options programmatically).
My bad. Hence the AFAIK, but in this case, turns out I don't lol. The source map stuff is mostly a black box to me, since any code I've worked on in Less.js hasn't (yet) touched it.
@seven-phases-max thx for the clarification, so to make the documentation valid, you'd suggest looking at the source code? Can you point me to the correct file/line-number or link a comment if it has been mentioned?
I'm going to open a PR as soon as possible, since this is long overdue.
@pixelass
Well, first of all, notice that most of options are already explained here.
For the rest see how lessc
works with all that (for the options look for sourceMap*
vars/properties).
I.e.:
- options - https://github.com/less/less.js/blob/v2.7.2/bin/lessc#L300-L341
- render - https://github.com/less/less.js/blob/v2.7.2/bin/lessc#L481
(you may find earlier versions more easy to read, e.g. https://github.com/less/less.js/blob/v2.2.0/bin/lessc, as in latests versions all these async callbacks buzzing on top of each other are really awful).