1
0
Преглед на файлове

Using amberc from amber-dev

Herbert Vojčík преди 11 години
родител
ревизия
34c0258ab1
променени са 5 файла, в които са добавени 6 реда и са изтрити 728 реда
  1. 1 1
      Gruntfile.js
  2. 1 1
      cli/support/amberc-cli.js
  3. 0 587
      cli/support/amberc.js
  4. 0 137
      grunt/tasks/grunt-amberc.js
  5. 4 2
      package.json

+ 1 - 1
Gruntfile.js

@@ -1,7 +1,7 @@
 module.exports = function(grunt) {
   var path = require('path');
 
-  grunt.loadTasks('./grunt/tasks');
+  grunt.loadNpmTasks('amber-dev');
 
   grunt.loadNpmTasks('grunt-contrib-jshint');
 

+ 1 - 1
cli/support/amberc-cli.js

@@ -1,7 +1,7 @@
 #!/usr/bin/env node
 
 var path = require('path');
-var amberc = require('./amberc.js');
+var amberc = require('amber-dev/lib/amberc');
 
 // get parameters passed to the command line script
 // discard the first two parameters which are the node binary and the script name

+ 0 - 587
cli/support/amberc.js

@@ -1,587 +0,0 @@
-/**
- * This is a "compiler" for Amber code.
- * Put the following code into compiler.js:
- *     var amberc = require('amberc');
- *     var compiler = new amberc.Compiler('path/to/amber');
- *     var options = amberc.createDefaults();
- *     // edit options entries
- *     compiler.main(options);
- */
-
-/**
- * Helper for concatenating Amber generated AMD modules.
- * The produced output can be exported and run as an independent program.
- *
- * var concatenator = createConcatenator();
- * concatenator.start(); // write the required AMD define header
- * concatenator.add(module1);
- * concatenator.addId(module1_ID);
- * //...
- * concatenator.finish("//some last code");
- * var concatenation = concatenator.toString();
- * // The variable concatenation contains the concatenated result
- * // which can either be stored in a file or interpreted with eval().
- */
-function createConcatenator () {
-	return {
-		elements: [],
-		ids: [],
-		add: function () {
-			this.elements.push.apply(this.elements, arguments);
-		},
-		addId: function () {
-			this.ids.push.apply(this.ids, arguments);
-		},
-		forEach: function () {
-			this.elements.forEach.apply(this.elements, arguments);
-		},
-		start: function () {
-			this.add(
-				'var define = (' + require('amdefine') + ')(null, function (id) { throw new Error("Dependency not found: " +  id); }), requirejs = define.require;',
-				'define("amber_vm/browser-compatibility", [], {});'
-			);
-		},
-		finish: function (realWork) {
-			this.add(
-				'define("amber_vm/_init", ["amber_vm/smalltalk", "amber_vm/globals", "' + this.ids.join('","') + '"], function (vm, globals) {',
-				'vm.initialize();',
-				realWork,
-				'});',
-				'requirejs("amber_vm/_init");'
-			);
-		},
-		toString: function () {
-			return this.elements.join('\n');
-		}
-	};
-}
-
-
-var path = require('path'),
-	fs = require('fs'),
-	Promise = require('es6-promise').Promise;
-
-/**
- * AmberCompiler constructor function.
- * amber_dir: points to the location of an amber installation
- */
-function AmberCompiler(amber_dir) {
-	if (undefined === amber_dir || !fs.existsSync(amber_dir)) {
-		throw new Error('amber_dir needs to be a valid directory');
-	}
-
-	this.amber_dir = amber_dir;
-	this.kernel_libraries = ['boot', 'smalltalk', 'globals', 'nil', '_st', 'Kernel-Objects', 'Kernel-Classes', 'Kernel-Methods',
-							'Kernel-Collections', 'Kernel-Infrastructure', 'Kernel-Exceptions', 'Kernel-Transcript',
-							'Kernel-Announcements'];
-	this.compiler_libraries = this.kernel_libraries.concat(['parser', 'Kernel-ImportExport', 'Compiler-Exceptions',
-							'Compiler-Core', 'Compiler-AST', 'Compiler-Exceptions', 'Compiler-IR', 'Compiler-Inlining', 'Compiler-Semantic']);
-}
-
-
-/**
- * Default values.
- */
-var createDefaultConfiguration = function() {
-	return {
-		'load': [],
-		'main': undefined,
-		'mainfile': undefined,
-		'stFiles': [],
-		'jsFiles': [],
-		'jsGlobals': [],
-		'amd_namespace': 'amber_core',
-		'suffix': '',
-		'loadsuffix': '',
-		'suffix_used': '',
-		'libraries': [],
-		'jsLibraryDirs': [],
-		'compile': [],
-		'compiled': [],
-		'program': undefined,
-		'output_dir': undefined,
-		'verbose': false
-	};
-};
-
-
-/**
- * Main function for executing the compiler.
- * If check_configuration_ok() returns successfully
- * the configuration is used to trigger the following compilation steps.
- */
-AmberCompiler.prototype.main = function(configuration, finished_callback) {
-	console.time('Compile Time');
-
-	if (configuration.amd_namespace.length === 0) {
-		configuration.amd_namespace = 'amber_core';
-	}
-
-	if (undefined !== configuration.jsLibraryDirs) {
-		configuration.jsLibraryDirs.push(path.join(this.amber_dir, 'src'));
-		configuration.jsLibraryDirs.push(path.join(this.amber_dir, 'support'));
-	}
-
-	console.ambercLog = console.log;
-	if (false === configuration.verbose) {
-		console.log = function() {};
-	}
-
-	// the evaluated compiler will be stored in this variable (see create_compiler)
-	configuration.vm = {};
-	configuration.globals = {};
-	configuration.kernel_libraries = this.kernel_libraries;
-	configuration.compiler_libraries = this.compiler_libraries;
-	configuration.amber_dir = this.amber_dir;
-
-	check_configuration(configuration)
-	.then(collect_st_files)
-	.then(collect_js_files)
-	.then(resolve_kernel)
-	.then(create_compiler)
-	.then(compile)
-	.then(category_export)
-	.then(verify)
-	.then(compose_js_files)
-	.then(function () {
-		console.timeEnd('Compile Time');
-	}, function(error) {
-		console.error(error);
-	})
-	.then(function () {
-		console.log = console.ambercLog;
-		finished_callback && finished_callback();
-	});
-};
-
-
-/**
- * Check if the passed in configuration object has sufficient/nonconflicting values.
- * Returns a Promise which resolves into the configuration object.
- */
-function check_configuration(configuration) {
-	return new Promise(function(resolve, reject) {
-		if (undefined === configuration) {
-			reject(Error('AmberCompiler.check_configuration_ok(): missing configuration object'));
-		}
-
-		if (0 === configuration.jsFiles.length && 0 === configuration.stFiles.length) {
-			reject(Error('AmberCompiler.check_configuration_ok(): no files to compile/link specified in configuration object'));
-		}
-
-		resolve(configuration);
-	});
-};
-
-
-/**
- * Check if the file given as parameter exists in any of the following directories:
- *  1. current local directory
- *  2. configuration.jsLibraryDirs
- *  3. $AMBER/src/
- *  3. $AMBER/support/
- *
- * @param filename name of a file without '.js' prefix
- * @param configuration the main amberc configuration object
- */
-function resolve_js(filename, configuration) {
-	var baseName = path.basename(filename, '.js');
-	var jsFile = baseName + configuration.loadsuffix + '.js';
-	return resolve_file(jsFile, configuration.jsLibraryDirs);
-};
-
-
-/**
- * Check if the file given as parameter exists in any of the following directories:
- *  1. current local directory
- *  2. $AMBER/
- *
- * @param filename name of a .st file
- * @param configuration the main amberc configuration object
- */
-function resolve_st(filename, configuration) {
-	return resolve_file(filename, [configuration.amber_dir]);
-};
-
-
-/**
- * Resolve the location of a file given as parameter filename.
- * First check if the file exists at given location,
- * then check in each of the directories specified in parameter searchDirectories.
- */
-function resolve_file(filename, searchDirectories) {
-	return new Promise(function(resolve, reject) {
-		console.log('Resolving: ' + filename);
-		fs.exists(filename, function(exists) {
-			if (exists) {
-				resolve(filename);
-			} else {
-				var alternativeFile = '';
-				// check for filename in any of the given searchDirectories
-				var found = searchDirectories.some(function(directory) {
-					alternativeFile = path.join(directory, filename);
-					return fs.existsSync(alternativeFile);
-				});
-				if (found) {
-					resolve(alternativeFile);
-				} else {
-					reject(Error('File not found: ' + alternativeFile));
-				}
-			}
-		});
-	});
-};
-
-
-/**
- * Resolve st files given by stFiles and add them to configuration.compile.
- * Returns a Promise which resolves into the configuration object.
- */
-function collect_st_files(configuration) {
-	return Promise.all(
-		configuration.stFiles.map(function(stFile) {
-			return resolve_st(stFile, configuration);
-		})
-	)
-	.then(function(data) {
-		configuration.compile = configuration.compile.concat(data);
-		return configuration;
-	});
-}
-
-
-/**
- * Resolve js files given by jsFiles and add them to configuration.libraries.
- * Returns a Promise which resolves into the configuration object.
- */
-function collect_js_files(configuration) {
-	return Promise.all(
-		configuration.jsFiles.map(function(file) {
-			return resolve_js(file, configuration);
-		})
-	)
-	.then(function(data) {
-		configuration.libraries = configuration.libraries.concat(data);
-		return configuration;
-	});
-}
-
-
-/**
- * Resolve .js files needed by kernel.
- * Returns a Promise which resolves into the configuration object.
- */
-function resolve_kernel(configuration) {
-	var kernel_files = configuration.kernel_libraries.concat(configuration.load);
-	return Promise.all(
-		kernel_files.map(function(file) {
-			return resolve_js(file, configuration);
-		})
-	)
-	.then(function(data) {
-		// boot.js and Kernel files need to be used first
-		// otherwise the global objects 'vm' and 'globals' are undefined
-		configuration.libraries = data.concat(configuration.libraries);
-		return configuration;
-	});
-}
-
-
-/**
- * Resolve .js files needed by compiler, read and eval() them.
- * The finished Compiler gets stored in configuration.{vm,globals}.
- * Returns a Promise object which resolves into the configuration object.
- */
-function create_compiler(configuration) {
-	var compiler_files = configuration.compiler_libraries;
-	var include_files = configuration.load;
-	var builder;
-	return Promise.all(
-		compiler_files.map(function(file) {
-			return resolve_js(file, configuration);
-		})
-	)
-	.then(function(compilerFilesArray) {
-		return Promise.all(
-			compilerFilesArray.map(function(file) {
-				return new Promise(function(resolve, reject) {
-					console.log('Loading file: ' + file);
-					fs.readFile(file, function(err, data) {
-						if (err)
-							reject(err);
-						else
-							resolve(data);
-					});
-				});
-			})
-		)
-	})
-	.then(function(files) {
-		builder = createConcatenator();
-		builder.add('(function() {');
-		builder.start();
-
-		files.forEach(function(data) {
-			// data is an array where index 0 is the error code and index 1 contains the data
-			builder.add(data);
-			// matches and returns the "module_id" string in the AMD definition: define("module_id", ...)
-			var match = ('' + data).match(/^define\("([^"]*)"/);
-			if (match) {
-				builder.addId(match[1]);
-			}
-		});
-	})
-	.then(function () { return Promise.all(
-		include_files.map(function(file) {
-			return resolve_js(file, configuration);
-		})
-	); })
-	.then(function(includeFilesArray) {
-		return Promise.all(
-			includeFilesArray.map(function(file) {
-				return new Promise(function(resolve, reject) {
-					console.log('Loading library file: ' + file);
-					fs.readFile(file, function(err, data) {
-						if (err)
-							reject(err);
-						else
-							resolve(data);
-					});
-				});
-			})
-		)
-	})
-	.then(function(files) {
-		var loadIds = [];
-		files.forEach(function(data) {
-			// data is an array where index 0 is the error code and index 1 contains the data
-			builder.add(data);
-			// matches and returns the "module_id" string in the AMD definition: define("module_id", ...)
-			var match = ('' + data).match(/^define\("([^"]*)"/);
-			if (match) {
-				loadIds.push(match[1]);
-			}
-		});
-		// store the generated smalltalk env in configuration.{vm,globals}
-		builder.finish('configuration.vm = vm; configuration.globals = globals;');
-		loadIds.forEach(function (id) {
-			builder.add('requirejs("' + id + '");');
-		});
-		builder.add('})();');
-
-		eval(builder.toString());
-
-		console.log('Compiler loaded');
-
-		configuration.globals.ErrorHandler._register_(configuration.globals.RethrowErrorHandler._new());
-
-		if(0 !== configuration.jsGlobals.length) {
-			var jsGlobalVariables = configuration.vm.globalJsVariables;
-			jsGlobalVariables.push.apply(jsGlobalVariables, configuration.jsGlobals);
-		}
-
-		return configuration;
-	});
-}
-
-
-/**
- * Compile all given .st files by importing them.
- * Returns a Promise object that resolves into the configuration object.
- */
-function compile(configuration) {
-	// return function which does the actual work
-	// and use the compile function to reference the configuration object
-	return Promise.all(
-		configuration.compile.map(function(stFile) {
-			return new Promise(function(resolve, reject) {
-				if (/\.st/.test(stFile)) {
-					console.ambercLog('Reading: ' + stFile);
-					fs.readFile(stFile, 'utf8', function(err, data) {
-						if (!err)
-							resolve(data);
-						else
-							reject(Error('Could not read: ' + stFile));
-					});
-				}
-			});
-		})
-	)
-	.then(function(fileContents) {
-		console.log('Compiling collected .st files');
-		// import/compile content of .st files
-		return Promise.all(
-			fileContents.map(function(code) {
-				return new Promise(function(resolve, reject) {
-					var importer = configuration.globals.Importer._new();
-					try {
-						importer._import_(code._stream());
-						resolve(true);
-					} catch (ex) {
-						reject(Error("Compiler error in section:\n" +
-							importer._lastSection() + "\n\n" +
-							"while processing chunk:\n" +
-							importer._lastChunk() + "\n\n" +
-							(ex._messageText && ex._messageText() || ex.message || ex))
-						);
-					}
-				});
-			})
-		);
-	})
-	.then(function () {
-		return configuration;
-	});
-}
-
-
-/**
- * Export compiled categories to JavaScript files.
- * Returns a Promise() that resolves into the configuration object.
- */
-function category_export(configuration) {
-	return Promise.all(
-		configuration.compile.map(function(stFile) {
-			return new Promise(function(resolve, reject) {
-				var category = path.basename(stFile, '.st');
-				var jsFilePath = configuration.output_dir;
-				if (undefined === jsFilePath) {
-					jsFilePath = path.dirname(stFile);
-				}
-				var jsFile = category + configuration.suffix_used + '.js';
-				jsFile = path.join(jsFilePath, jsFile);
-				configuration.compiled.push(jsFile);
-				var smalltalkGlobals = configuration.globals;
-				var packageObject = smalltalkGlobals.Package._named_(category);
-				packageObject._transport()._namespace_(configuration.amd_namespace);
-				fs.writeFile(jsFile, smalltalkGlobals.String._streamContents_(function (stream) {
-					smalltalkGlobals.AmdExporter._new()._exportPackage_on_(packageObject, stream);
-				}), function(err) {
-					if (err)
-						reject(err);
-					else
-						resolve(true);
-				});
-			});
-		})
-	)
-	.then(function() {
-		return configuration;
-	});
-}
-
-
-/**
- * Verify if all .st files have been compiled.
- * Returns a Promise() that resolves into the configuration object.
- */
-function verify(configuration) {
-	console.log('Verifying if all .st files were compiled');
-	return Promise.all(
-		configuration.compiled.map(function(file) {
-			return new Promise(function(resolve, reject) {
-				fs.exists(file, function(exists) {
-					if (exists)
-						resolve(true);
-					else
-						reject(Error('Compilation failed of: ' + file));
-				});
-			});
-		})
-	)
-	.then(function() {
-		return configuration;
-	});
-}
-
-
-/**
- * Synchronous function.
- * Concatenates compiled JavaScript files into one file in the correct order.
- * The name of the produced file is given by configuration.program.
- * Returns a Promise which resolves into the configuration object.
- */
-function compose_js_files(configuration) {
-	return new Promise(function(resolve, reject) {
-		var programFile = configuration.program;
-		if (undefined === programFile) {
-			resolve(configuration);
-			return;
-		}
-		if (undefined !== configuration.output_dir) {
-			programFile = path.join(configuration.output_dir, programFile);
-		}
-
-		var program_files = [];
-		if (0 !== configuration.libraries.length) {
-			console.log('Collecting libraries: ' + configuration.libraries);
-			program_files.push.apply(program_files, configuration.libraries);
-		}
-
-		if (0 !== configuration.compiled.length) {
-			var compiledFiles = configuration.compiled.slice(0);
-
-			console.log('Collecting compiled files: ' + compiledFiles);
-			program_files.push.apply(program_files, compiledFiles);
-		}
-
-		console.ambercLog('Writing program file: %s.js', programFile);
-
-		var fileStream = fs.createWriteStream(programFile + configuration.suffix_used + '.js');
-		fileStream.on('error', function(error) {
-			fileStream.end();
-			console.ambercLog(error);
-			reject(error);
-		});
-
-		fileStream.on('close', function(){
-			resolve(configuration);
-		});
-
-		var builder = createConcatenator();
-		builder.add('#!/usr/bin/env node');
-		builder.start();
-
-		program_files.forEach(function(file) {
-			if(fs.existsSync(file)) {
-				console.log('Adding : ' + file);
-				var buffer = fs.readFileSync(file);
-				// matches and returns the "module_id" string in the AMD define: define("module_id", ...)
-				var match = buffer.toString().match(/^define\("([^"]*)"/);
-				if (match /*&& match[1].slice(0,9) !== "amber_vm/"*/) {
-					builder.addId(match[1]);
-				}
-				builder.add(buffer);
-			} else {
-				fileStream.end();
-				reject(Error('Can not find file ' + file));
-			}
-		});
-
-		var mainFunctionOrFile = '';
-
-		if (undefined !== configuration.main) {
-			console.log('Adding call to: %s>>main', configuration.main);
-			mainFunctionOrFile += 'globals.' + configuration.main + '._main();';
-		}
-
-		if (undefined !== configuration.mainfile && fs.existsSync(configuration.mainfile)) {
-			console.log('Adding main file: ' + configuration.mainfile);
-			mainFunctionOrFile += '\nvar smalltalk = vm; // backward compatibility\n' + fs.readFileSync(configuration.mainfile);
-		}
-
-		builder.finish(mainFunctionOrFile);
-
-		console.log('Writing...');
-		builder.forEach(function (element) {
-			fileStream.write(element);
-			fileStream.write('\n');
-		});
-		console.log('Done.');
-		fileStream.end();
-	});
-}
-
-
-module.exports.Compiler = AmberCompiler;
-module.exports.createDefaultConfiguration = createDefaultConfiguration;

+ 0 - 137
grunt/tasks/grunt-amberc.js

@@ -1,137 +0,0 @@
-module.exports = function(grunt) {
-
-  var path = require('path');
-  var fs = require('fs');
-  var amberc = require('../../cli/support/amberc.js');
-
-  /**
-     A full example entry for a Gruntfile.js is available below.
-     Please note that the verbose level is either specified globally
-     or on a target specific level.
-     However, it can additionally be triggered on the commandline by
-     adding the '-v' or '--verbose' flag.
-
-     Example Gruntfile.js entry:
-
-     amberc: {
-       options: {
-         amber_dir: process.cwd(),                // REQUIRED
-         library_dirs: ['dir1', '/usr/local/js'], // optional
-         verbose: true                            // optional
-       },
-       helloWorld: {
-         // this 'options' object is optional as well as all parameters inside it
-         // they can be used to override the global 'options'
-         options: {
-           library_dirs: ['dir1', '/usr/local/js'], // optional
-           verbose: true
-         },
-         src: ['projects/HelloWorld/src/HelloWorld.st'], // REQUIRED
-         output_dir: 'projects/HelloWorld/src',  // optional
-         libraries: 'Web',                       // optional
-         jsGlobals: ['global1', 'global2'],     // optional
-         main_class: 'HelloWorld',              // optional
-         output_name: 'helloWorld',             // optional
-         amd_namespace: 'MyNamespace',          // optional (default: 'amber')
-         main_file: 'myMain.js',                // optional
-         output_suffix: 'mySuffix',             // optional
-         library_suffix: '-0.9'                 // optional
-       },
-     },
-
-   */
-  grunt.registerMultiTask('amberc', 'Compile Smalltalk files with the amberc compiler', function() {
-    // mark task as async task
-    var done = this.async();
-
-    var options = this.options({
-      amber_dir: undefined,
-      library_dirs: [],
-      verbose: grunt.option('verbose') || false
-    });
-    this.data.verbose = options.verbose;
-    this.data.library_dirs = options.library_dirs;
-
-    // mark required properties
-    this.requiresConfig('amberc.options.amber_dir');
-    // raise error on missing source files
-    if (this.filesSrc.length === 0) {
-        grunt.fail.fatal('No source files to compile or link.');
-    }
-
-    // create and initialize amberc
-    var compiler = new amberc.Compiler(grunt.config('amberc.options.amber_dir'));
-
-    // generate the amberc configuration out of the given target properties
-    var configuration = generateCompilerConfiguration(this.data, this.filesSrc);
-
-    // run the compiler and call the async callback once finished
-    var self = this;
-    compiler.main(configuration, function(){
-      // signal that task has finished
-      done();
-    });
-  });
-
-
-  function generateCompilerConfiguration(data, sourceFiles) {
-    var configuration = amberc.createDefaultConfiguration();
-    var parameters = [];
-
-    var libraries = data.libraries;
-    if (undefined !== libraries) {
-      configuration.load = libraries;
-    }
-    var library_dirs = data.library_dirs;
-    if (undefined !== library_dirs) {
-      configuration.jsLibraryDirs = library_dirs;
-    }
-    var mainClass = data.main_class;
-    if (undefined !== mainClass) {
-      configuration.main = mainClass;
-    }
-    var mainFile = data.main_file;
-    if (undefined !== mainFile) {
-      configuration.mainfile = mainFile;
-    }
-    var outputSuffix = data.output_suffix;
-    if (undefined !== outputSuffix) {
-      configuration.suffix = outputSuffix;
-      configuration.suffix_used = outputSuffix;
-    }
-    var librarySuffix = data.library_suffix;
-    if (undefined !== librarySuffix) {
-      configuration.loadsuffix = librarySuffix;
-      configuration.suffix_used = librarySuffix;
-    }
-    if (undefined !== sourceFiles) {
-      sourceFiles.forEach(function(currentItem){
-        var fileSuffix = path.extname(currentItem);
-        switch (fileSuffix) {
-          case '.st':
-            configuration.stFiles.push(currentItem);
-          break;
-          case '.js':
-            configuration.jsFiles.push(currentItem);
-          break;
-        }
-      });
-    }
-    var outputName = data.output_name;
-    if (undefined !== outputName) {
-      configuration.program = outputName;
-    }
-    var amdNamespace = data.amd_namespace;
-    if (undefined !== amdNamespace) {
-      configuration.amd_namespace = amdNamespace;
-    }
-    if (undefined !== data.output_dir) {
-      configuration.output_dir = data.output_dir;
-    }
-    if (undefined !== data.jsGlobals) {
-      configuration.jsGlobals.push.apply(configuration.jsGlobals, data.jsGlobals);
-    }
-    configuration.verbose = data.verbose;
-    return configuration;
-  }
-};

+ 4 - 2
package.json

@@ -38,10 +38,12 @@
     "amber": "~0.12.4",
     "grunt-init-amber": "git://github.com/amber-smalltalk/grunt-init-amber.git",
     "grunt-init": "~0.3.1",
-    "bower": "~1.3.2"
+    "bower": "~1.3.2",
+    "amber-dev": "git://github.com/amber-smalltalk/amber-dev.git"
   },
   "devDependencies": {
     "grunt": "~0.4.0",
-    "grunt-contrib-jshint": "~0.3.0"
+    "grunt-contrib-jshint": "~0.3.0",
+    "amber-dev": "git://github.com/amber-smalltalk/amber-dev.git"
   }
 }