Importer-Exporter.st 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275
  1. Smalltalk current createPackage: 'Importer-Exporter'!
  2. Object subclass: #AbstractExporter
  3. instanceVariableNames: ''
  4. package: 'Importer-Exporter'!
  5. !AbstractExporter commentStamp!
  6. I am an abstract exporter for Amber source code.!
  7. !AbstractExporter methodsFor: 'convenience'!
  8. chunkEscape: aString
  9. "Replace all occurrences of !! with !!!! and trim at both ends."
  10. ^(aString replace: '!!' with: '!!!!') trimBoth
  11. !
  12. classNameFor: aClass
  13. ^aClass isMetaclass
  14. ifTrue: [ aClass instanceClass name, ' class' ]
  15. ifFalse: [
  16. aClass isNil
  17. ifTrue: [ 'nil' ]
  18. ifFalse: [ aClass name ] ]
  19. ! !
  20. !AbstractExporter methodsFor: 'fileOut'!
  21. recipe
  22. "Recipe to export a given package."
  23. self subclassResponsibility
  24. ! !
  25. AbstractExporter class instanceVariableNames: 'default'!
  26. !AbstractExporter class methodsFor: 'instance creation'!
  27. default
  28. ^ default ifNil: [ default := self new ]
  29. ! !
  30. AbstractExporter subclass: #ChunkExporter
  31. instanceVariableNames: ''
  32. package: 'Importer-Exporter'!
  33. !ChunkExporter commentStamp!
  34. I am an exporter dedicated to outputting Amber source code in the classic Smalltalk chunk format.
  35. I do not output any compiled code.!
  36. !ChunkExporter methodsFor: 'accessing'!
  37. extensionCategoriesOfPackage: aPackage
  38. "Issue #143: sort protocol alphabetically"
  39. | name map result |
  40. name := aPackage name.
  41. result := OrderedCollection new.
  42. (Package sortedClasses: Smalltalk current classes) do: [:each |
  43. {each. each class} do: [:aClass |
  44. map := Dictionary new.
  45. aClass protocolsDo: [:category :methods |
  46. (category match: '^\*', name) ifTrue: [ map at: category put: methods ]].
  47. result addAll: ((map keys sorted: [:a :b | a <= b ]) collect: [:category |
  48. MethodCategory name: category theClass: aClass methods: (map at: category)]) ]].
  49. ^result
  50. !
  51. methodsOfCategory: aCategory
  52. "Issue #143: sort methods alphabetically"
  53. ^(aCategory methods) sorted: [:a :b | a selector <= b selector]
  54. !
  55. ownCategoriesOfClass: aClass
  56. "Answer the protocols of aClassthat are not package extensions"
  57. "Issue #143: sort protocol alphabetically"
  58. | map |
  59. map := Dictionary new.
  60. aClass protocolsDo: [:category :methods |
  61. (category match: '^\*') ifFalse: [ map at: category put: methods ]].
  62. ^(map keys sorted: [:a :b | a <= b ]) collect: [:category |
  63. MethodCategory name: category theClass: aClass methods: (map at: category) ]
  64. !
  65. ownCategoriesOfMetaClass: aClass
  66. "Issue #143: sort protocol alphabetically"
  67. ^self ownCategoriesOfClass: aClass class
  68. ! !
  69. !ChunkExporter methodsFor: 'fileOut'!
  70. recipe
  71. "Export a given package."
  72. | exportCategoryRecipe |
  73. exportCategoryRecipe := {
  74. self -> #exportCategoryPrologueOf:on:.
  75. {
  76. self -> #methodsOfCategory:.
  77. self -> #exportMethod:on: }.
  78. self -> #exportCategoryEpilogueOf:on: }.
  79. ^{
  80. self -> #exportPackageDefinitionOf:on:.
  81. {
  82. PluggableExporter -> #ownClassesOfPackage:.
  83. self -> #exportDefinitionOf:on:.
  84. { self -> #ownCategoriesOfClass: }, exportCategoryRecipe.
  85. self -> #exportMetaDefinitionOf:on:.
  86. { self -> #ownCategoriesOfMetaClass: }, exportCategoryRecipe }.
  87. { self -> #extensionCategoriesOfPackage: }, exportCategoryRecipe
  88. }
  89. ! !
  90. !ChunkExporter methodsFor: 'output'!
  91. exportCategoryEpilogueOf: aCategory on: aStream
  92. aStream nextPutAll: ' !!'; lf; lf
  93. !
  94. exportCategoryPrologueOf: aCategory on: aStream
  95. aStream
  96. nextPutAll: '!!', (self classNameFor: aCategory theClass);
  97. nextPutAll: ' methodsFor: ''', aCategory name, '''!!'
  98. !
  99. exportDefinitionOf: aClass on: aStream
  100. "Chunk format."
  101. aStream
  102. nextPutAll: (self classNameFor: aClass superclass);
  103. nextPutAll: ' subclass: #', (self classNameFor: aClass); lf;
  104. tab; nextPutAll: 'instanceVariableNames: '''.
  105. aClass instanceVariableNames
  106. do: [:each | aStream nextPutAll: each]
  107. separatedBy: [aStream nextPutAll: ' '].
  108. aStream
  109. nextPutAll: ''''; lf;
  110. tab; nextPutAll: 'package: ''', aClass category, '''!!'; lf.
  111. aClass comment notEmpty ifTrue: [
  112. aStream
  113. nextPutAll: '!!', (self classNameFor: aClass), ' commentStamp!!';lf;
  114. nextPutAll: (self chunkEscape: aClass comment), '!!';lf].
  115. aStream lf
  116. !
  117. exportMetaDefinitionOf: aClass on: aStream
  118. aClass class instanceVariableNames isEmpty ifFalse: [
  119. aStream
  120. nextPutAll: (self classNameFor: aClass class);
  121. nextPutAll: ' instanceVariableNames: '''.
  122. aClass class instanceVariableNames
  123. do: [:each | aStream nextPutAll: each]
  124. separatedBy: [aStream nextPutAll: ' '].
  125. aStream
  126. nextPutAll: '''!!'; lf; lf]
  127. !
  128. exportMethod: aMethod on: aStream
  129. aStream
  130. lf; lf; nextPutAll: (self chunkEscape: aMethod source); lf;
  131. nextPutAll: '!!'
  132. !
  133. exportPackageDefinitionOf: aPackage on: aStream
  134. "Chunk format."
  135. aStream
  136. nextPutAll: 'Smalltalk current createPackage: ''', aPackage name, '''!!';
  137. lf
  138. ! !
  139. !ChunkExporter class methodsFor: 'exporting-accessing'!
  140. extensionCategoriesOfPackage: package
  141. "Issue #143: sort protocol alphabetically"
  142. | name map result |
  143. name := package name.
  144. result := OrderedCollection new.
  145. (Package sortedClasses: Smalltalk current classes) do: [:each |
  146. {each. each class} do: [:aClass |
  147. map := Dictionary new.
  148. aClass protocolsDo: [:category :methods |
  149. (category match: '^\*', name) ifTrue: [ map at: category put: methods ]].
  150. result addAll: ((map keys sorted: [:a :b | a <= b ]) collect: [:category |
  151. MethodCategory name: category theClass: aClass methods: (map at: category)]) ]].
  152. ^result
  153. !
  154. methodsOfCategory: category
  155. "Issue #143: sort methods alphabetically"
  156. ^(category methods) sorted: [:a :b | a selector <= b selector]
  157. !
  158. ownCategoriesOfClass: aClass
  159. "Issue #143: sort protocol alphabetically"
  160. | map |
  161. map := Dictionary new.
  162. aClass protocolsDo: [:category :methods |
  163. (category match: '^\*') ifFalse: [ map at: category put: methods ]].
  164. ^(map keys sorted: [:a :b | a <= b ]) collect: [:category |
  165. MethodCategory name: category theClass: aClass methods: (map at: category) ]
  166. !
  167. ownCategoriesOfMetaClass: aClass
  168. "Issue #143: sort protocol alphabetically"
  169. ^self ownCategoriesOfClass: aClass class
  170. ! !
  171. !ChunkExporter class methodsFor: 'exporting-output'!
  172. exportCategoryEpilogueOf: category on: aStream
  173. aStream nextPutAll: ' !!'; lf; lf
  174. !
  175. exportCategoryPrologueOf: category on: aStream
  176. aStream
  177. nextPutAll: '!!', (self classNameFor: category theClass);
  178. nextPutAll: ' methodsFor: ''', category name, '''!!'
  179. !
  180. exportDefinitionOf: aClass on: aStream
  181. "Chunk format."
  182. aStream
  183. nextPutAll: (self classNameFor: aClass superclass);
  184. nextPutAll: ' subclass: #', (self classNameFor: aClass); lf;
  185. tab; nextPutAll: 'instanceVariableNames: '''.
  186. aClass instanceVariableNames
  187. do: [:each | aStream nextPutAll: each]
  188. separatedBy: [aStream nextPutAll: ' '].
  189. aStream
  190. nextPutAll: ''''; lf;
  191. tab; nextPutAll: 'package: ''', aClass category, '''!!'; lf.
  192. aClass comment notEmpty ifTrue: [
  193. aStream
  194. nextPutAll: '!!', (self classNameFor: aClass), ' commentStamp!!';lf;
  195. nextPutAll: (self chunkEscape: aClass comment), '!!';lf].
  196. aStream lf
  197. !
  198. exportMetaDefinitionOf: aClass on: aStream
  199. aClass class instanceVariableNames isEmpty ifFalse: [
  200. aStream
  201. nextPutAll: (self classNameFor: aClass class);
  202. nextPutAll: ' instanceVariableNames: '''.
  203. aClass class instanceVariableNames
  204. do: [:each | aStream nextPutAll: each]
  205. separatedBy: [aStream nextPutAll: ' '].
  206. aStream
  207. nextPutAll: '''!!'; lf; lf]
  208. !
  209. exportMethod: aMethod on: aStream
  210. aStream
  211. lf; lf; nextPutAll: (self chunkEscape: aMethod source); lf;
  212. nextPutAll: '!!'
  213. !
  214. exportPackageDefinitionOf: package on: aStream
  215. "Chunk format."
  216. aStream
  217. nextPutAll: 'Smalltalk current createPackage: ''', package name, '''!!';
  218. lf
  219. ! !
  220. !ChunkExporter class methodsFor: 'fileOut'!
  221. recipe
  222. "Export a given package."
  223. | exportCategoryRecipe |
  224. exportCategoryRecipe := {
  225. self -> #exportCategoryPrologueOf:on:.
  226. {
  227. self -> #methodsOfCategory:.
  228. self -> #exportMethod:on: }.
  229. self -> #exportCategoryEpilogueOf:on: }.
  230. ^{
  231. self -> #exportPackageDefinitionOf:on:.
  232. {
  233. PluggableExporter -> #ownClassesOfPackage:.
  234. self -> #exportDefinitionOf:on:.
  235. { self -> #ownCategoriesOfClass: }, exportCategoryRecipe.
  236. self -> #exportMetaDefinitionOf:on:.
  237. { self -> #ownCategoriesOfMetaClass: }, exportCategoryRecipe }.
  238. { self -> #extensionCategoriesOfPackage: }, exportCategoryRecipe
  239. }
  240. ! !
  241. !ChunkExporter class methodsFor: 'private'!
  242. chunkEscape: aString
  243. "Replace all occurrences of !! with !!!! and trim at both ends."
  244. ^(aString replace: '!!' with: '!!!!') trimBoth
  245. !
  246. classNameFor: aClass
  247. ^aClass isMetaclass
  248. ifTrue: [aClass instanceClass name, ' class']
  249. ifFalse: [
  250. aClass isNil
  251. ifTrue: ['nil']
  252. ifFalse: [aClass name]]
  253. ! !
  254. AbstractExporter subclass: #Exporter
  255. instanceVariableNames: ''
  256. package: 'Importer-Exporter'!
  257. !Exporter commentStamp!
  258. I am responsible for outputting Amber code into a JavaScript string.
  259. The generated output is enough to reconstruct the exported data, including Smalltalk source code and other metadata.
  260. ## Use case
  261. I am typically used to save code outside of the Amber runtime (committing to disk, etc.).
  262. ## API
  263. Use `#exportAll`, `#exportClass:` or `#exportPackage:` methods.!
  264. !Exporter methodsFor: 'accessing'!
  265. extensionMethodsOfPackage: aPackage
  266. "Issue #143: sort classes and methods alphabetically"
  267. | name result |
  268. name := aPackage name.
  269. result := OrderedCollection new.
  270. (Package sortedClasses: Smalltalk current classes) do: [:each |
  271. {each. each class} do: [:aClass |
  272. result addAll: (((aClass methodDictionary values)
  273. sorted: [:a :b | a selector <= b selector])
  274. select: [:method | method category match: '^\*', name]) ]].
  275. ^result
  276. !
  277. ownMethodsOfClass: aClass
  278. "Issue #143: sort methods alphabetically"
  279. ^((aClass methodDictionary values) sorted: [:a :b | a selector <= b selector])
  280. reject: [:each | (each category match: '^\*')]
  281. !
  282. ownMethodsOfMetaClass: aClass
  283. "Issue #143: sort methods alphabetically"
  284. ^self ownMethodsOfClass: aClass class
  285. ! !
  286. !Exporter methodsFor: 'convenience'!
  287. classNameFor: aClass
  288. ^aClass isMetaclass
  289. ifTrue: [ aClass instanceClass name, '.klass' ]
  290. ifFalse: [
  291. aClass isNil
  292. ifTrue: [ 'nil' ]
  293. ifFalse: [ aClass name ] ]
  294. ! !
  295. !Exporter methodsFor: 'fileOut'!
  296. amdRecipe
  297. "Export a given package with amd transport type."
  298. | legacy result |
  299. legacy := self recipe.
  300. result := (legacy copyFrom: 1 to: 2),
  301. { AmdExporter -> #exportPackageTransportOf:on: },
  302. (legacy copyFrom: 3 to: legacy size).
  303. result first key: AmdExporter.
  304. result last key: AmdExporter.
  305. ^result
  306. !
  307. recipe
  308. "Export a given package."
  309. ^{
  310. self -> #exportPackagePrologueOf:on:.
  311. self -> #exportPackageDefinitionOf:on:.
  312. {
  313. PluggableExporter -> #ownClassesOfPackage:.
  314. self -> #exportDefinitionOf:on:.
  315. {
  316. self -> #ownMethodsOfClass:.
  317. self -> #exportMethod:on: }.
  318. self -> #exportMetaDefinitionOf:on:.
  319. {
  320. self -> #ownMethodsOfMetaClass:.
  321. self -> #exportMethod:on: } }.
  322. {
  323. self -> #extensionMethodsOfPackage:.
  324. self -> #exportMethod:on: }.
  325. self -> #exportPackageEpilogueOf:on:
  326. }
  327. ! !
  328. !Exporter methodsFor: 'output'!
  329. exportDefinitionOf: aClass on: aStream
  330. aStream
  331. lf;
  332. nextPutAll: 'smalltalk.addClass(';
  333. nextPutAll: '''', (self classNameFor: aClass), ''', ';
  334. nextPutAll: 'smalltalk.', (self classNameFor: aClass superclass);
  335. nextPutAll: ', ['.
  336. aClass instanceVariableNames
  337. do: [:each | aStream nextPutAll: '''', each, '''']
  338. separatedBy: [aStream nextPutAll: ', '].
  339. aStream
  340. nextPutAll: '], ''';
  341. nextPutAll: aClass category, '''';
  342. nextPutAll: ');'.
  343. aClass comment notEmpty ifTrue: [
  344. aStream
  345. lf;
  346. nextPutAll: 'smalltalk.';
  347. nextPutAll: (self classNameFor: aClass);
  348. nextPutAll: '.comment=';
  349. nextPutAll: aClass comment asJavascript;
  350. nextPutAll: ';'].
  351. aStream lf
  352. !
  353. exportMetaDefinitionOf: aClass on: aStream
  354. aStream lf.
  355. aClass class instanceVariableNames isEmpty ifFalse: [
  356. aStream
  357. nextPutAll: 'smalltalk.', (self classNameFor: aClass class);
  358. nextPutAll: '.iVarNames = ['.
  359. aClass class instanceVariableNames
  360. do: [:each | aStream nextPutAll: '''', each, '''']
  361. separatedBy: [aStream nextPutAll: ','].
  362. aStream nextPutAll: '];', String lf]
  363. !
  364. exportMethod: aMethod on: aStream
  365. aStream
  366. nextPutAll: 'smalltalk.addMethod(';lf;
  367. "nextPutAll: aMethod selector asSelector asJavascript, ',';lf;"
  368. nextPutAll: 'smalltalk.method({';lf;
  369. nextPutAll: 'selector: ', aMethod selector asJavascript, ',';lf;
  370. nextPutAll: 'category: ''', aMethod category, ''',';lf;
  371. nextPutAll: 'fn: ', aMethod fn compiledSource, ',';lf;
  372. nextPutAll: 'args: ', aMethod arguments asJavascript, ','; lf;
  373. nextPutAll: 'source: ', aMethod source asJavascript, ',';lf;
  374. nextPutAll: 'messageSends: ', aMethod messageSends asJavascript, ',';lf;
  375. nextPutAll: 'referencedClasses: ', aMethod referencedClasses asJavascript.
  376. aStream
  377. lf;
  378. nextPutAll: '}),';lf;
  379. nextPutAll: 'smalltalk.', (self classNameFor: aMethod methodClass);
  380. nextPutAll: ');';lf;lf
  381. !
  382. exportPackageDefinitionOf: aPackage on: aStream
  383. aStream
  384. nextPutAll: 'smalltalk.addPackage(';
  385. nextPutAll: '''', aPackage name, ''');';
  386. lf
  387. !
  388. exportPackageEpilogueOf: aPackage on: aStream
  389. aStream
  390. nextPutAll: '})(global_smalltalk,global_nil,global__st);';
  391. lf
  392. !
  393. exportPackagePrologueOf: aPackage on: aStream
  394. aStream
  395. nextPutAll: '(function(smalltalk,nil,_st){';
  396. lf
  397. ! !
  398. !Exporter class methodsFor: 'exporting-accessing'!
  399. extensionMethodsOfPackage: package
  400. "Issue #143: sort classes and methods alphabetically"
  401. | name result |
  402. name := package name.
  403. result := OrderedCollection new.
  404. (Package sortedClasses: Smalltalk current classes) do: [:each |
  405. {each. each class} do: [:aClass |
  406. result addAll: (((aClass methodDictionary values)
  407. sorted: [:a :b | a selector <= b selector])
  408. select: [:method | method category match: '^\*', name]) ]].
  409. ^result
  410. !
  411. ownMethodsOfClass: aClass
  412. "Issue #143: sort methods alphabetically"
  413. ^((aClass methodDictionary values) sorted: [:a :b | a selector <= b selector])
  414. reject: [:each | (each category match: '^\*')]
  415. !
  416. ownMethodsOfMetaClass: aClass
  417. "Issue #143: sort methods alphabetically"
  418. ^self ownMethodsOfClass: aClass class
  419. ! !
  420. !Exporter class methodsFor: 'exporting-output'!
  421. exportDefinitionOf: aClass on: aStream
  422. aStream
  423. lf;
  424. nextPutAll: 'smalltalk.addClass(';
  425. nextPutAll: '''', (self classNameFor: aClass), ''', ';
  426. nextPutAll: 'smalltalk.', (self classNameFor: aClass superclass);
  427. nextPutAll: ', ['.
  428. aClass instanceVariableNames
  429. do: [:each | aStream nextPutAll: '''', each, '''']
  430. separatedBy: [aStream nextPutAll: ', '].
  431. aStream
  432. nextPutAll: '], ''';
  433. nextPutAll: aClass category, '''';
  434. nextPutAll: ');'.
  435. aClass comment notEmpty ifTrue: [
  436. aStream
  437. lf;
  438. nextPutAll: 'smalltalk.';
  439. nextPutAll: (self classNameFor: aClass);
  440. nextPutAll: '.comment=';
  441. nextPutAll: aClass comment asJavascript;
  442. nextPutAll: ';'].
  443. aStream lf
  444. !
  445. exportMetaDefinitionOf: aClass on: aStream
  446. aStream lf.
  447. aClass class instanceVariableNames isEmpty ifFalse: [
  448. aStream
  449. nextPutAll: 'smalltalk.', (self classNameFor: aClass class);
  450. nextPutAll: '.iVarNames = ['.
  451. aClass class instanceVariableNames
  452. do: [:each | aStream nextPutAll: '''', each, '''']
  453. separatedBy: [aStream nextPutAll: ','].
  454. aStream nextPutAll: '];', String lf]
  455. !
  456. exportMethod: aMethod on: aStream
  457. aStream
  458. nextPutAll: 'smalltalk.addMethod(';lf;
  459. "nextPutAll: aMethod selector asSelector asJavascript, ',';lf;"
  460. nextPutAll: 'smalltalk.method({';lf;
  461. nextPutAll: 'selector: ', aMethod selector asJavascript, ',';lf;
  462. nextPutAll: 'category: ''', aMethod category, ''',';lf;
  463. nextPutAll: 'fn: ', aMethod fn compiledSource, ',';lf;
  464. nextPutAll: 'args: ', aMethod arguments asJavascript, ','; lf;
  465. nextPutAll: 'source: ', aMethod source asJavascript, ',';lf;
  466. nextPutAll: 'messageSends: ', aMethod messageSends asJavascript, ',';lf;
  467. nextPutAll: 'referencedClasses: ', aMethod referencedClasses asJavascript.
  468. aStream
  469. lf;
  470. nextPutAll: '}),';lf;
  471. nextPutAll: 'smalltalk.', (self classNameFor: aMethod methodClass);
  472. nextPutAll: ');';lf;lf
  473. !
  474. exportPackageDefinitionOf: package on: aStream
  475. aStream
  476. nextPutAll: 'smalltalk.addPackage(';
  477. nextPutAll: '''', package name, ''');';
  478. lf
  479. !
  480. exportPackageEpilogueOf: aPackage on: aStream
  481. aStream
  482. nextPutAll: '})(global_smalltalk,global_nil,global__st);';
  483. lf
  484. !
  485. exportPackagePrologueOf: aPackage on: aStream
  486. aStream
  487. nextPutAll: '(function(smalltalk,nil,_st){';
  488. lf
  489. ! !
  490. !Exporter class methodsFor: 'fileOut'!
  491. recipe
  492. "Export a given package."
  493. ^{
  494. AmdExporter -> #exportPackagePrologueOf:on:.
  495. self -> #exportPackageDefinitionOf:on:.
  496. {
  497. PluggableExporter -> #ownClassesOfPackage:.
  498. self -> #exportDefinitionOf:on:.
  499. {
  500. self -> #ownMethodsOfClass:.
  501. self -> #exportMethod:on: }.
  502. self -> #exportMetaDefinitionOf:on:.
  503. {
  504. self -> #ownMethodsOfMetaClass:.
  505. self -> #exportMethod:on: } }.
  506. {
  507. self -> #extensionMethodsOfPackage:.
  508. self -> #exportMethod:on: }.
  509. AmdExporter -> #exportPackageEpilogueOf:on:
  510. }
  511. ! !
  512. !Exporter class methodsFor: 'private'!
  513. classNameFor: aClass
  514. ^aClass isMetaclass
  515. ifTrue: [aClass instanceClass name, '.klass']
  516. ifFalse: [
  517. aClass isNil
  518. ifTrue: ['nil']
  519. ifFalse: [aClass name]]
  520. ! !
  521. Exporter subclass: #StrippedExporter
  522. instanceVariableNames: ''
  523. package: 'Importer-Exporter'!
  524. !StrippedExporter commentStamp!
  525. I export Amber code into a JavaScript string, but without any optional associated data like the Amber source code.!
  526. !StrippedExporter methodsFor: 'output'!
  527. exportDefinitionOf: aClass on: aStream
  528. aStream
  529. lf;
  530. nextPutAll: 'smalltalk.addClass(';
  531. nextPutAll: '''', (self classNameFor: aClass), ''', ';
  532. nextPutAll: 'smalltalk.', (self classNameFor: aClass superclass);
  533. nextPutAll: ', ['.
  534. aClass instanceVariableNames
  535. do: [:each | aStream nextPutAll: '''', each, '''']
  536. separatedBy: [aStream nextPutAll: ', '].
  537. aStream
  538. nextPutAll: '], ''';
  539. nextPutAll: aClass category, '''';
  540. nextPutAll: ');'.
  541. aStream lf
  542. !
  543. exportMethod: aMethod on: aStream
  544. aStream
  545. nextPutAll: 'smalltalk.addMethod(';lf;
  546. "nextPutAll: aMethod selector asSelector asJavascript, ',';lf;"
  547. nextPutAll: 'smalltalk.method({';lf;
  548. nextPutAll: 'selector: ', aMethod selector asJavascript, ',';lf;
  549. nextPutAll: 'fn: ', aMethod fn compiledSource, ',';lf;
  550. nextPutAll: 'messageSends: ', aMethod messageSends asJavascript;
  551. nextPutAll: '}),';lf;
  552. nextPutAll: 'smalltalk.', (self classNameFor: aMethod methodClass);
  553. nextPutAll: ');';lf;lf
  554. ! !
  555. !StrippedExporter class methodsFor: 'exporting-output'!
  556. exportDefinitionOf: aClass on: aStream
  557. aStream
  558. lf;
  559. nextPutAll: 'smalltalk.addClass(';
  560. nextPutAll: '''', (self classNameFor: aClass), ''', ';
  561. nextPutAll: 'smalltalk.', (self classNameFor: aClass superclass);
  562. nextPutAll: ', ['.
  563. aClass instanceVariableNames
  564. do: [:each | aStream nextPutAll: '''', each, '''']
  565. separatedBy: [aStream nextPutAll: ', '].
  566. aStream
  567. nextPutAll: '], ''';
  568. nextPutAll: aClass category, '''';
  569. nextPutAll: ');'.
  570. aStream lf
  571. !
  572. exportMethod: aMethod on: aStream
  573. aStream
  574. nextPutAll: 'smalltalk.addMethod(';lf;
  575. "nextPutAll: aMethod selector asSelector asJavascript, ',';lf;"
  576. nextPutAll: 'smalltalk.method({';lf;
  577. nextPutAll: 'selector: ', aMethod selector asJavascript, ',';lf;
  578. nextPutAll: 'fn: ', aMethod fn compiledSource, ',';lf;
  579. nextPutAll: 'messageSends: ', aMethod messageSends asJavascript;
  580. nextPutAll: '}),';lf;
  581. nextPutAll: 'smalltalk.', (self classNameFor: aMethod methodClass);
  582. nextPutAll: ');';lf;lf
  583. ! !
  584. Object subclass: #AmdExporter
  585. instanceVariableNames: ''
  586. package: 'Importer-Exporter'!
  587. !AmdExporter class methodsFor: 'exporting-output'!
  588. exportPackageEpilogueOf: aPackage on: aStream
  589. aStream
  590. nextPutAll: '});';
  591. lf
  592. !
  593. exportPackagePrologueOf: aPackage on: aStream
  594. aStream
  595. nextPutAll: 'define("';
  596. nextPutAll: (aPackage amdNamespace ifNil: [ 'amber' ]); "ifNil: only for LegacyPH, it should not happen with AmdPH"
  597. nextPutAll: '/';
  598. nextPutAll: aPackage name;
  599. nextPutAll: '", ';
  600. nextPutAll: (#('amber_vm/smalltalk' 'amber_vm/nil' 'amber_vm/_st'), (self amdNamesOfPackages: aPackage loadDependencies)) asJavascript;
  601. nextPutAll: ', function(smalltalk,nil,_st){';
  602. lf
  603. !
  604. exportPackageTransportOf: aPackage on: aStream
  605. aStream
  606. nextPutAll: 'smalltalk.packages[';
  607. nextPutAll: aPackage name asJavascript;
  608. nextPutAll: '].transport = ';
  609. nextPutAll: aPackage transportJson;
  610. nextPutAll: ';';
  611. lf
  612. ! !
  613. !AmdExporter class methodsFor: 'private'!
  614. amdNamesOfPackages: anArray
  615. | deps depNames |
  616. ^(anArray
  617. select: [ :each | each amdNamespace notNil ])
  618. collect: [ :each | each amdNamespace, '/', each name ]
  619. ! !
  620. Object subclass: #ChunkParser
  621. instanceVariableNames: 'stream'
  622. package: 'Importer-Exporter'!
  623. !ChunkParser commentStamp!
  624. I am responsible for parsing aStream contents in the chunk format.
  625. ## API
  626. ChunkParser new
  627. stream: aStream;
  628. nextChunk!
  629. !ChunkParser methodsFor: 'accessing'!
  630. stream: aStream
  631. stream := aStream
  632. ! !
  633. !ChunkParser methodsFor: 'reading'!
  634. nextChunk
  635. "The chunk format (Smalltalk Interchange Format or Fileout format)
  636. is a trivial format but can be a bit tricky to understand:
  637. - Uses the exclamation mark as delimiter of chunks.
  638. - Inside a chunk a normal exclamation mark must be doubled.
  639. - A non empty chunk must be a valid Smalltalk expression.
  640. - A chunk on top level with a preceding empty chunk is an instruction chunk:
  641. - The object created by the expression then takes over reading chunks.
  642. This metod returns next chunk as a String (trimmed), empty String (all whitespace) or nil."
  643. | char result chunk |
  644. result := '' writeStream.
  645. [char := stream next.
  646. char notNil] whileTrue: [
  647. char = '!!' ifTrue: [
  648. stream peek = '!!'
  649. ifTrue: [stream next "skipping the escape double"]
  650. ifFalse: [^result contents trimBoth "chunk end marker found"]].
  651. result nextPut: char].
  652. ^nil "a chunk needs to end with !!"
  653. ! !
  654. !ChunkParser class methodsFor: 'not yet classified'!
  655. on: aStream
  656. ^self new stream: aStream
  657. ! !
  658. Object subclass: #Importer
  659. instanceVariableNames: ''
  660. package: 'Importer-Exporter'!
  661. !Importer commentStamp!
  662. I can import Amber code from a string in the chunk format.
  663. ## API
  664. Importer new import: aString!
  665. !Importer methodsFor: 'fileIn'!
  666. import: aStream
  667. | chunk result parser lastEmpty |
  668. parser := ChunkParser on: aStream.
  669. lastEmpty := false.
  670. [chunk := parser nextChunk.
  671. chunk isNil] whileFalse: [
  672. chunk isEmpty
  673. ifTrue: [lastEmpty := true]
  674. ifFalse: [
  675. result := Compiler new evaluateExpression: chunk.
  676. lastEmpty
  677. ifTrue: [
  678. lastEmpty := false.
  679. result scanFrom: parser]]]
  680. ! !
  681. Object subclass: #MethodCategory
  682. instanceVariableNames: 'methods name theClass'
  683. package: 'Importer-Exporter'!
  684. !MethodCategory commentStamp!
  685. I am an abstraction for a method category in a class / metaclass.
  686. I know of my class, name and methods.
  687. I am used when exporting a package.!
  688. !MethodCategory methodsFor: 'accessing'!
  689. methods
  690. ^methods
  691. !
  692. methods: aCollection
  693. methods := aCollection
  694. !
  695. name
  696. ^name
  697. !
  698. name: aString
  699. name := aString
  700. !
  701. theClass
  702. ^theClass
  703. !
  704. theClass: aClass
  705. theClass := aClass
  706. ! !
  707. !MethodCategory class methodsFor: 'not yet classified'!
  708. name: aString theClass: aClass methods: anArray
  709. ^self new
  710. name: aString;
  711. theClass: aClass;
  712. methods: anArray;
  713. yourself
  714. ! !
  715. InterfacingObject subclass: #PackageHandler
  716. instanceVariableNames: ''
  717. package: 'Importer-Exporter'!
  718. !PackageHandler commentStamp!
  719. I am responsible for handling package loading and committing.
  720. I should not be used directly. Instead, use the corresponding `Package` methods.!
  721. !PackageHandler methodsFor: 'committing'!
  722. commit: aPackage
  723. self commitChannels
  724. do: [ :commitStrategyFactory || fileContents commitStrategy |
  725. commitStrategy := commitStrategyFactory value: aPackage.
  726. fileContents := String streamContents: [ :stream |
  727. (PluggableExporter forRecipe: commitStrategy key) exportPackage: aPackage on: stream ].
  728. self ajaxPutAt: commitStrategy value data: fileContents ]
  729. displayingProgress: 'Committing package ', aPackage name
  730. !
  731. commitChannels
  732. self subclassResponsibility
  733. ! !
  734. !PackageHandler methodsFor: 'private'!
  735. ajaxPutAt: aURL data: aString
  736. self
  737. ajax: #{
  738. 'url' -> aURL.
  739. 'type' -> 'PUT'.
  740. 'data' -> aString.
  741. 'contentType' -> 'text/plain;charset=UTF-8'.
  742. 'error' -> [ :xhr | self error: 'Commiting ' , aURL , ' failed with reason: "' , (xhr responseText) , '"'] }
  743. ! !
  744. PackageHandler class instanceVariableNames: 'registry'!
  745. !PackageHandler class methodsFor: 'accessing'!
  746. classRegisteredFor: aString
  747. ^registry at: aString
  748. !
  749. for: aString
  750. ^(self classRegisteredFor: aString) new
  751. ! !
  752. !PackageHandler class methodsFor: 'initialization'!
  753. initialize
  754. super initialize.
  755. registry := #{}
  756. ! !
  757. !PackageHandler class methodsFor: 'registry'!
  758. register: aClass for: aString
  759. registry at: aString put: aClass
  760. !
  761. registerFor: aString
  762. PackageHandler register: self for: aString
  763. ! !
  764. PackageHandler subclass: #AmdPackageHandler
  765. instanceVariableNames: ''
  766. package: 'Importer-Exporter'!
  767. !AmdPackageHandler commentStamp!
  768. I am responsible for handling package loading and committing.
  769. I should not be used directly. Instead, use the corresponding `Package` methods.!
  770. !AmdPackageHandler methodsFor: 'committing'!
  771. commitChannels
  772. ^{
  773. [ :pkg | Exporter default amdRecipe -> (pkg commitPathJs, '/', pkg name, '.js') ].
  774. [ :pkg | StrippedExporter default amdRecipe -> (pkg commitPathJs, '/', pkg name, '.deploy.js') ].
  775. [ :pkg | ChunkExporter default recipe -> (pkg commitPathSt, '/', pkg name, '.st') ]
  776. }
  777. !
  778. commitPathJsFor: aPackage
  779. ^self toUrl: (self namespaceFor: aPackage)
  780. !
  781. commitPathStFor: aPackage
  782. "if _source is not mapped, .st commit will likely fail"
  783. ^self toUrl: (self namespaceFor: aPackage), '/_source'.
  784. !
  785. namespaceFor: aPackage
  786. ^aPackage amdNamespace
  787. ifNil: [ aPackage amdNamespace: self class defaultNamespace; amdNamespace ]
  788. ! !
  789. !AmdPackageHandler methodsFor: 'private'!
  790. toUrl: aString
  791. (Smalltalk current at: '_amd_require')
  792. ifNil: [ self error: 'AMD loader not present' ]
  793. ifNotNil: [ :require | ^(require basicAt: 'toUrl') value: aString ]
  794. ! !
  795. AmdPackageHandler class instanceVariableNames: 'defaultNamespace'!
  796. !AmdPackageHandler class methodsFor: 'commit paths'!
  797. commitPathsFromLoader
  798. "TODO"
  799. !
  800. defaultNamespace
  801. ^ defaultNamespace ifNil: [ self error: 'AMD default namespace not set.' ]
  802. !
  803. defaultNamespace: aString
  804. defaultNamespace := aString
  805. !
  806. resetCommitPaths
  807. defaultNamespace := nil
  808. ! !
  809. !AmdPackageHandler class methodsFor: 'initialization'!
  810. initialize
  811. super initialize.
  812. self registerFor: 'amd'.
  813. self commitPathsFromLoader
  814. ! !
  815. PackageHandler subclass: #LegacyPackageHandler
  816. instanceVariableNames: ''
  817. package: 'Importer-Exporter'!
  818. !LegacyPackageHandler commentStamp!
  819. I am responsible for handling package loading and committing.
  820. I should not be used directly. Instead, use the corresponding `Package` methods.!
  821. !LegacyPackageHandler methodsFor: 'committing'!
  822. commitChannels
  823. ^{
  824. [ :pkg | Exporter default recipe -> (pkg commitPathJs, '/', pkg name, '.js') ].
  825. [ :pkg | StrippedExporter default recipe -> (pkg commitPathJs, '/', pkg name, '.deploy.js') ].
  826. [ :pkg | ChunkExporter default recipe -> (pkg commitPathSt, '/', pkg name, '.st') ]
  827. }
  828. !
  829. commitPathJsFor: aPackage
  830. ^self class defaultCommitPathJs
  831. !
  832. commitPathStFor: aPackage
  833. ^self class defaultCommitPathSt
  834. ! !
  835. !LegacyPackageHandler methodsFor: 'loading'!
  836. loadPackage: packageName prefix: aString
  837. | url |
  838. url := '/', aString, '/js/', packageName, '.js'.
  839. self
  840. ajax: #{
  841. 'url' -> url.
  842. 'type' -> 'GET'.
  843. 'dataType' -> 'script'.
  844. 'complete' -> [ :jqXHR :textStatus |
  845. jqXHR readyState = 4
  846. ifTrue: [ self setupPackageNamed: packageName prefix: aString ] ].
  847. 'error' -> [ self alert: 'Could not load package at: ', url ]
  848. }
  849. !
  850. loadPackages: aCollection prefix: aString
  851. aCollection do: [ :each |
  852. self loadPackage: each prefix: aString ]
  853. ! !
  854. !LegacyPackageHandler methodsFor: 'private'!
  855. setupPackageNamed: packageName prefix: aString
  856. (Package named: packageName)
  857. setupClasses;
  858. commitPathJs: '/', aString, '/js';
  859. commitPathSt: '/', aString, '/st'
  860. ! !
  861. LegacyPackageHandler class instanceVariableNames: 'defaultCommitPathJs defaultCommitPathSt'!
  862. !LegacyPackageHandler class methodsFor: 'commit paths'!
  863. commitPathsFromLoader
  864. <
  865. var commitPath = typeof amber !!== 'undefined' && amber.commitPath;
  866. if (!!commitPath) return;
  867. if (commitPath.js) self._defaultCommitPathJs_(commitPath.js);
  868. if (commitPath.st) self._defaultCommitPathSt_(commitPath.st);
  869. >
  870. !
  871. defaultCommitPathJs
  872. ^ defaultCommitPathJs ifNil: [ defaultCommitPathJs := 'js']
  873. !
  874. defaultCommitPathJs: aString
  875. defaultCommitPathJs := aString
  876. !
  877. defaultCommitPathSt
  878. ^ defaultCommitPathSt ifNil: [ defaultCommitPathSt := 'st']
  879. !
  880. defaultCommitPathSt: aString
  881. defaultCommitPathSt := aString
  882. !
  883. resetCommitPaths
  884. defaultCommitPathJs := nil.
  885. defaultCommitPathSt := nil
  886. ! !
  887. !LegacyPackageHandler class methodsFor: 'initialization'!
  888. initialize
  889. super initialize.
  890. self registerFor: 'unknown'.
  891. self commitPathsFromLoader
  892. ! !
  893. !LegacyPackageHandler class methodsFor: 'loading'!
  894. loadPackages: aCollection prefix: aString
  895. ^ self new loadPackages: aCollection prefix: aString
  896. ! !
  897. Object subclass: #PluggableExporter
  898. instanceVariableNames: 'recipe'
  899. package: 'Importer-Exporter'!
  900. !PluggableExporter commentStamp!
  901. I am an engine for exporting structured data on a Stream.
  902. My instances are created using
  903. PluggableExporter newUsing: recipe,
  904. where recipe is structured description of the exporting algorithm.
  905. The actual exporting is done by interpreting the recipe using a `RecipeInterpreter`.
  906. I am used to export amber packages, so I have a convenience method
  907. `exportPackage: aPackage on: aStream`
  908. which exports `aPackage` using the `recipe`
  909. (it is otherwise no special, so it may be renamed to export:on:)!
  910. !PluggableExporter methodsFor: 'accessing'!
  911. interpreter
  912. ^ RecipeInterpreter new
  913. !
  914. recipe
  915. ^recipe
  916. !
  917. recipe: anArray
  918. recipe := anArray
  919. ! !
  920. !PluggableExporter methodsFor: 'fileOut'!
  921. exportAllPackages
  922. "Export all packages in the system."
  923. ^String streamContents: [:stream |
  924. Smalltalk current packages do: [:pkg |
  925. self exportPackage: pkg on: stream]]
  926. !
  927. exportPackage: aPackage on: aStream
  928. self interpreter interpret: self recipe for: aPackage on: aStream
  929. ! !
  930. !PluggableExporter class methodsFor: 'convenience'!
  931. ownClassesOfPackage: package
  932. "Export classes in dependency order.
  933. Update (issue #171): Remove duplicates for export"
  934. ^package sortedClasses asSet
  935. ! !
  936. !PluggableExporter class methodsFor: 'instance creation'!
  937. forRecipe: aRecipe
  938. ^self new recipe: aRecipe; yourself
  939. ! !
  940. Object subclass: #RecipeInterpreter
  941. instanceVariableNames: ''
  942. package: 'Importer-Exporter'!
  943. !RecipeInterpreter commentStamp!
  944. I am an interpreter for export recipes.
  945. ## Recipe format
  946. Recipe is an array, which can contain two kinds of elements:
  947. - an assocation where the key is the receiver and the value is a two-arguments selector
  948. In this case, `receiver perform: selector withArguments: { data. stream }` is called.
  949. This essentially defines one step of export process.
  950. The key (eg. receiver) is presumed to be some kind of 'repository' of the exporting methods
  951. that just format appropriate aspect of data into a stream; like a class or a singleton,
  952. so that the recipe itself can be decoupled from data.
  953. - a subarray, where first element is special and the rest is recursive recipe.
  954. `subarray first` must be an association similar to one above,
  955. with key being the 'repository' receiver, but value is one-arg selector.
  956. In this case, `receiver perform: selector withArguments: { data }` should create a collection.
  957. Then, the sub-recipe (`subarray allButFirst`) is applied to every element of a collection, eg.
  958. collection do: [ :each | self export: each using: sa allButFirst on: stream ]!
  959. !RecipeInterpreter methodsFor: 'accessing'!
  960. recipe
  961. ^recipe
  962. !
  963. recipe: anArray
  964. recipe := anArray
  965. ! !
  966. !RecipeInterpreter methodsFor: 'interpreting'!
  967. interpret: aRecipe for: anObject on: aStream
  968. | recipeStream |
  969. recipeStream := aRecipe readStream.
  970. [ recipeStream atEnd ] whileFalse: [
  971. self
  972. interpretStep: recipeStream next
  973. for: anObject
  974. on: aStream ]
  975. !
  976. interpretStep: aRecipeStep for: anObject on: aStream
  977. aRecipeStep value == aRecipeStep ifTrue: [
  978. ^ self interpretSubRecipe: aRecipeStep for: anObject on: aStream ].
  979. aRecipeStep key perform: aRecipeStep value withArguments: { anObject. aStream }
  980. !
  981. interpretSubRecipe: aRecipe for: anObject on: aStream
  982. | selection |
  983. selection := aRecipe first key
  984. perform: aRecipe first value
  985. withArguments: { anObject }.
  986. selection do: [ :each |
  987. self interpret: aRecipe allButFirst for: each on: aStream ]
  988. ! !
  989. !Package methodsFor: '*Importer-Exporter'!
  990. amdNamespace
  991. <return (self.transport && self.transport.amdNamespace) || nil>
  992. !
  993. amdNamespace: aString
  994. <
  995. if (!!self.transport) { self.transport = { type: 'amd' }; }
  996. if (self.transport.type !!== 'amd') { throw new Error('Package '+self._name()+' has transport type '+self.transport.type+', not "amd".'); }
  997. self.transport.amdNamespace = aString;
  998. >
  999. !
  1000. commit
  1001. ^ self transport commit: self
  1002. !
  1003. commitPathJs
  1004. ^ (extension ifNil: [ extension := #{} ]) at: #commitPathJs ifAbsent: [self transport commitPathJsFor: self]
  1005. !
  1006. commitPathJs: aString
  1007. ^ (extension ifNil: [ extension := #{} ]) at: #commitPathJs put: aString
  1008. !
  1009. commitPathSt
  1010. ^ (extension ifNil: [ extension := #{} ]) at: #commitPathSt ifAbsent: [self transport commitPathStFor: self]
  1011. !
  1012. commitPathSt: aString
  1013. ^ (extension ifNil: [ extension := #{} ]) at: #commitPathSt put: aString
  1014. !
  1015. transport
  1016. ^ PackageHandler for: self transportType
  1017. !
  1018. transportJson
  1019. <return JSON.stringify(self.transport || null);>
  1020. !
  1021. transportType
  1022. <return (self.transport && self.transport.type) || 'unknown';>
  1023. ! !