-
Notifications
You must be signed in to change notification settings - Fork 95
/
run.js
500 lines (433 loc) · 20.2 KB
/
run.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
/*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
'use strict';
const CodeGen = require('@accordproject/concerto-tools').CodeGen;
const FileWriter = require('@accordproject/concerto-tools').FileWriter;
const HtmlTransformer = require('@accordproject/markdown-html').HtmlTransformer;
const CiceroMarkTransformer = require('@accordproject/markdown-cicero').CiceroMarkTransformer;
const Template = require('@accordproject/cicero-core').Template;
const Clause = require('@accordproject/cicero-core').Clause;
const rimraf = require('rimraf');
const path = require('path');
const nunjucks = require('nunjucks');
const plantumlEncoder = require('plantuml-encoder');
const showdown = require('showdown');
const uuidv1 = require('uuid/v1');
const semver = require('semver');
const {
promisify
} = require('util');
const {
resolve
} = require('path');
const fs = require('fs-extra')
const readdir = promisify(fs.readdir);
const rename = promisify(fs.rename);
const stat = promisify(fs.stat);
const mkdirp = require('mkdirp');
const writeFile = promisify(fs.writeFile);
const jsdom = require("jsdom");
const jquery = require("jquery");
/**
* GLOBALS
*/
const rootDir = resolve(__dirname, './src');
const buildDir = resolve(__dirname, './build/');
const archiveDir = resolve(__dirname, './build/archives');
const serverRoot = process.env.SERVER_ROOT ? process.env.SERVER_ROOT : 'https://templates.accordproject.org';
const studioRoot = 'https://studio.accordproject.org';
const githubRoot = `https://github.dev/accordproject/cicero-template-library/blob/master`;
const ciceroMark = new CiceroMarkTransformer();
const htmlMark = new HtmlTransformer();
nunjucks.configure('./views', {
autoescape: false,
});
/**
* Generating a static website from a template library
*
* - Scans the 'src' directory for templates
* - Loads each template using Template.fromDirectory
* - Runs any tests for the templates that are in the `test` directory for the template using Mocha
* - Generates an archive for the template and saves to the 'build/archives' directory
* - Generates HTML and other resources for the template
* - Generates an index.html and json file for all the templates
*
* Options (environment variables):
* - SKIP_GENERATION : do not write anything to disk
* - SKIP_TESTS : do not run the unit tests
* - SKIP_DROPDOWNS : do not regenerate update dropdowns on old html versions to point to the latest releases
* - DELETE_ALL : clear the build directory. Use with extreme caution as all old versions of templates
* will be removed from the build archives folder!
* - FORCE_CREATE_ARCHIVE : regenerate an existing archive even if it exists. Warning the new archive
* may change because it will re-download external dependencies
* Options (command line)
* - template name (only this template gets built)
*/
(async function () {
try {
let templateName = process.argv.slice(2);
if(templateName && templateName.length > 0) {
console.log('Only building template: ' + templateName);
} else {
templateName = null;
}
if(process.env.DELETE_ALL) {
// delete build directory
rimraf.sync(buildDir);
}
const templateIndex = await buildTemplates(templateUnitTester, templatePageGenerator, templateName );
if(!process.env.SKIP_GENERATION) {
// copy the logo to build directory
await fs.copy('assets', './build/assets');
await fs.copy('styles.css', './build/styles.css');
await fs.copy('_headers', './build/_headers');
// get the latest versions of each template
const latestIndex = filterTemplateIndex(templateIndex);
// generate the index html page
const templateResult = nunjucks.render('index.njk', {
serverRoot: serverRoot,
templateIndex: latestIndex,
});
await writeFile('./build/index.html', templateResult);
}
}
catch(err) {
console.log(err);
}
})();
/**
* Returns a template index that only contains the latest version
* of each template
*
* @param {object} templateIndex - the template index
* @returns {object} a new template index that only contains the latest version of each template
*/
function filterTemplateIndex(templateIndex) {
const result = {};
const nameToVersion = {};
// build a map of the latest version of each template
for(let template of Object.keys(templateIndex)) {
const atIndex = template.indexOf('@');
const name = template.substring(0,atIndex);
const version = template.substring(atIndex+1);
const existingVersion = nameToVersion[name];
if(!existingVersion || semver.lt(existingVersion, version)) {
nameToVersion[name] = version;
}
}
// now build the result
for(let name in nameToVersion) {
const id = `${name}@${nameToVersion[name]}`;
result[id] = templateIndex[id];
}
return result;
}
/**
* Get all the files beneath a subdirectory
*
* @param {String} dir - the root directory
*/
async function getFiles(dir) {
const subdirs = await readdir(dir);
const files = await Promise.all(subdirs.map(async (subdir) => {
const res = resolve(dir, subdir);
return (await stat(res)).isDirectory() ? getFiles(res) : res;
}));
return files.reduce((a, f) => a.concat(f), []);
}
/**
* Builds all the templates and copies the valid templates
* to the ./build/archives directory
* @param {Function} preProcessor - a function that is called for each valid template
* @param {Function} postProcessor - a function that is called for each valid template (after the preProcessor)
* @param {String} [selectedTemplate] - optional name of a template. If specified this is the only template that is built
* @returns {Object} the index of clause and contract templates
*/
async function buildTemplates(preProcessor, postProcessor, selectedTemplate) {
// load the index
const templateLibraryPath = `${buildDir}/template-library.json`;
let templateIndex = {};
const indexExists = await fs.pathExists(templateLibraryPath);
if(indexExists) {
const indexContent = fs.readFileSync(templateLibraryPath, 'utf8');
templateIndex = JSON.parse(indexContent);
}
const files = await getFiles(rootDir);
for (const file of files) {
const fileName = path.basename(file);
let selected = false;
// assume all package.json files that are not inside node_modules are templates
if (fileName === 'package.json' && file.indexOf('/node_modules/') === -1) {
selected = true;
}
// unless a given template name has been specified
if(selected && selectedTemplate) {
const packageJson = fs.readFileSync(file, 'utf8');
const pkgJson = JSON.parse(packageJson);
if(pkgJson.name != selectedTemplate) {
selected = false;
}
}
if(selected) {
// read the parent directory as a template
const templatePath = path.dirname(file);
console.log(`Processing ${templatePath}`);
const dest = templatePath.replace('/src/', '/build/');
await fs.ensureDir(archiveDir);
try {
const template = await Template.fromDirectory(templatePath);
// call the pre template processor
await preProcessor(templatePath, template);
if(!process.env.SKIP_GENERATION) {
const templateVersions = Object.keys(templateIndex).filter(
item => {
const atIndex = item.indexOf("@");
const name = item.substring(0, atIndex);
return name == template.getName();
}
);
templateVersions.forEach(versionToUpdate => {
const templateResult = nunjucks.render("dropdown.njk", {
identifier: versionToUpdate,
templateVersions: templateVersions,
});
fs.readFile(
"build/" + versionToUpdate + ".html",
"utf8",
(err, data) => {
if (err) {
console.log(`Failed reading build/${versionToUpdate}.html with ${err}`);
}
const dom = new jsdom.JSDOM(data);
const $ = jquery(dom.window);
if (!process.env.SKIP_DROPDOWNS) {
const dropdownContentElement = $(".dropdown-content");
if (dropdownContentElement.length) {
dropdownContentElement.html(templateResult);
}
}
if(process.env.ADD_VSCODE_BUTTON){
const dropdownContentElement = $("a.button.open-studio");
if (dropdownContentElement.length) {
const githubURL = `${githubRoot}/src/${encodeURIComponent(template.getName())}/README.md`;
dropdownContentElement.after(`\n<a href="${githubURL}" class="button is-rounded is-primary open-studio">Open in VSCode Web</a>`);
}
}
if (!process.env.SKIP_DROPDOWNS || process.env.ADD_VSCODE_BUTTON) {
fs.writeFile(
"build/" + versionToUpdate + ".html",
dom.serialize(),
err => {
if (err) {
console.log(`Failed saving build/${versionToUpdate}.html with ${err}`);
} else {
console.log("VSCode button added for template: " + "build/" + versionToUpdate + ".html" );
}
}
);
}
}
);
});
// get the name of the generated archive
const destPath = path.dirname(dest);
await fs.ensureDir(destPath);
const archiveFileName = `${template.getIdentifier()}.cta`;
const archiveFilePath = `${archiveDir}/${archiveFileName}`;
const archiveFileExists = await fs.pathExists(archiveFilePath);
const ciceroArchiveFileName = `${template.getIdentifier()}-cicero.cta`;
const ciceroArchiveFilePath = `${archiveDir}/${ciceroArchiveFileName}`;
const ciceroArchiveFileExists = await fs.pathExists(ciceroArchiveFilePath);
if(!ciceroArchiveFileExists || process.env.FORCE_CREATE_ARCHIVE) {
const ciceroArchive = await template.toArchive('es6');
await writeFile(ciceroArchiveFilePath, ciceroArchive);
console.log('Copied: ' + ciceroArchiveFilePath);
}
if(!archiveFileExists || process.env.FORCE_CREATE_ARCHIVE) {
const ergoArchive = await template.toArchive('ergo');
await writeFile(archiveFilePath, ergoArchive);
console.log('Copied: ' + archiveFileName);
}
if(!ciceroArchiveFileExists || !archiveFileExists || process.env.FORCE_CREATE_ARCHIVE) {
// update the index
const m = template.getMetadata();
const templateHash = template.getHash();
const indexData = {
uri: `ap://${template.getIdentifier()}#${templateHash}`,
url: `${serverRoot}/archives/${archiveFileName}`,
ciceroUrl: `${serverRoot}/archives/${ciceroArchiveFileName}`,
name : m.getName(),
displayName: m.getDisplayName(),
description : m.getDescription(),
version: m.getVersion(),
ciceroVersion: m.getCiceroVersion(),
type: m.getTemplateType(),
logo: m.getLogo() ? m.getLogo().toString('base64') : null,
author: m.getAuthor() ? m.getAuthor() : null,
}
templateIndex[template.getIdentifier()] = indexData;
// call the post template processor
await postProcessor(templateIndex, templatePath, template);
}
else {
console.log(`Skipped: ${archiveFileName} (already exists).`);
}
}
} catch (err) {
console.log(err);
console.log(`Failed processing ${file} with ${err}`);
}
}
}
// save the index
await writeFile(templateLibraryPath, JSON.stringify(templateIndex, null, 4));
// return the updated index
return templateIndex;
};
/**
* Runs the standard tests for a template
* @param {String} templatePath - the location of the template on disk
* @param {Template} template
*/
async function templateUnitTester(templatePath, template) {
// check that all the samples parse
const samples = template.getMetadata().getSamples();
if(samples) {
const sampleValues = Object.values(samples);
// should be TemplateInstance
const instance = new Clause(template);
for(const s of sampleValues ) {
instance.parse(s);
}
}
}
/**
* Generate a sample instance for a template's type
* @param {Template} template the template
* @param {string} type the fully qualified type name
*/
function sampleInstance(template, type) {
// generate the sample json instances
const sampleGenerationOptions = {};
sampleGenerationOptions.generate = true;
sampleGenerationOptions.includeOptionalFields = true;
const classDecl = template.getModelManager().getType(type);
let result = {};
result.abstract = 'this is an abstract type';
if (!classDecl.isAbstract()) {
if (classDecl.getIdentifierFieldName()) {
result = template.getFactory().newResource( classDecl.getNamespace(), classDecl.getName(), uuidv1(), sampleGenerationOptions);
} else {
result = template.getFactory().newResource( classDecl.getNamespace(), classDecl.getName(), null, sampleGenerationOptions);
}
}
return result;
}
/**
* Generates html and other resources from a valid template
* @param {object} templateIndex - the existing template index
* @param {String} templatePath - the location of the template on disk
* @param {Template} template
*/
async function templatePageGenerator(templateIndex, templatePath, template) {
console.log(`Generating html for ${templatePath}`);
const archiveFileName = `${template.getIdentifier()}.cta`;
const archiveFilePath = `${archiveDir}/${archiveFileName}`;
const templatePageHtml = archiveFileName.replace('.cta', '.html');
const pumlFilePath = `${buildDir}/${template.getIdentifier()}.puml`;
// generate UML
const modelDecls = template.getTemplateModel().getModelFile();
const models = template.getModelManager().getModels();
const modelFile = models[models.length-1].content;
const visitor = new CodeGen.PlantUMLVisitor();
const fileWriter = new FileWriter(buildDir);
fileWriter.openFile(pumlFilePath);
fileWriter.writeLine(0, '@startuml');
const params = {fileWriter : fileWriter};
modelDecls.accept(visitor, params);
fileWriter.writeLine(0, '@enduml');
fileWriter.closeFile();
const pumlContent = fs.readFileSync(pumlFilePath, 'utf8');
const encoded = plantumlEncoder.encode(pumlContent);
const umlURL = `https://www.plantuml.com/plantuml/svg/${encoded}`;
const umlCardURL = `https://www.plantuml.com/plantuml/png/${encoded}`;
const studioURL = `${studioRoot}/?template=${encodeURIComponent('ap://' + template.getIdentifier() + '#hash')}`;
const githubURL = `${githubRoot}/src/${encodeURIComponent(template.getName())}/README.md`;
const converter = new showdown.Converter();
const readmeHtml = converter.makeHtml(template.getMetadata().getREADME());
let sampleInstanceText = null;
// parse the default sample and use it as the sample instance
const samples = template.getMetadata().getSamples();
if(samples.default) {
// should be TemplateInstance
const instance = new Clause(template);
instance.parse(samples.default);
sampleInstanceText = JSON.stringify(instance.getData(), null, 4);
}
else {
// no sample was found, so we generate one
const classDecl = template.getTemplateModel();
sampleInstanceText = JSON.stringify(sampleInstance(template, classDecl.getFullyQualifiedName()), null, 4);
}
const requestTypes = {};
for(let type of template.getRequestTypes()) {
requestTypes[type] = JSON.stringify(sampleInstance(template, type), null, 4);
}
const responseTypes = {};
for(let type of template.getResponseTypes()) {
responseTypes[type] = JSON.stringify(sampleInstance(template, type), null, 4);
}
const stateTypes = {}
for(let type of template.getStateTypes()) {
stateTypes[type] = JSON.stringify(sampleInstance(template, type), null, 4);
}
const eventTypes = {}
for(let type of template.getEmitTypes()) {
eventTypes[type] = JSON.stringify(sampleInstance(template, type), null, 4);
}
// get all the versions of the template
const templateVersions = Object.keys(templateIndex).filter((item) => {
const atIndex = item.indexOf('@');
const name = item.substring(0,atIndex);
return name == template.getName();
});
const sample = template.getMetadata().getSample();
const logo = template.getMetadata().getLogo() ? template.getMetadata().getLogo().toString('base64') : null;
const author = template.getMetadata().getAuthor() ? template.getMetadata().getAuthor() : null;
let sampleHTML = htmlMark.toHtml(ciceroMark.fromMarkdown(sample,'json'));
// XXX HTML cleanup hack for rendering in page. Would be best done with the right option in markdown-transform
sampleHTML = sampleHTML.replace('<html>\n<body>\n<div class="document">','').replace('</div>\n</body>\n</html>','');
const templateResult = nunjucks.render('template.njk', {
serverRoot: serverRoot,
umlURL : umlURL,
umlCardURL : umlCardURL,
studioURL : studioURL,
githubURL : githubURL,
filePath: templatePageHtml,
template: template,
modelFile: modelFile,
sample: sample,
sampleHTML: sampleHTML,
readmeHtml: readmeHtml,
requestTypes: requestTypes,
responseTypes: responseTypes,
stateTypes: stateTypes,
instance: sampleInstanceText,
eventTypes: eventTypes,
templateVersions: templateVersions,
logo: logo,
author: author,
});
await writeFile(`./build/${templatePageHtml}`, templateResult);
}