2010-07-25 01:23:37 -04:00
|
|
|
fs = require 'fs'
|
2010-11-04 22:53:07 -04:00
|
|
|
path = require 'path'
|
2011-07-05 07:58:04 -04:00
|
|
|
{extend} = require './lib/coffee-script/helpers'
|
2010-07-25 01:23:37 -04:00
|
|
|
CoffeeScript = require './lib/coffee-script'
|
|
|
|
{spawn, exec} = require 'child_process'
|
2010-02-16 01:04:48 -05:00
|
|
|
|
2010-06-15 21:33:53 -04:00
|
|
|
# ANSI Terminal Colors.
|
2011-10-07 00:39:24 -04:00
|
|
|
bold = red = green = reset = ''
|
2013-02-01 18:50:32 -05:00
|
|
|
unless process.env.NODE_DISABLE_COLORS
|
2012-01-09 12:50:51 -05:00
|
|
|
bold = '\x1B[0;1m'
|
|
|
|
red = '\x1B[0;31m'
|
|
|
|
green = '\x1B[0;32m'
|
|
|
|
reset = '\x1B[0m'
|
2010-06-15 21:33:53 -04:00
|
|
|
|
2010-11-10 23:06:26 -05:00
|
|
|
# Built file header.
|
|
|
|
header = """
|
|
|
|
/**
|
|
|
|
* CoffeeScript Compiler v#{CoffeeScript.VERSION}
|
|
|
|
* http://coffeescript.org
|
|
|
|
*
|
2011-01-31 22:39:12 -05:00
|
|
|
* Copyright 2011, Jeremy Ashkenas
|
2010-11-10 23:06:26 -05:00
|
|
|
* Released under the MIT License
|
|
|
|
*/
|
|
|
|
"""
|
|
|
|
|
2010-02-16 19:17:57 -05:00
|
|
|
# Run a CoffeeScript through our node/coffee interpreter.
|
2011-07-18 17:47:31 -04:00
|
|
|
run = (args, cb) ->
|
2012-03-25 06:17:46 -04:00
|
|
|
proc = spawn 'node', ['bin/coffee'].concat(args)
|
2010-10-24 12:48:42 -04:00
|
|
|
proc.stderr.on 'data', (buffer) -> console.log buffer.toString()
|
2011-07-18 17:47:31 -04:00
|
|
|
proc.on 'exit', (status) ->
|
|
|
|
process.exit(1) if status != 0
|
|
|
|
cb() if typeof cb is 'function'
|
2010-02-16 01:04:48 -05:00
|
|
|
|
2010-06-15 21:33:53 -04:00
|
|
|
# Log a message with a color.
|
2010-07-25 01:23:37 -04:00
|
|
|
log = (message, color, explanation) ->
|
2010-10-24 12:48:42 -04:00
|
|
|
console.log color + message + reset + ' ' + (explanation or '')
|
2010-06-15 21:33:53 -04:00
|
|
|
|
2010-02-25 21:53:42 -05:00
|
|
|
option '-p', '--prefix [DIR]', 'set the installation prefix for `cake install`'
|
2010-02-25 21:43:42 -05:00
|
|
|
|
|
|
|
task 'install', 'install CoffeeScript into /usr/local (or --prefix)', (options) ->
|
2010-07-25 01:23:37 -04:00
|
|
|
base = options.prefix or '/usr/local'
|
2010-08-07 08:02:16 -04:00
|
|
|
lib = "#{base}/lib/coffee-script"
|
|
|
|
bin = "#{base}/bin"
|
2010-07-25 01:23:37 -04:00
|
|
|
node = "~/.node_libraries/coffee-script"
|
2010-10-24 12:48:42 -04:00
|
|
|
console.log "Installing CoffeeScript to #{lib}"
|
|
|
|
console.log "Linking to #{node}"
|
|
|
|
console.log "Linking 'coffee' to #{bin}/coffee"
|
2010-02-17 01:24:02 -05:00
|
|
|
exec([
|
2010-08-07 08:02:16 -04:00
|
|
|
"mkdir -p #{lib} #{bin}"
|
|
|
|
"cp -rf bin lib LICENSE README package.json src #{lib}"
|
2010-12-26 20:34:26 -05:00
|
|
|
"ln -sfn #{lib}/bin/coffee #{bin}/coffee"
|
|
|
|
"ln -sfn #{lib}/bin/cake #{bin}/cake"
|
2010-06-15 20:40:10 -04:00
|
|
|
"mkdir -p ~/.node_libraries"
|
2011-07-05 07:58:04 -04:00
|
|
|
"ln -sfn #{lib}/lib/coffee-script #{node}"
|
2010-02-26 19:49:12 -05:00
|
|
|
].join(' && '), (err, stdout, stderr) ->
|
2010-10-24 15:50:18 -04:00
|
|
|
if err then console.log stderr.trim() else log 'done', green
|
2010-02-26 19:49:12 -05:00
|
|
|
)
|
2010-02-17 01:24:02 -05:00
|
|
|
|
|
|
|
|
2011-07-18 17:47:31 -04:00
|
|
|
task 'build', 'build the CoffeeScript language from source', build = (cb) ->
|
2010-07-25 01:23:37 -04:00
|
|
|
files = fs.readdirSync 'src'
|
2013-01-05 17:05:09 -05:00
|
|
|
files = ('src/' + file for file in files when file.match(/\.(lit)?coffee$/))
|
2011-08-05 18:39:55 -04:00
|
|
|
run ['-c', '-o', 'lib/coffee-script'].concat(files), cb
|
2010-02-16 19:17:57 -05:00
|
|
|
|
2010-02-16 20:42:10 -05:00
|
|
|
|
2011-10-07 00:39:24 -04:00
|
|
|
task 'build:full', 'rebuild the source twice, and run the tests', ->
|
2011-07-18 17:47:31 -04:00
|
|
|
build ->
|
|
|
|
build ->
|
2011-09-04 12:14:13 -04:00
|
|
|
csPath = './lib/coffee-script'
|
2013-01-13 17:59:05 -05:00
|
|
|
csDir = path.dirname require.resolve csPath
|
|
|
|
|
|
|
|
for mod of require.cache when csDir is mod[0 ... csDir.length]
|
|
|
|
delete require.cache[mod]
|
|
|
|
|
2011-10-07 00:39:24 -04:00
|
|
|
unless runTests require csPath
|
2011-08-10 21:26:16 -04:00
|
|
|
process.exit 1
|
2010-03-08 06:34:07 -05:00
|
|
|
|
|
|
|
|
2010-02-21 11:40:52 -05:00
|
|
|
task 'build:parser', 'rebuild the Jison parser (run build first)', ->
|
2010-12-18 17:44:28 -05:00
|
|
|
extend global, require('util')
|
2010-06-28 00:26:45 -04:00
|
|
|
require 'jison'
|
2011-08-09 01:34:09 -04:00
|
|
|
parser = require('./lib/coffee-script/grammar').parser
|
2011-08-09 01:37:48 -04:00
|
|
|
fs.writeFile 'lib/coffee-script/parser.js', parser.generate()
|
2010-02-16 01:04:48 -05:00
|
|
|
|
2010-02-16 20:42:10 -05:00
|
|
|
|
2010-02-17 00:22:06 -05:00
|
|
|
task 'build:ultraviolet', 'build and install the Ultraviolet syntax highlighter', ->
|
2010-03-14 11:59:55 -04:00
|
|
|
exec 'plist2syntax ../coffee-script-tmbundle/Syntaxes/CoffeeScript.tmLanguage', (err) ->
|
|
|
|
throw err if err
|
2010-02-17 00:22:06 -05:00
|
|
|
exec 'sudo mv coffeescript.yaml /usr/local/lib/ruby/gems/1.8/gems/ultraviolet-0.10.2/syntax/coffeescript.syntax'
|
|
|
|
|
|
|
|
|
2010-02-24 19:57:29 -05:00
|
|
|
task 'build:browser', 'rebuild the merged script for inclusion in the browser', ->
|
2010-11-10 23:06:26 -05:00
|
|
|
code = ''
|
|
|
|
for name in ['helpers', 'rewriter', 'lexer', 'parser', 'scope', 'nodes', 'coffee-script', 'browser']
|
|
|
|
code += """
|
|
|
|
require['./#{name}'] = new function() {
|
|
|
|
var exports = this;
|
2011-07-05 07:58:04 -04:00
|
|
|
#{fs.readFileSync "lib/coffee-script/#{name}.js"}
|
2010-11-10 23:06:26 -05:00
|
|
|
};
|
|
|
|
"""
|
2011-01-15 15:20:01 -05:00
|
|
|
code = """
|
2011-11-05 06:04:53 -04:00
|
|
|
(function(root) {
|
|
|
|
var CoffeeScript = function() {
|
|
|
|
function require(path){ return require[path]; }
|
|
|
|
#{code}
|
2011-12-15 11:21:38 -05:00
|
|
|
return require['./coffee-script'];
|
2011-11-05 06:04:53 -04:00
|
|
|
}();
|
|
|
|
|
2011-12-15 11:21:38 -05:00
|
|
|
if (typeof define === 'function' && define.amd) {
|
|
|
|
define(function() { return CoffeeScript; });
|
2012-11-29 12:43:18 -05:00
|
|
|
} else {
|
|
|
|
root.CoffeeScript = CoffeeScript;
|
2011-12-15 11:21:38 -05:00
|
|
|
}
|
2011-11-05 06:04:53 -04:00
|
|
|
}(this));
|
2010-11-10 23:06:26 -05:00
|
|
|
"""
|
2011-01-15 15:20:01 -05:00
|
|
|
unless process.env.MINIFY is 'false'
|
2012-11-29 12:43:18 -05:00
|
|
|
{code} = require('uglify-js').minify code, fromString: true
|
2010-11-10 23:06:26 -05:00
|
|
|
fs.writeFileSync 'extras/coffee-script.js', header + '\n' + code
|
2011-01-15 15:12:47 -05:00
|
|
|
console.log "built ... running browser tests:"
|
2010-11-10 23:06:26 -05:00
|
|
|
invoke 'test:browser'
|
2010-02-24 19:57:29 -05:00
|
|
|
|
|
|
|
|
2010-03-06 19:02:31 -05:00
|
|
|
task 'doc:site', 'watch and continually rebuild the documentation for the website', ->
|
2010-10-24 21:50:34 -04:00
|
|
|
exec 'rake doc', (err) ->
|
|
|
|
throw err if err
|
2010-02-24 19:57:29 -05:00
|
|
|
|
|
|
|
|
2010-03-06 19:02:31 -05:00
|
|
|
task 'doc:source', 'rebuild the internal documentation', ->
|
2011-08-05 18:39:55 -04:00
|
|
|
exec 'docco src/*.coffee && cp -rf docs documentation && rm -r docs', (err) ->
|
2010-03-06 20:30:40 -05:00
|
|
|
throw err if err
|
2010-03-06 19:02:31 -05:00
|
|
|
|
|
|
|
|
2010-03-07 19:07:37 -05:00
|
|
|
task 'doc:underscore', 'rebuild the Underscore.coffee documentation page', ->
|
2010-03-16 19:13:13 -04:00
|
|
|
exec 'docco examples/underscore.coffee && cp -rf docs documentation && rm -r docs', (err) ->
|
|
|
|
throw err if err
|
2010-03-07 19:07:37 -05:00
|
|
|
|
2011-10-07 00:39:24 -04:00
|
|
|
task 'bench', 'quick benchmark of compilation time', ->
|
2011-07-05 07:58:04 -04:00
|
|
|
{Rewriter} = require './lib/coffee-script/rewriter'
|
2013-02-01 20:36:05 -05:00
|
|
|
sources = ['coffee-script', 'grammar', 'helpers', 'lexer', 'nodes', 'rewriter']
|
|
|
|
coffee = sources.map((name) -> fs.readFileSync "src/#{name}.coffee").join '\n'
|
|
|
|
litcoffee = fs.readFileSync("src/scope.litcoffee").toString()
|
2010-11-20 20:28:45 -05:00
|
|
|
fmt = (ms) -> " #{bold}#{ " #{ms}".slice -4 }#{reset} ms"
|
2010-11-18 07:33:56 -05:00
|
|
|
total = 0
|
2010-11-17 11:34:23 -05:00
|
|
|
now = Date.now()
|
2010-11-18 07:33:56 -05:00
|
|
|
time = -> total += ms = -(now - now = Date.now()); fmt ms
|
2013-02-01 20:36:05 -05:00
|
|
|
tokens = CoffeeScript.tokens coffee, rewrite: no
|
|
|
|
littokens = CoffeeScript.tokens litcoffee, rewrite: no, literate: yes
|
|
|
|
tokens = tokens.concat(littokens)
|
2010-11-18 07:33:56 -05:00
|
|
|
console.log "Lex #{time()} (#{tokens.length} tokens)"
|
2010-11-17 11:34:23 -05:00
|
|
|
tokens = new Rewriter().rewrite tokens
|
2010-11-18 07:33:56 -05:00
|
|
|
console.log "Rewrite#{time()} (#{tokens.length} tokens)"
|
2010-11-17 11:34:23 -05:00
|
|
|
nodes = CoffeeScript.nodes tokens
|
2010-11-18 07:33:56 -05:00
|
|
|
console.log "Parse #{time()}"
|
2013-02-01 20:36:05 -05:00
|
|
|
js = nodes.compile bare: yes
|
2010-11-18 07:33:56 -05:00
|
|
|
console.log "Compile#{time()} (#{js.length} chars)"
|
2010-11-20 20:28:45 -05:00
|
|
|
console.log "total #{ fmt total }"
|
2010-03-07 19:07:37 -05:00
|
|
|
|
2010-06-11 18:47:48 -04:00
|
|
|
|
2010-12-11 20:30:48 -05:00
|
|
|
# Run the CoffeeScript test suite.
|
2011-10-07 00:39:24 -04:00
|
|
|
runTests = (CoffeeScript) ->
|
2010-12-11 20:30:48 -05:00
|
|
|
startTime = Date.now()
|
|
|
|
currentFile = null
|
2010-12-09 23:59:50 -05:00
|
|
|
passedTests = 0
|
2010-12-11 20:30:48 -05:00
|
|
|
failures = []
|
2010-11-28 12:27:06 -05:00
|
|
|
|
2011-11-05 11:12:50 -04:00
|
|
|
global[name] = func for name, func of require 'assert'
|
2010-12-21 15:45:46 -05:00
|
|
|
|
2010-12-11 20:30:48 -05:00
|
|
|
# Convenience aliases.
|
2010-09-27 01:17:05 -04:00
|
|
|
global.CoffeeScript = CoffeeScript
|
2013-01-21 13:02:04 -05:00
|
|
|
global.Repl = require './lib/coffee-script/repl'
|
2010-12-11 20:30:48 -05:00
|
|
|
|
|
|
|
# Our test helper function for delimiting different test cases.
|
2010-12-09 23:59:50 -05:00
|
|
|
global.test = (description, fn) ->
|
2010-12-11 20:30:48 -05:00
|
|
|
try
|
2011-01-03 04:17:00 -05:00
|
|
|
fn.test = {description, currentFile}
|
|
|
|
fn.call(fn)
|
2011-11-05 11:12:50 -04:00
|
|
|
++passedTests
|
2010-12-09 23:59:50 -05:00
|
|
|
catch e
|
2012-11-14 16:19:17 -05:00
|
|
|
failures.push
|
|
|
|
filename: currentFile
|
|
|
|
error: e
|
|
|
|
description: description if description?
|
|
|
|
source: fn.toString() if fn.toString?
|
2010-11-28 12:27:06 -05:00
|
|
|
|
2010-12-18 14:48:43 -05:00
|
|
|
# See http://wiki.ecmascript.org/doku.php?id=harmony:egal
|
2011-11-05 11:14:09 -04:00
|
|
|
egal = (a, b) ->
|
2010-12-16 00:12:11 -05:00
|
|
|
if a is b
|
|
|
|
a isnt 0 or 1/a is 1/b
|
2011-11-05 11:14:09 -04:00
|
|
|
else
|
|
|
|
a isnt a and b isnt b
|
|
|
|
|
|
|
|
# A recursive functional equivalence helper; uses egal for testing equivalence.
|
|
|
|
arrayEgal = (a, b) ->
|
|
|
|
if egal a, b then yes
|
2010-12-16 00:12:11 -05:00
|
|
|
else if a instanceof Array and b instanceof Array
|
|
|
|
return no unless a.length is b.length
|
2011-11-05 11:14:09 -04:00
|
|
|
return no for el, idx in a when not arrayEgal el, b[idx]
|
2010-12-16 00:12:11 -05:00
|
|
|
yes
|
|
|
|
|
2013-01-26 03:29:16 -05:00
|
|
|
global.eq = (a, b, msg) -> ok egal(a, b), msg ? "Expected #{a} to equal #{b}"
|
|
|
|
global.arrayEq = (a, b, msg) -> ok arrayEgal(a,b), msg ? "Expected #{a} to deep equal #{b}"
|
2010-12-16 00:12:11 -05:00
|
|
|
|
2011-08-14 16:39:38 -04:00
|
|
|
# When all the tests have run, collect and print errors.
|
|
|
|
# If a stacktrace is available, output the compiled function source.
|
|
|
|
process.on 'exit', ->
|
|
|
|
time = ((Date.now() - startTime) / 1000).toFixed(2)
|
2011-10-07 00:39:24 -04:00
|
|
|
message = "passed #{passedTests} tests in #{time} seconds#{reset}"
|
2011-08-14 16:39:38 -04:00
|
|
|
return log(message, green) unless failures.length
|
|
|
|
log "failed #{failures.length} and #{message}", red
|
|
|
|
for fail in failures
|
2012-11-14 16:19:17 -05:00
|
|
|
{error, filename, description, source} = fail
|
2011-08-14 16:39:38 -04:00
|
|
|
jsFilename = filename.replace(/\.coffee$/,'.js')
|
|
|
|
match = error.stack?.match(new RegExp(fail.file+":(\\d+):(\\d+)"))
|
|
|
|
match = error.stack?.match(/on line (\d+):/) unless match
|
|
|
|
[match, line, col] = match if match
|
|
|
|
console.log ''
|
2012-11-14 16:19:17 -05:00
|
|
|
log " #{description}", red if description
|
2011-08-14 16:39:38 -04:00
|
|
|
log " #{error.stack}", red
|
|
|
|
log " #{jsFilename}: line #{line ? 'unknown'}, column #{col ? 'unknown'}", red
|
2012-11-14 16:19:17 -05:00
|
|
|
console.log " #{source}" if source
|
2011-08-14 16:39:38 -04:00
|
|
|
return
|
|
|
|
|
2010-12-11 20:30:48 -05:00
|
|
|
# Run every test in the `test` folder, recording failures.
|
2011-07-18 17:47:31 -04:00
|
|
|
files = fs.readdirSync 'test'
|
2012-09-25 20:35:02 -04:00
|
|
|
for file in files when file.match /\.(lit)?coffee$/i
|
|
|
|
literate = path.extname(file) is '.litcoffee'
|
2011-07-18 17:47:31 -04:00
|
|
|
currentFile = filename = path.join 'test', file
|
|
|
|
code = fs.readFileSync filename
|
|
|
|
try
|
2012-09-25 20:35:02 -04:00
|
|
|
CoffeeScript.run code.toString(), {filename, literate}
|
2011-07-18 17:47:31 -04:00
|
|
|
catch error
|
|
|
|
failures.push {filename, error}
|
|
|
|
return !failures.length
|
2010-10-11 10:42:13 -04:00
|
|
|
|
|
|
|
|
2011-10-07 00:39:24 -04:00
|
|
|
task 'test', 'run the CoffeeScript language test suite', ->
|
|
|
|
runTests CoffeeScript
|
2010-10-11 10:42:13 -04:00
|
|
|
|
|
|
|
|
2011-10-07 00:39:24 -04:00
|
|
|
task 'test:browser', 'run the test suite against the merged browser script', ->
|
2010-10-11 10:42:13 -04:00
|
|
|
source = fs.readFileSync 'extras/coffee-script.js', 'utf-8'
|
2010-10-11 13:27:05 -04:00
|
|
|
result = {}
|
2010-12-12 21:41:04 -05:00
|
|
|
global.testingBrowser = yes
|
2010-10-11 13:27:05 -04:00
|
|
|
(-> eval source).call result
|
2011-10-07 00:39:24 -04:00
|
|
|
runTests result.CoffeeScript
|