|
@@ -0,0 +1,900 @@
|
|
|
+Smalltalk current createPackage: 'AmberCli'!
|
|
|
+Object subclass: #AmberCli
|
|
|
+ instanceVariableNames: ''
|
|
|
+ package: 'AmberCli'!
|
|
|
+
|
|
|
+!AmberCli class methodsFor: 'commandline'!
|
|
|
+
|
|
|
+commandLineSwitches
|
|
|
+ "Collect all methodnames from the 'commands' protocol of the class
|
|
|
+ and select the ones with only one parameter.
|
|
|
+ Then remove the ':' at the end of the name.
|
|
|
+ Additionally all uppercase letters are made lowercase and preceded by a '-'.
|
|
|
+ Example: fallbackPage: becomes --fallback-page.
|
|
|
+ Return the Array containing the commandline switches."
|
|
|
+ | switches |
|
|
|
+ switches := ((self class methodsInProtocol: 'commands') collect: [ :each | each selector]).
|
|
|
+ switches := switches select: [ :each | each match: '^[^:]*:$'].
|
|
|
+ switches :=switches collect: [ :each |
|
|
|
+ (each allButLast replace: '([A-Z])' with: '-$1') asLowercase].
|
|
|
+ ^switches
|
|
|
+!
|
|
|
+
|
|
|
+handleArguments: args
|
|
|
+ | command |
|
|
|
+ command := self selectorForCommandLineSwitch: (args first).
|
|
|
+ args remove: args first.
|
|
|
+ self perform: command withArguments: (Array with: args).
|
|
|
+!
|
|
|
+
|
|
|
+selectorForCommandLineSwitch: aSwitch
|
|
|
+ "Add ':' at the end
|
|
|
+ and replace all occurences of a lowercase letter preceded by a '-' with
|
|
|
+ the Uppercase letter.
|
|
|
+ Example: fallback-page becomes fallbackPage:"
|
|
|
+ ^(aSwitch replace: '-[a-z]' with: [ :each | each second asUppercase ]), ':'
|
|
|
+! !
|
|
|
+
|
|
|
+!AmberCli class methodsFor: 'commands'!
|
|
|
+
|
|
|
+help: args
|
|
|
+ console log: 'Available Commands:'.
|
|
|
+ self commandLineSwitches do: [ :each | console log: each ]
|
|
|
+!
|
|
|
+
|
|
|
+repl: args
|
|
|
+ ^Repl new createInterface
|
|
|
+!
|
|
|
+
|
|
|
+server: args
|
|
|
+ ^(FileServer createServerWithArguments: args) start
|
|
|
+! !
|
|
|
+
|
|
|
+!AmberCli class methodsFor: 'startup'!
|
|
|
+
|
|
|
+main
|
|
|
+ "Main entry point for Amber applications.
|
|
|
+ Parses commandline arguments and starts the according subprogram."
|
|
|
+ | args |
|
|
|
+ args := process argv.
|
|
|
+ "Remove the first args which contain the path to the node executable and the script file."
|
|
|
+ args removeFrom: 1 to: 3.
|
|
|
+
|
|
|
+ (args isEmpty)
|
|
|
+ ifTrue: [self help: nil]
|
|
|
+ ifFalse: [^self handleArguments: args]
|
|
|
+! !
|
|
|
+
|
|
|
+Object subclass: #FileServer
|
|
|
+ instanceVariableNames: 'path http fs url host port basePath util username password fallbackPage'
|
|
|
+ package: 'AmberCli'!
|
|
|
+
|
|
|
+!FileServer methodsFor: 'accessing'!
|
|
|
+
|
|
|
+basePath
|
|
|
+ ^basePath ifNil: ['./']
|
|
|
+!
|
|
|
+
|
|
|
+basePath: aString
|
|
|
+ basePath := aString
|
|
|
+!
|
|
|
+
|
|
|
+fallbackPage
|
|
|
+ ^fallbackPage
|
|
|
+!
|
|
|
+
|
|
|
+fallbackPage: aString
|
|
|
+ fallbackPage := aString
|
|
|
+!
|
|
|
+
|
|
|
+host
|
|
|
+ ^host
|
|
|
+!
|
|
|
+
|
|
|
+host: hostname
|
|
|
+ host := hostname
|
|
|
+!
|
|
|
+
|
|
|
+password: aPassword
|
|
|
+ password := aPassword.
|
|
|
+!
|
|
|
+
|
|
|
+port
|
|
|
+ ^port
|
|
|
+!
|
|
|
+
|
|
|
+port: aNumber
|
|
|
+ port := aNumber
|
|
|
+!
|
|
|
+
|
|
|
+username: aUsername
|
|
|
+ username := aUsername.
|
|
|
+!
|
|
|
+
|
|
|
+username: aUsername password: aPassword
|
|
|
+ username := aUsername.
|
|
|
+ password := aPassword.
|
|
|
+! !
|
|
|
+
|
|
|
+!FileServer methodsFor: 'initialization'!
|
|
|
+
|
|
|
+checkDirectoryLayout
|
|
|
+ (fs existsSync: self basePath, 'index.html') ifFalse: [
|
|
|
+ console warn: 'Warning: project directory does not contain index.html'].
|
|
|
+ (fs existsSync: self basePath, 'st') ifFalse: [
|
|
|
+ console warn: 'Warning: project directory is missing an "st" directory'].
|
|
|
+ (fs existsSync: self basePath, 'js') ifFalse: [
|
|
|
+ console warn: 'Warning: project directory is missing a "js" directory'].
|
|
|
+!
|
|
|
+
|
|
|
+initialize
|
|
|
+ super initialize.
|
|
|
+ path := self require: 'path'.
|
|
|
+ http := self require: 'http'.
|
|
|
+ fs := self require: 'fs'.
|
|
|
+ util := self require: 'util'.
|
|
|
+ url := self require: 'url'.
|
|
|
+ host := self class defaultHost.
|
|
|
+ port := self class defaultPort.
|
|
|
+ username := nil.
|
|
|
+ password := nil.
|
|
|
+ fallbackPage := nil.
|
|
|
+! !
|
|
|
+
|
|
|
+!FileServer methodsFor: 'private'!
|
|
|
+
|
|
|
+base64Decode: aString
|
|
|
+ <return (new Buffer(aString, 'base64').toString())>
|
|
|
+!
|
|
|
+
|
|
|
+isAuthenticated: aRequest
|
|
|
+ "Basic HTTP Auth: http://stackoverflow.com/a/5957629/293175
|
|
|
+ and https://gist.github.com/1686663"
|
|
|
+ | header token auth parts|
|
|
|
+
|
|
|
+ (username isNil and: [password isNil]) ifTrue: [^true].
|
|
|
+
|
|
|
+ "get authentication header"
|
|
|
+ header := (aRequest headers at: 'authorization') ifNil:[''].
|
|
|
+ (header isEmpty)
|
|
|
+ ifTrue: [^false]
|
|
|
+ ifFalse: [
|
|
|
+ "get authentication token"
|
|
|
+ token := (header tokenize: ' ') ifNil:[''].
|
|
|
+ "convert back from base64"
|
|
|
+ auth := self base64Decode: (token at: 2).
|
|
|
+ "split token at colon"
|
|
|
+ parts := auth tokenize: ':'.
|
|
|
+
|
|
|
+ ((username = (parts at: 1)) and: [password = (parts at: 2)])
|
|
|
+ ifTrue: [^true]
|
|
|
+ ifFalse: [^false]
|
|
|
+ ].
|
|
|
+!
|
|
|
+
|
|
|
+require: aModuleString
|
|
|
+ "call to the require function"
|
|
|
+ ^require value: aModuleString
|
|
|
+!
|
|
|
+
|
|
|
+writeData: data toFileNamed: aFilename
|
|
|
+ console log: aFilename
|
|
|
+! !
|
|
|
+
|
|
|
+!FileServer methodsFor: 'request handling'!
|
|
|
+
|
|
|
+handleGETRequest: aRequest respondTo: aResponse
|
|
|
+ | uri filename |
|
|
|
+ uri := (url parse: aRequest url) pathname.
|
|
|
+ filename := path join: self basePath with: uri.
|
|
|
+ fs exists: filename do: [:aBoolean |
|
|
|
+ aBoolean
|
|
|
+ ifFalse: [self respondNotFoundTo: aResponse]
|
|
|
+ ifTrue: [self respondFileNamed: filename to: aResponse]]
|
|
|
+!
|
|
|
+
|
|
|
+handleOPTIONSRequest: aRequest respondTo: aResponse
|
|
|
+ aResponse writeHead: 200 options: #{'Access-Control-Allow-Origin' -> '*'.
|
|
|
+ 'Access-Control-Allow-Methods' -> 'GET, PUT, POST, DELETE, OPTIONS'.
|
|
|
+ 'Access-Control-Allow-Headers' -> 'Content-Type, Accept'.
|
|
|
+ 'Content-Length' -> 0.
|
|
|
+ 'Access-Control-Max-Age' -> 10}.
|
|
|
+ aResponse end
|
|
|
+!
|
|
|
+
|
|
|
+handlePUTRequest: aRequest respondTo: aResponse
|
|
|
+ | file stream |
|
|
|
+ (self isAuthenticated: aRequest)
|
|
|
+ ifFalse: [self respondAuthenticationRequiredTo: aResponse. ^nil].
|
|
|
+
|
|
|
+ file := '.', aRequest url.
|
|
|
+ stream := fs createWriteStream: file.
|
|
|
+
|
|
|
+ stream on: 'error' do: [:error |
|
|
|
+ console warn: 'Error creating WriteStream for file ', file.
|
|
|
+ console warn: ' Did you forget to create the necessary js/ or st/ directory in your project?'.
|
|
|
+ console warn: ' The exact error is: ', error.
|
|
|
+ self respondNotCreatedTo: aResponse].
|
|
|
+
|
|
|
+ stream on: 'close' do: [
|
|
|
+ self respondCreatedTo: aResponse].
|
|
|
+
|
|
|
+ aRequest setEncoding: 'utf8'.
|
|
|
+ aRequest on: 'data' do: [:data |
|
|
|
+ stream write: data].
|
|
|
+
|
|
|
+ aRequest on: 'end' do: [
|
|
|
+ stream writable ifTrue: [stream end]]
|
|
|
+!
|
|
|
+
|
|
|
+handleRequest: aRequest respondTo: aResponse
|
|
|
+ aRequest method = 'PUT'
|
|
|
+ ifTrue: [self handlePUTRequest: aRequest respondTo: aResponse].
|
|
|
+ aRequest method = 'GET'
|
|
|
+ ifTrue:[self handleGETRequest: aRequest respondTo: aResponse].
|
|
|
+ aRequest method = 'OPTIONS'
|
|
|
+ ifTrue:[self handleOPTIONSRequest: aRequest respondTo: aResponse]
|
|
|
+!
|
|
|
+
|
|
|
+respondAuthenticationRequiredTo: aResponse
|
|
|
+ aResponse
|
|
|
+ writeHead: 401 options: #{'WWW-Authenticate' -> 'Basic realm="Secured Developer Area"'};
|
|
|
+ write: '<html><body>Authentication needed</body></html>';
|
|
|
+ end.
|
|
|
+!
|
|
|
+
|
|
|
+respondCreatedTo: aResponse
|
|
|
+ aResponse
|
|
|
+ writeHead: 201 options: #{'Content-Type' -> 'text/plain'. 'Access-Control-Allow-Origin' -> '*'};
|
|
|
+ end.
|
|
|
+!
|
|
|
+
|
|
|
+respondFileNamed: aFilename to: aResponse
|
|
|
+ | type filename |
|
|
|
+
|
|
|
+ filename := aFilename.
|
|
|
+ (fs statSync: aFilename) isDirectory ifTrue: [
|
|
|
+ filename := filename, 'index.html'].
|
|
|
+
|
|
|
+ fs readFile: filename do: [:ex :file |
|
|
|
+ ex notNil
|
|
|
+ ifTrue: [
|
|
|
+ console log: filename, ' does not exist'.
|
|
|
+ self respondInternalErrorTo: aResponse]
|
|
|
+ ifFalse: [
|
|
|
+ type := self class mimeTypeFor: filename.
|
|
|
+ type = 'application/javascript'
|
|
|
+ ifTrue: [ type:=type,';charset=utf-8' ].
|
|
|
+ aResponse
|
|
|
+ writeHead: 200 options: #{'Content-Type' -> type};
|
|
|
+ write: file encoding: 'binary';
|
|
|
+ end]]
|
|
|
+!
|
|
|
+
|
|
|
+respondInternalErrorTo: aResponse
|
|
|
+ aResponse
|
|
|
+ writeHead: 500 options: #{'Content-Type' -> 'text/plain'};
|
|
|
+ write: '500 Internal server error';
|
|
|
+ end
|
|
|
+!
|
|
|
+
|
|
|
+respondNotCreatedTo: aResponse
|
|
|
+ aResponse
|
|
|
+ writeHead: 400 options: #{'Content-Type' -> 'text/plain'};
|
|
|
+ write: 'File could not be created. Did you forget to create the st/js directories on the server?';
|
|
|
+ end.
|
|
|
+!
|
|
|
+
|
|
|
+respondNotFoundTo: aResponse
|
|
|
+ self fallbackPage isNil ifFalse: [^self respondFileNamed: self fallbackPage to: aResponse].
|
|
|
+ aResponse
|
|
|
+ writeHead: 404 options: #{'Content-Type' -> 'text/plain'};
|
|
|
+ write: '404 Not found';
|
|
|
+ end
|
|
|
+!
|
|
|
+
|
|
|
+respondOKTo: aResponse
|
|
|
+ aResponse
|
|
|
+ writeHead: 200 options: #{'Content-Type' -> 'text/plain'. 'Access-Control-Allow-Origin' -> '*'};
|
|
|
+ end.
|
|
|
+! !
|
|
|
+
|
|
|
+!FileServer methodsFor: 'starting'!
|
|
|
+
|
|
|
+start
|
|
|
+ "Checks if required directory layout is present (issue warning if not).
|
|
|
+ Afterwards start the server."
|
|
|
+ self checkDirectoryLayout.
|
|
|
+ (http createServer: [:request :response |
|
|
|
+ self handleRequest: request respondTo: response])
|
|
|
+ on: 'error' do: [:error | console log: 'Error starting server: ', error];
|
|
|
+ on: 'listening' do: [console log: 'Starting file server on ', self host, ':', self port asString];
|
|
|
+ listen: self port host: self host.
|
|
|
+!
|
|
|
+
|
|
|
+startOn: aPort
|
|
|
+ self port: aPort.
|
|
|
+ self start
|
|
|
+! !
|
|
|
+
|
|
|
+FileServer class instanceVariableNames: 'mimeTypes'!
|
|
|
+
|
|
|
+!FileServer class methodsFor: 'accessing'!
|
|
|
+
|
|
|
+commandLineSwitches
|
|
|
+ "Collect all methodnames from the 'accessing' protocol
|
|
|
+ and select the ones with only one parameter.
|
|
|
+ Then remove the ':' at the end of the name
|
|
|
+ and add a '--' at the beginning.
|
|
|
+ Additionally all uppercase letters are made lowercase and preceded by a '-'.
|
|
|
+ Example: fallbackPage: becomes --fallback-page.
|
|
|
+ Return the Array containing the commandline switches."
|
|
|
+ | switches |
|
|
|
+ switches := ((self methodsInProtocol: 'accessing') collect: [ :each | each selector]).
|
|
|
+ switches := switches select: [ :each | each match: '^[^:]*:$'].
|
|
|
+ switches :=switches collect: [ :each |
|
|
|
+ (each allButLast replace: '([A-Z])' with: '-$1') asLowercase replace: '^([a-z])' with: '--$1' ].
|
|
|
+ ^switches
|
|
|
+!
|
|
|
+
|
|
|
+defaultHost
|
|
|
+ ^'127.0.0.1'
|
|
|
+!
|
|
|
+
|
|
|
+defaultMimeTypes
|
|
|
+ ^ #{
|
|
|
+ '%' -> 'application/x-trash'.
|
|
|
+ '323' -> 'text/h323'.
|
|
|
+ 'abw' -> 'application/x-abiword'.
|
|
|
+ 'ai' -> 'application/postscript'.
|
|
|
+ 'aif' -> 'audio/x-aiff'.
|
|
|
+ 'aifc' -> 'audio/x-aiff'.
|
|
|
+ 'aiff' -> 'audio/x-aiff'.
|
|
|
+ 'alc' -> 'chemical/x-alchemy'.
|
|
|
+ 'art' -> 'image/x-jg'.
|
|
|
+ 'asc' -> 'text/plain'.
|
|
|
+ 'asf' -> 'video/x-ms-asf'.
|
|
|
+ 'asn' -> 'chemical/x-ncbi-asn1-spec'.
|
|
|
+ 'aso' -> 'chemical/x-ncbi-asn1-binary'.
|
|
|
+ 'asx' -> 'video/x-ms-asf'.
|
|
|
+ 'au' -> 'audio/basic'.
|
|
|
+ 'avi' -> 'video/x-msvideo'.
|
|
|
+ 'b' -> 'chemical/x-molconn-Z'.
|
|
|
+ 'bak' -> 'application/x-trash'.
|
|
|
+ 'bat' -> 'application/x-msdos-program'.
|
|
|
+ 'bcpio' -> 'application/x-bcpio'.
|
|
|
+ 'bib' -> 'text/x-bibtex'.
|
|
|
+ 'bin' -> 'application/octet-stream'.
|
|
|
+ 'bmp' -> 'image/x-ms-bmp'.
|
|
|
+ 'book' -> 'application/x-maker'.
|
|
|
+ 'bsd' -> 'chemical/x-crossfire'.
|
|
|
+ 'c' -> 'text/x-csrc'.
|
|
|
+ 'c++' -> 'text/x-c++src'.
|
|
|
+ 'c3d' -> 'chemical/x-chem3d'.
|
|
|
+ 'cac' -> 'chemical/x-cache'.
|
|
|
+ 'cache' -> 'chemical/x-cache'.
|
|
|
+ 'cascii' -> 'chemical/x-cactvs-binary'.
|
|
|
+ 'cat' -> 'application/vnd.ms-pki.seccat'.
|
|
|
+ 'cbin' -> 'chemical/x-cactvs-binary'.
|
|
|
+ 'cc' -> 'text/x-c++src'.
|
|
|
+ 'cdf' -> 'application/x-cdf'.
|
|
|
+ 'cdr' -> 'image/x-coreldraw'.
|
|
|
+ 'cdt' -> 'image/x-coreldrawtemplate'.
|
|
|
+ 'cdx' -> 'chemical/x-cdx'.
|
|
|
+ 'cdy' -> 'application/vnd.cinderella'.
|
|
|
+ 'cef' -> 'chemical/x-cxf'.
|
|
|
+ 'cer' -> 'chemical/x-cerius'.
|
|
|
+ 'chm' -> 'chemical/x-chemdraw'.
|
|
|
+ 'chrt' -> 'application/x-kchart'.
|
|
|
+ 'cif' -> 'chemical/x-cif'.
|
|
|
+ 'class' -> 'application/java-vm'.
|
|
|
+ 'cls' -> 'text/x-tex'.
|
|
|
+ 'cmdf' -> 'chemical/x-cmdf'.
|
|
|
+ 'cml' -> 'chemical/x-cml'.
|
|
|
+ 'cod' -> 'application/vnd.rim.cod'.
|
|
|
+ 'com' -> 'application/x-msdos-program'.
|
|
|
+ 'cpa' -> 'chemical/x-compass'.
|
|
|
+ 'cpio' -> 'application/x-cpio'.
|
|
|
+ 'cpp' -> 'text/x-c++src'.
|
|
|
+ 'cpt' -> 'image/x-corelphotopaint'.
|
|
|
+ 'crl' -> 'application/x-pkcs7-crl'.
|
|
|
+ 'crt' -> 'application/x-x509-ca-cert'.
|
|
|
+ 'csf' -> 'chemical/x-cache-csf'.
|
|
|
+ 'csh' -> 'text/x-csh'.
|
|
|
+ 'csm' -> 'chemical/x-csml'.
|
|
|
+ 'csml' -> 'chemical/x-csml'.
|
|
|
+ 'css' -> 'text/css'.
|
|
|
+ 'csv' -> 'text/comma-separated-values'.
|
|
|
+ 'ctab' -> 'chemical/x-cactvs-binary'.
|
|
|
+ 'ctx' -> 'chemical/x-ctx'.
|
|
|
+ 'cu' -> 'application/cu-seeme'.
|
|
|
+ 'cub' -> 'chemical/x-gaussian-cube'.
|
|
|
+ 'cxf' -> 'chemical/x-cxf'.
|
|
|
+ 'cxx' -> 'text/x-c++src'.
|
|
|
+ 'dat' -> 'chemical/x-mopac-input'.
|
|
|
+ 'dcr' -> 'application/x-director'.
|
|
|
+ 'deb' -> 'application/x-debian-package'.
|
|
|
+ 'dif' -> 'video/dv'.
|
|
|
+ 'diff' -> 'text/plain'.
|
|
|
+ 'dir' -> 'application/x-director'.
|
|
|
+ 'djv' -> 'image/vnd.djvu'.
|
|
|
+ 'djvu' -> 'image/vnd.djvu'.
|
|
|
+ 'dl' -> 'video/dl'.
|
|
|
+ 'dll' -> 'application/x-msdos-program'.
|
|
|
+ 'dmg' -> 'application/x-apple-diskimage'.
|
|
|
+ 'dms' -> 'application/x-dms'.
|
|
|
+ 'doc' -> 'application/msword'.
|
|
|
+ 'dot' -> 'application/msword'.
|
|
|
+ 'dv' -> 'video/dv'.
|
|
|
+ 'dvi' -> 'application/x-dvi'.
|
|
|
+ 'dx' -> 'chemical/x-jcamp-dx'.
|
|
|
+ 'dxr' -> 'application/x-director'.
|
|
|
+ 'emb' -> 'chemical/x-embl-dl-nucleotide'.
|
|
|
+ 'embl' -> 'chemical/x-embl-dl-nucleotide'.
|
|
|
+ 'ent' -> 'chemical/x-pdb'.
|
|
|
+ 'eps' -> 'application/postscript'.
|
|
|
+ 'etx' -> 'text/x-setext'.
|
|
|
+ 'exe' -> 'application/x-msdos-program'.
|
|
|
+ 'ez' -> 'application/andrew-inset'.
|
|
|
+ 'fb' -> 'application/x-maker'.
|
|
|
+ 'fbdoc' -> 'application/x-maker'.
|
|
|
+ 'fch' -> 'chemical/x-gaussian-checkpoint'.
|
|
|
+ 'fchk' -> 'chemical/x-gaussian-checkpoint'.
|
|
|
+ 'fig' -> 'application/x-xfig'.
|
|
|
+ 'flac' -> 'application/x-flac'.
|
|
|
+ 'fli' -> 'video/fli'.
|
|
|
+ 'fm' -> 'application/x-maker'.
|
|
|
+ 'frame' -> 'application/x-maker'.
|
|
|
+ 'frm' -> 'application/x-maker'.
|
|
|
+ 'gal' -> 'chemical/x-gaussian-log'.
|
|
|
+ 'gam' -> 'chemical/x-gamess-input'.
|
|
|
+ 'gamin' -> 'chemical/x-gamess-input'.
|
|
|
+ 'gau' -> 'chemical/x-gaussian-input'.
|
|
|
+ 'gcd' -> 'text/x-pcs-gcd'.
|
|
|
+ 'gcf' -> 'application/x-graphing-calculator'.
|
|
|
+ 'gcg' -> 'chemical/x-gcg8-sequence'.
|
|
|
+ 'gen' -> 'chemical/x-genbank'.
|
|
|
+ 'gf' -> 'application/x-tex-gf'.
|
|
|
+ 'gif' -> 'image/gif'.
|
|
|
+ 'gjc' -> 'chemical/x-gaussian-input'.
|
|
|
+ 'gjf' -> 'chemical/x-gaussian-input'.
|
|
|
+ 'gl' -> 'video/gl'.
|
|
|
+ 'gnumeric' -> 'application/x-gnumeric'.
|
|
|
+ 'gpt' -> 'chemical/x-mopac-graph'.
|
|
|
+ 'gsf' -> 'application/x-font'.
|
|
|
+ 'gsm' -> 'audio/x-gsm'.
|
|
|
+ 'gtar' -> 'application/x-gtar'.
|
|
|
+ 'h' -> 'text/x-chdr'.
|
|
|
+ 'h++' -> 'text/x-c++hdr'.
|
|
|
+ 'hdf' -> 'application/x-hdf'.
|
|
|
+ 'hh' -> 'text/x-c++hdr'.
|
|
|
+ 'hin' -> 'chemical/x-hin'.
|
|
|
+ 'hpp' -> 'text/x-c++hdr'.
|
|
|
+ 'hqx' -> 'application/mac-binhex40'.
|
|
|
+ 'hs' -> 'text/x-haskell'.
|
|
|
+ 'hta' -> 'application/hta'.
|
|
|
+ 'htc' -> 'text/x-component'.
|
|
|
+ 'htm' -> 'text/html'.
|
|
|
+ 'html' -> 'text/html'.
|
|
|
+ 'hxx' -> 'text/x-c++hdr'.
|
|
|
+ 'ica' -> 'application/x-ica'.
|
|
|
+ 'ice' -> 'x-conference/x-cooltalk'.
|
|
|
+ 'ico' -> 'image/x-icon'.
|
|
|
+ 'ics' -> 'text/calendar'.
|
|
|
+ 'icz' -> 'text/calendar'.
|
|
|
+ 'ief' -> 'image/ief'.
|
|
|
+ 'iges' -> 'model/iges'.
|
|
|
+ 'igs' -> 'model/iges'.
|
|
|
+ 'iii' -> 'application/x-iphone'.
|
|
|
+ 'inp' -> 'chemical/x-gamess-input'.
|
|
|
+ 'ins' -> 'application/x-internet-signup'.
|
|
|
+ 'iso' -> 'application/x-iso9660-image'.
|
|
|
+ 'isp' -> 'application/x-internet-signup'.
|
|
|
+ 'ist' -> 'chemical/x-isostar'.
|
|
|
+ 'istr' -> 'chemical/x-isostar'.
|
|
|
+ 'jad' -> 'text/vnd.sun.j2me.app-descriptor'.
|
|
|
+ 'jar' -> 'application/java-archive'.
|
|
|
+ 'java' -> 'text/x-java'.
|
|
|
+ 'jdx' -> 'chemical/x-jcamp-dx'.
|
|
|
+ 'jmz' -> 'application/x-jmol'.
|
|
|
+ 'jng' -> 'image/x-jng'.
|
|
|
+ 'jnlp' -> 'application/x-java-jnlp-file'.
|
|
|
+ 'jpe' -> 'image/jpeg'.
|
|
|
+ 'jpeg' -> 'image/jpeg'.
|
|
|
+ 'jpg' -> 'image/jpeg'.
|
|
|
+ 'js' -> 'application/javascript'.
|
|
|
+ 'kar' -> 'audio/midi'.
|
|
|
+ 'key' -> 'application/pgp-keys'.
|
|
|
+ 'kil' -> 'application/x-killustrator'.
|
|
|
+ 'kin' -> 'chemical/x-kinemage'.
|
|
|
+ 'kpr' -> 'application/x-kpresenter'.
|
|
|
+ 'kpt' -> 'application/x-kpresenter'.
|
|
|
+ 'ksp' -> 'application/x-kspread'.
|
|
|
+ 'kwd' -> 'application/x-kword'.
|
|
|
+ 'kwt' -> 'application/x-kword'.
|
|
|
+ 'latex' -> 'application/x-latex'.
|
|
|
+ 'lha' -> 'application/x-lha'.
|
|
|
+ 'lhs' -> 'text/x-literate-haskell'.
|
|
|
+ 'lsf' -> 'video/x-la-asf'.
|
|
|
+ 'lsx' -> 'video/x-la-asf'.
|
|
|
+ 'ltx' -> 'text/x-tex'.
|
|
|
+ 'lzh' -> 'application/x-lzh'.
|
|
|
+ 'lzx' -> 'application/x-lzx'.
|
|
|
+ 'm3u' -> 'audio/x-mpegurl'.
|
|
|
+ 'm4a' -> 'audio/mpeg'.
|
|
|
+ 'maker' -> 'application/x-maker'.
|
|
|
+ 'man' -> 'application/x-troff-man'.
|
|
|
+ 'mcif' -> 'chemical/x-mmcif'.
|
|
|
+ 'mcm' -> 'chemical/x-macmolecule'.
|
|
|
+ 'mdb' -> 'application/msaccess'.
|
|
|
+ 'me' -> 'application/x-troff-me'.
|
|
|
+ 'mesh' -> 'model/mesh'.
|
|
|
+ 'mid' -> 'audio/midi'.
|
|
|
+ 'midi' -> 'audio/midi'.
|
|
|
+ 'mif' -> 'application/x-mif'.
|
|
|
+ 'mm' -> 'application/x-freemind'.
|
|
|
+ 'mmd' -> 'chemical/x-macromodel-input'.
|
|
|
+ 'mmf' -> 'application/vnd.smaf'.
|
|
|
+ 'mml' -> 'text/mathml'.
|
|
|
+ 'mmod' -> 'chemical/x-macromodel-input'.
|
|
|
+ 'mng' -> 'video/x-mng'.
|
|
|
+ 'moc' -> 'text/x-moc'.
|
|
|
+ 'mol' -> 'chemical/x-mdl-molfile'.
|
|
|
+ 'mol2' -> 'chemical/x-mol2'.
|
|
|
+ 'moo' -> 'chemical/x-mopac-out'.
|
|
|
+ 'mop' -> 'chemical/x-mopac-input'.
|
|
|
+ 'mopcrt' -> 'chemical/x-mopac-input'.
|
|
|
+ 'mov' -> 'video/quicktime'.
|
|
|
+ 'movie' -> 'video/x-sgi-movie'.
|
|
|
+ 'mp2' -> 'audio/mpeg'.
|
|
|
+ 'mp3' -> 'audio/mpeg'.
|
|
|
+ 'mp4' -> 'video/mp4'.
|
|
|
+ 'mpc' -> 'chemical/x-mopac-input'.
|
|
|
+ 'mpe' -> 'video/mpeg'.
|
|
|
+ 'mpeg' -> 'video/mpeg'.
|
|
|
+ 'mpega' -> 'audio/mpeg'.
|
|
|
+ 'mpg' -> 'video/mpeg'.
|
|
|
+ 'mpga' -> 'audio/mpeg'.
|
|
|
+ 'ms' -> 'application/x-troff-ms'.
|
|
|
+ 'msh' -> 'model/mesh'.
|
|
|
+ 'msi' -> 'application/x-msi'.
|
|
|
+ 'mvb' -> 'chemical/x-mopac-vib'.
|
|
|
+ 'mxu' -> 'video/vnd.mpegurl'.
|
|
|
+ 'nb' -> 'application/mathematica'.
|
|
|
+ 'nc' -> 'application/x-netcdf'.
|
|
|
+ 'nwc' -> 'application/x-nwc'.
|
|
|
+ 'o' -> 'application/x-object'.
|
|
|
+ 'oda' -> 'application/oda'.
|
|
|
+ 'odb' -> 'application/vnd.oasis.opendocument.database'.
|
|
|
+ 'odc' -> 'application/vnd.oasis.opendocument.chart'.
|
|
|
+ 'odf' -> 'application/vnd.oasis.opendocument.formula'.
|
|
|
+ 'odg' -> 'application/vnd.oasis.opendocument.graphics'.
|
|
|
+ 'odi' -> 'application/vnd.oasis.opendocument.image'.
|
|
|
+ 'odm' -> 'application/vnd.oasis.opendocument.text-master'.
|
|
|
+ 'odp' -> 'application/vnd.oasis.opendocument.presentation'.
|
|
|
+ 'ods' -> 'application/vnd.oasis.opendocument.spreadsheet'.
|
|
|
+ 'odt' -> 'application/vnd.oasis.opendocument.text'.
|
|
|
+ 'ogg' -> 'application/ogg'.
|
|
|
+ 'old' -> 'application/x-trash'.
|
|
|
+ 'oth' -> 'application/vnd.oasis.opendocument.text-web'.
|
|
|
+ 'oza' -> 'application/x-oz-application'.
|
|
|
+ 'p' -> 'text/x-pascal'.
|
|
|
+ 'p7r' -> 'application/x-pkcs7-certreqresp'.
|
|
|
+ 'pac' -> 'application/x-ns-proxy-autoconfig'.
|
|
|
+ 'pas' -> 'text/x-pascal'.
|
|
|
+ 'pat' -> 'image/x-coreldrawpattern'.
|
|
|
+ 'pbm' -> 'image/x-portable-bitmap'.
|
|
|
+ 'pcf' -> 'application/x-font'.
|
|
|
+ 'pcf.Z' -> 'application/x-font'.
|
|
|
+ 'pcx' -> 'image/pcx'.
|
|
|
+ 'pdb' -> 'chemical/x-pdb'.
|
|
|
+ 'pdf' -> 'application/pdf'.
|
|
|
+ 'pfa' -> 'application/x-font'.
|
|
|
+ 'pfb' -> 'application/x-font'.
|
|
|
+ 'pgm' -> 'image/x-portable-graymap'.
|
|
|
+ 'pgn' -> 'application/x-chess-pgn'.
|
|
|
+ 'pgp' -> 'application/pgp-signature'.
|
|
|
+ 'pk' -> 'application/x-tex-pk'.
|
|
|
+ 'pl' -> 'text/x-perl'.
|
|
|
+ 'pls' -> 'audio/x-scpls'.
|
|
|
+ 'pm' -> 'text/x-perl'.
|
|
|
+ 'png' -> 'image/png'.
|
|
|
+ 'pnm' -> 'image/x-portable-anymap'.
|
|
|
+ 'pot' -> 'text/plain'.
|
|
|
+ 'ppm' -> 'image/x-portable-pixmap'.
|
|
|
+ 'pps' -> 'application/vnd.ms-powerpoint'.
|
|
|
+ 'ppt' -> 'application/vnd.ms-powerpoint'.
|
|
|
+ 'prf' -> 'application/pics-rules'.
|
|
|
+ 'prt' -> 'chemical/x-ncbi-asn1-ascii'.
|
|
|
+ 'ps' -> 'application/postscript'.
|
|
|
+ 'psd' -> 'image/x-photoshop'.
|
|
|
+ 'psp' -> 'text/x-psp'.
|
|
|
+ 'py' -> 'text/x-python'.
|
|
|
+ 'pyc' -> 'application/x-python-code'.
|
|
|
+ 'pyo' -> 'application/x-python-code'.
|
|
|
+ 'qt' -> 'video/quicktime'.
|
|
|
+ 'qtl' -> 'application/x-quicktimeplayer'.
|
|
|
+ 'ra' -> 'audio/x-realaudio'.
|
|
|
+ 'ram' -> 'audio/x-pn-realaudio'.
|
|
|
+ 'rar' -> 'application/rar'.
|
|
|
+ 'ras' -> 'image/x-cmu-raster'.
|
|
|
+ 'rd' -> 'chemical/x-mdl-rdfile'.
|
|
|
+ 'rdf' -> 'application/rdf+xml'.
|
|
|
+ 'rgb' -> 'image/x-rgb'.
|
|
|
+ 'rm' -> 'audio/x-pn-realaudio'.
|
|
|
+ 'roff' -> 'application/x-troff'.
|
|
|
+ 'ros' -> 'chemical/x-rosdal'.
|
|
|
+ 'rpm' -> 'application/x-redhat-package-manager'.
|
|
|
+ 'rss' -> 'application/rss+xml'.
|
|
|
+ 'rtf' -> 'text/rtf'.
|
|
|
+ 'rtx' -> 'text/richtext'.
|
|
|
+ 'rxn' -> 'chemical/x-mdl-rxnfile'.
|
|
|
+ 'sct' -> 'text/scriptlet'.
|
|
|
+ 'sd' -> 'chemical/x-mdl-sdfile'.
|
|
|
+ 'sd2' -> 'audio/x-sd2'.
|
|
|
+ 'sda' -> 'application/vnd.stardivision.draw'.
|
|
|
+ 'sdc' -> 'application/vnd.stardivision.calc'.
|
|
|
+ 'sdd' -> 'application/vnd.stardivision.impress'.
|
|
|
+ 'sdf' -> 'chemical/x-mdl-sdfile'.
|
|
|
+ 'sdp' -> 'application/vnd.stardivision.impress'.
|
|
|
+ 'sdw' -> 'application/vnd.stardivision.writer'.
|
|
|
+ 'ser' -> 'application/java-serialized-object'.
|
|
|
+ 'sgf' -> 'application/x-go-sgf'.
|
|
|
+ 'sgl' -> 'application/vnd.stardivision.writer-global'.
|
|
|
+ 'sh' -> 'text/x-sh'.
|
|
|
+ 'shar' -> 'application/x-shar'.
|
|
|
+ 'shtml' -> 'text/html'.
|
|
|
+ 'sid' -> 'audio/prs.sid'.
|
|
|
+ 'sik' -> 'application/x-trash'.
|
|
|
+ 'silo' -> 'model/mesh'.
|
|
|
+ 'sis' -> 'application/vnd.symbian.install'.
|
|
|
+ 'sit' -> 'application/x-stuffit'.
|
|
|
+ 'skd' -> 'application/x-koan'.
|
|
|
+ 'skm' -> 'application/x-koan'.
|
|
|
+ 'skp' -> 'application/x-koan'.
|
|
|
+ 'skt' -> 'application/x-koan'.
|
|
|
+ 'smf' -> 'application/vnd.stardivision.math'.
|
|
|
+ 'smi' -> 'application/smil'.
|
|
|
+ 'smil' -> 'application/smil'.
|
|
|
+ 'snd' -> 'audio/basic'.
|
|
|
+ 'spc' -> 'chemical/x-galactic-spc'.
|
|
|
+ 'spl' -> 'application/x-futuresplash'.
|
|
|
+ 'src' -> 'application/x-wais-source'.
|
|
|
+ 'stc' -> 'application/vnd.sun.xml.calc.template'.
|
|
|
+ 'std' -> 'application/vnd.sun.xml.draw.template'.
|
|
|
+ 'sti' -> 'application/vnd.sun.xml.impress.template'.
|
|
|
+ 'stl' -> 'application/vnd.ms-pki.stl'.
|
|
|
+ 'stw' -> 'application/vnd.sun.xml.writer.template'.
|
|
|
+ 'sty' -> 'text/x-tex'.
|
|
|
+ 'sv4cpio' -> 'application/x-sv4cpio'.
|
|
|
+ 'sv4crc' -> 'application/x-sv4crc'.
|
|
|
+ 'svg' -> 'image/svg+xml'.
|
|
|
+ 'svgz' -> 'image/svg+xml'.
|
|
|
+ 'sw' -> 'chemical/x-swissprot'.
|
|
|
+ 'swf' -> 'application/x-shockwave-flash'.
|
|
|
+ 'swfl' -> 'application/x-shockwave-flash'.
|
|
|
+ 'sxc' -> 'application/vnd.sun.xml.calc'.
|
|
|
+ 'sxd' -> 'application/vnd.sun.xml.draw'.
|
|
|
+ 'sxg' -> 'application/vnd.sun.xml.writer.global'.
|
|
|
+ 'sxi' -> 'application/vnd.sun.xml.impress'.
|
|
|
+ 'sxm' -> 'application/vnd.sun.xml.math'.
|
|
|
+ 'sxw' -> 'application/vnd.sun.xml.writer'.
|
|
|
+ 't' -> 'application/x-troff'.
|
|
|
+ 'tar' -> 'application/x-tar'.
|
|
|
+ 'taz' -> 'application/x-gtar'.
|
|
|
+ 'tcl' -> 'text/x-tcl'.
|
|
|
+ 'tex' -> 'text/x-tex'.
|
|
|
+ 'texi' -> 'application/x-texinfo'.
|
|
|
+ 'texinfo' -> 'application/x-texinfo'.
|
|
|
+ 'text' -> 'text/plain'.
|
|
|
+ 'tgf' -> 'chemical/x-mdl-tgf'.
|
|
|
+ 'tgz' -> 'application/x-gtar'.
|
|
|
+ 'tif' -> 'image/tiff'.
|
|
|
+ 'tiff' -> 'image/tiff'.
|
|
|
+ 'tk' -> 'text/x-tcl'.
|
|
|
+ 'tm' -> 'text/texmacs'.
|
|
|
+ 'torrent' -> 'application/x-bittorrent'.
|
|
|
+ 'tr' -> 'application/x-troff'.
|
|
|
+ 'ts' -> 'text/texmacs'.
|
|
|
+ 'tsp' -> 'application/dsptype'.
|
|
|
+ 'tsv' -> 'text/tab-separated-values'.
|
|
|
+ 'txt' -> 'text/plain'.
|
|
|
+ 'udeb' -> 'application/x-debian-package'.
|
|
|
+ 'uls' -> 'text/iuls'.
|
|
|
+ 'ustar' -> 'application/x-ustar'.
|
|
|
+ 'val' -> 'chemical/x-ncbi-asn1-binary'.
|
|
|
+ 'vcd' -> 'application/x-cdlink'.
|
|
|
+ 'vcf' -> 'text/x-vcard'.
|
|
|
+ 'vcs' -> 'text/x-vcalendar'.
|
|
|
+ 'vmd' -> 'chemical/x-vmd'.
|
|
|
+ 'vms' -> 'chemical/x-vamas-iso14976'.
|
|
|
+ 'vor' -> 'application/vnd.stardivision.writer'.
|
|
|
+ 'vrm' -> 'x-world/x-vrml'.
|
|
|
+ 'vrml' -> 'x-world/x-vrml'.
|
|
|
+ 'vsd' -> 'application/vnd.visio'.
|
|
|
+ 'wad' -> 'application/x-doom'.
|
|
|
+ 'wav' -> 'audio/x-wav'.
|
|
|
+ 'wax' -> 'audio/x-ms-wax'.
|
|
|
+ 'wbmp' -> 'image/vnd.wap.wbmp'.
|
|
|
+ 'wbxml' -> 'application/vnd.wap.wbxml'.
|
|
|
+ 'wk' -> 'application/x-123'.
|
|
|
+ 'wm' -> 'video/x-ms-wm'.
|
|
|
+ 'wma' -> 'audio/x-ms-wma'.
|
|
|
+ 'wmd' -> 'application/x-ms-wmd'.
|
|
|
+ 'wml' -> 'text/vnd.wap.wml'.
|
|
|
+ 'wmlc' -> 'application/vnd.wap.wmlc'.
|
|
|
+ 'wmls' -> 'text/vnd.wap.wmlscript'.
|
|
|
+ 'wmlsc' -> 'application/vnd.wap.wmlscriptc'.
|
|
|
+ 'wmv' -> 'video/x-ms-wmv'.
|
|
|
+ 'wmx' -> 'video/x-ms-wmx'.
|
|
|
+ 'wmz' -> 'application/x-ms-wmz'.
|
|
|
+ 'wp5' -> 'application/wordperfect5.1'.
|
|
|
+ 'wpd' -> 'application/wordperfect'.
|
|
|
+ 'wrl' -> 'x-world/x-vrml'.
|
|
|
+ 'wsc' -> 'text/scriptlet'.
|
|
|
+ 'wvx' -> 'video/x-ms-wvx'.
|
|
|
+ 'wz' -> 'application/x-wingz'.
|
|
|
+ 'xbm' -> 'image/x-xbitmap'.
|
|
|
+ 'xcf' -> 'application/x-xcf'.
|
|
|
+ 'xht' -> 'application/xhtml+xml'.
|
|
|
+ 'xhtml' -> 'application/xhtml+xml'.
|
|
|
+ 'xlb' -> 'application/vnd.ms-excel'.
|
|
|
+ 'xls' -> 'application/vnd.ms-excel'.
|
|
|
+ 'xlt' -> 'application/vnd.ms-excel'.
|
|
|
+ 'xml' -> 'application/xml'.
|
|
|
+ 'xpi' -> 'application/x-xpinstall'.
|
|
|
+ 'xpm' -> 'image/x-xpixmap'.
|
|
|
+ 'xsl' -> 'application/xml'.
|
|
|
+ 'xtel' -> 'chemical/x-xtel'.
|
|
|
+ 'xul' -> 'application/vnd.mozilla.xul+xml'.
|
|
|
+ 'xwd' -> 'image/x-xwindowdump'.
|
|
|
+ 'xyz' -> 'chemical/x-xyz'.
|
|
|
+ 'zip' -> 'application/zip'.
|
|
|
+ 'zmt' -> 'chemical/x-mopac-input'.
|
|
|
+ '~' -> 'application/x-trash'
|
|
|
+ }
|
|
|
+!
|
|
|
+
|
|
|
+defaultPort
|
|
|
+ ^4000
|
|
|
+!
|
|
|
+
|
|
|
+mimeTypeFor: aString
|
|
|
+ ^self mimeTypes at: (aString replace: '.*[\.]' with: '') ifAbsent: ['text/plain']
|
|
|
+!
|
|
|
+
|
|
|
+mimeTypes
|
|
|
+ ^mimeTypes ifNil: [mimeTypes := self defaultMimeTypes]
|
|
|
+!
|
|
|
+
|
|
|
+printHelp
|
|
|
+ console log: 'Available commandline options are:'.
|
|
|
+ console log: '--help'.
|
|
|
+ self commandLineSwitches do: [ :each |
|
|
|
+ console log: each, ' <parameter>']
|
|
|
+!
|
|
|
+
|
|
|
+selectorForCommandLineSwitch: aSwitch
|
|
|
+ "Remove the trailing '--', add ':' at the end
|
|
|
+ and replace all occurences of a lowercase letter preceded by a '-' with
|
|
|
+ the Uppercase letter.
|
|
|
+ Example: --fallback-page becomes fallbackPage:"
|
|
|
+ ^((aSwitch replace: '^--' with: '')
|
|
|
+ replace: '-[a-z]' with: [ :each | each second asUppercase ]), ':'
|
|
|
+! !
|
|
|
+
|
|
|
+!FileServer class methodsFor: 'initialization'!
|
|
|
+
|
|
|
+createServerWithArguments: options
|
|
|
+ "If options are empty return a default FileServer instance.
|
|
|
+ If options are given loop through them and set the passed in values
|
|
|
+ on the FileServer instance.
|
|
|
+
|
|
|
+ Commanline options map directly to methods in the 'accessing' protocol
|
|
|
+ taking one parameter.
|
|
|
+ Adding a method to this protocol makes it directly settable through
|
|
|
+ command line options.
|
|
|
+ "
|
|
|
+ | server popFront front optionName optionValue switches |
|
|
|
+
|
|
|
+ switches := self commandLineSwitches.
|
|
|
+
|
|
|
+ server := self new.
|
|
|
+
|
|
|
+ options ifEmpty: [^server].
|
|
|
+
|
|
|
+ (options size even) ifFalse: [
|
|
|
+ console log: 'Using default parameters.'.
|
|
|
+ console log: 'Wrong commandline options or not enough arguments for: ' , options.
|
|
|
+ console log: 'Use any of the following ones: ', switches.
|
|
|
+ ^server].
|
|
|
+
|
|
|
+ popFront := [:args |
|
|
|
+ front := args first.
|
|
|
+ args remove: front.
|
|
|
+ front].
|
|
|
+
|
|
|
+ [options notEmpty] whileTrue: [
|
|
|
+ optionName := popFront value: options.
|
|
|
+ optionValue := popFront value: options.
|
|
|
+
|
|
|
+ (switches includes: optionName) ifTrue: [
|
|
|
+ optionName := self selectorForCommandLineSwitch: optionName.
|
|
|
+ server perform: optionName withArguments: (Array with: optionValue)]
|
|
|
+ ifFalse: [
|
|
|
+ console log: optionName, ' is not a valid commandline option'.
|
|
|
+ console log: 'Use any of the following ones: ', switches ]].
|
|
|
+ ^server.
|
|
|
+!
|
|
|
+
|
|
|
+main
|
|
|
+ "Main entry point for Amber applications.
|
|
|
+ Creates and starts a FileServer instance."
|
|
|
+ | fileServer args |
|
|
|
+ args := process argv.
|
|
|
+ "Remove the first args which contain the path to the node executable and the script file."
|
|
|
+ args removeFrom: 1 to: 3.
|
|
|
+
|
|
|
+ args detect: [ :each |
|
|
|
+ (each = '--help') ifTrue: [FileServer printHelp]]
|
|
|
+ ifNone: [
|
|
|
+ fileServer := FileServer createServerWithArguments: args.
|
|
|
+ ^fileServer start]
|
|
|
+! !
|
|
|
+
|
|
|
+Object subclass: #Repl
|
|
|
+ instanceVariableNames: 'readline interface util'
|
|
|
+ package: 'AmberCli'!
|
|
|
+
|
|
|
+!Repl methodsFor: 'accessing'!
|
|
|
+
|
|
|
+prompt
|
|
|
+ ^'amber >> '
|
|
|
+! !
|
|
|
+
|
|
|
+!Repl methodsFor: 'actions'!
|
|
|
+
|
|
|
+close
|
|
|
+ process stdin destroy
|
|
|
+!
|
|
|
+
|
|
|
+createInterface
|
|
|
+ "No completion for now"
|
|
|
+ interface := readline createInterface: process stdin stdout: process stdout.
|
|
|
+ interface on: 'line' do: [:buffer | self eval: buffer].
|
|
|
+ interface on: 'close' do: [self close].
|
|
|
+ self setPrompt.
|
|
|
+ interface prompt
|
|
|
+!
|
|
|
+
|
|
|
+eval: buffer
|
|
|
+ | result |
|
|
|
+ buffer isEmpty ifFalse: [
|
|
|
+ self try: [
|
|
|
+ result := Compiler new evaluateExpression: buffer.
|
|
|
+ Transcript show: result]
|
|
|
+ catch: [:e |
|
|
|
+ e isSmalltalkError
|
|
|
+ ifTrue: [ErrorHandler new handleError: e]
|
|
|
+ ifFalse: [process stdout write: e jsStack]]].
|
|
|
+ interface prompt
|
|
|
+!
|
|
|
+
|
|
|
+setPrompt
|
|
|
+ interface setPrompt: self prompt
|
|
|
+! !
|
|
|
+
|
|
|
+!Repl methodsFor: 'initialization'!
|
|
|
+
|
|
|
+initialize
|
|
|
+ super initialize.
|
|
|
+ readline := require value: 'readline'.
|
|
|
+ util := require value: 'util'
|
|
|
+! !
|
|
|
+
|
|
|
+!Repl class methodsFor: 'not yet classified'!
|
|
|
+
|
|
|
+main
|
|
|
+ self new createInterface
|
|
|
+! !
|
|
|
+
|