Home Reference Source

src/packages.js

  1. /*
  2. * OS.js - JavaScript Cloud/Web Desktop Platform
  3. *
  4. * Copyright (c) Anders Evenrud <andersevenrud@gmail.com>
  5. * All rights reserved.
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions are met:
  9. *
  10. * 1. Redistributions of source code must retain the above copyright notice, this
  11. * list of conditions and the following disclaimer
  12. * 2. Redistributions in binary form must reproduce the above copyright notice,
  13. * this list of conditions and the following disclaimer in the documentation
  14. * and/or other materials provided with the distribution
  15. *
  16. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  17. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  18. * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  19. * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
  20. * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  21. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  22. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  23. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  24. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  25. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. *
  27. * @author Anders Evenrud <andersevenrud@gmail.com>
  28. * @licence Simplified BSD License
  29. */
  30.  
  31. const fs = require('fs-extra');
  32. const fg = require('fast-glob');
  33. const path = require('path');
  34. const Package = require('./package.js');
  35. const consola = require('consola');
  36. const logger = consola.withTag('Packages');
  37.  
  38. const relative = filename => filename.replace(process.cwd(), '');
  39.  
  40. const readOrDefault = filename => fs.existsSync(filename)
  41. ? fs.readJsonSync(filename)
  42. : [];
  43.  
  44. /**
  45. * Package Service Options
  46. * @typedef {Object} PackagesOptions
  47. * @property {string} [manifestFile] Manifest filename
  48. * @property {string} [discoveredFile] Discovery output file
  49. */
  50.  
  51. /**
  52. * OS.js Package Management
  53. */
  54. class Packages {
  55.  
  56. /**
  57. * Create new instance
  58. * @param {Core} core Core reference
  59. * @param {PackagesOptions} [options] Instance options
  60. */
  61. constructor(core, options = {}) {
  62. /**
  63. * @type {Core}
  64. */
  65. this.core = core;
  66.  
  67. /**
  68. * @type {Package[]}
  69. */
  70. this.packages = [];
  71.  
  72. this.hotReloading = {};
  73.  
  74. /**
  75. * @type {PackagesOptions}
  76. */
  77. this.options = {
  78. manifestFile: null,
  79. discoveredFile: null,
  80. ...options
  81. };
  82. }
  83.  
  84. /**
  85. * Initializes packages
  86. */
  87. init() {
  88. this.core.on('osjs/application:socket:message', (ws, ...params) => {
  89. this.handleMessage(ws, params);
  90. });
  91.  
  92. return this.load();
  93. }
  94.  
  95. /**
  96. * Loads package manager
  97. * @return {Promise<boolean>}
  98. */
  99. load() {
  100. return this.createLoader()
  101. .then(packages => {
  102. this.packages = this.packages.concat(packages);
  103.  
  104. return true;
  105. });
  106. }
  107.  
  108. /**
  109. * Loads all packages
  110. * @return {Promise<Package[]>}
  111. */
  112. createLoader() {
  113. let result = [];
  114. const {discoveredFile, manifestFile} = this.options;
  115. const discovered = readOrDefault(discoveredFile);
  116. const manifest = readOrDefault(manifestFile);
  117. const sources = discovered.map(d => path.join(d, 'metadata.json'));
  118.  
  119. logger.info('Using package discovery file', relative(discoveredFile));
  120. logger.info('Using package manifest file', relative(manifestFile));
  121.  
  122. const stream = fg.stream(sources, {
  123. extension: false,
  124. brace: false,
  125. deep: 1,
  126. case: false
  127. });
  128.  
  129. stream.on('error', error => logger.error(error));
  130. stream.on('data', filename => {
  131. result.push(this.loadPackage(filename, manifest));
  132. });
  133.  
  134. return new Promise((resolve, reject) => {
  135. stream.once('end', () => {
  136. Promise.all(result)
  137. .then(result => result.filter(iter => !!iter.handler))
  138. .then(resolve)
  139. .catch(reject);
  140. });
  141. });
  142. }
  143.  
  144. /**
  145. * When a package dist has changed
  146. * @param {Package} pkg Package instance
  147. */
  148. onPackageChanged(pkg) {
  149. clearTimeout(this.hotReloading[pkg.metadata.name]);
  150.  
  151. this.hotReloading[pkg.metadata.name] = setTimeout(() => {
  152. logger.debug('Sending reload signal for', pkg.metadata.name);
  153. this.core.broadcast('osjs/packages:package:changed', [pkg.metadata.name]);
  154. }, 500);
  155. }
  156.  
  157. /**
  158. * Loads package data
  159. * @param {string} filename Filename
  160. * @param {PackageMetadata} manifest Manifest
  161. * @return {Promise<Package>}
  162. */
  163. loadPackage(filename, manifest) {
  164. const done = (pkg, error) => {
  165. if (error) {
  166. logger.warn(error);
  167. }
  168.  
  169. return Promise.resolve(pkg);
  170. };
  171.  
  172. return fs.readJson(filename)
  173. .then(metadata => {
  174. const pkg = new Package(this.core, {
  175. filename,
  176. metadata
  177. });
  178.  
  179. return this.initializePackage(pkg, manifest, done);
  180. });
  181. }
  182.  
  183. /**
  184. * Initializes a package
  185. * @return {Promise<Package>}
  186. */
  187. initializePackage(pkg, manifest, done) {
  188. if (pkg.validate(manifest)) {
  189. logger.info(`Loading ${relative(pkg.script)}`);
  190.  
  191. try {
  192. if (this.core.configuration.development) {
  193. pkg.watch(() => {
  194. this.onPackageChanged(pkg);
  195. });
  196. }
  197.  
  198. return pkg.init()
  199. .then(() => done(pkg))
  200. .catch(e => done(pkg, e));
  201. } catch (e) {
  202. return done(pkg, e);
  203. }
  204. }
  205.  
  206. return done(pkg);
  207. }
  208.  
  209. /**
  210. * Starts packages
  211. */
  212. start() {
  213. this.packages.forEach(pkg => pkg.start());
  214. }
  215.  
  216. /**
  217. * Destroys packages
  218. * @return {Promise<undefined>}
  219. */
  220. async destroy() {
  221. await Promise.all(this.packages.map(pkg => pkg.destroy()));
  222.  
  223. this.packages = [];
  224. }
  225.  
  226. /**
  227. * Handles an incoming message and signals an application
  228. *
  229. * This will call the 'onmessage' event in your application server script
  230. *
  231. * @param {WebSocket} ws Websocket Connection client
  232. * @param {Array} params A list of incoming parameters
  233. */
  234. handleMessage(ws, params) {
  235. const {pid, name, args} = params[0];
  236. const found = this.packages.findIndex(({metadata}) => metadata.name === name);
  237.  
  238. if (found !== -1) {
  239. const {handler} = this.packages[found];
  240. if (handler && typeof handler.onmessage === 'function') {
  241. const respond = (...respondParams) => ws.send(JSON.stringify({
  242. name: 'osjs/application:socket:message',
  243. params: [{
  244. pid,
  245. args: respondParams
  246. }]
  247. }));
  248.  
  249. handler.onmessage(ws, respond, args);
  250. }
  251. }
  252. }
  253. }
  254.  
  255. module.exports = Packages;