ensure-repo.ts 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. /*-----------------------------------------------------------------------------
  2. | Copyright (c) Jupyter Development Team.
  3. | Distributed under the terms of the Modified BSD License.
  4. |----------------------------------------------------------------------------*/
  5. /**
  6. * Ensure the integrity of the packages in the repo.
  7. *
  8. * Ensure the core package version dependencies match everywhere.
  9. * Ensure imported packages match dependencies.
  10. * Ensure a consistent version of all packages.
  11. * Manage the metapackage meta package.
  12. */
  13. import * as childProcess from 'child_process';
  14. import * as path from 'path';
  15. import * as fs from 'fs-extra';
  16. import * as utils from './utils';
  17. import { ensurePackage } from './ensure-package';
  18. // Data to ignore.
  19. let MISSING: { [key: string]: string[] } = {
  20. '@jupyterlab/buildutils': ['path']
  21. };
  22. let UNUSED: { [key: string]: string[] } = {
  23. '@jupyterlab/apputils': ['@types/react'],
  24. '@jupyterlab/apputils-extension': ['es6-promise'],
  25. '@jupyterlab/theme-dark-extension': ['font-awesome'],
  26. '@jupyterlab/theme-light-extension': ['font-awesome'],
  27. '@jupyterlab/vega2-extension': ['d3', 'vega', 'vega-lite']
  28. };
  29. let pkgData: { [key: string]: any } = {};
  30. let pkgPaths: { [key: string]: string } = {};
  31. let pkgNames: { [key: string]: string } = {};
  32. let depCache: { [key: string]: string } = {};
  33. /**
  34. * Ensure the metapackage package.
  35. *
  36. * @returns An array of messages for changes.
  37. */
  38. function ensureMetaPackage(): string[] {
  39. let basePath = path.resolve('.');
  40. let metaPackagePath = path.join(basePath, 'packages', 'metapackage');
  41. let metaPackageJson = path.join(metaPackagePath, 'package.json');
  42. let metaPackageData = utils.readJSONFile(metaPackageJson);
  43. let indexPath = path.join(metaPackagePath, 'src', 'index.ts');
  44. let index = fs.readFileSync(indexPath, 'utf8').split('\r\n').join('\n');
  45. let lines = index.split('\n').slice(0, 3);
  46. let messages: string[] = [];
  47. let seen: { [key: string]: boolean } = {};
  48. utils.getCorePaths().forEach(pkgPath => {
  49. if (path.resolve(pkgPath) === path.resolve(metaPackagePath)) {
  50. return;
  51. }
  52. let name = pkgNames[pkgPath];
  53. if (!name) {
  54. return;
  55. }
  56. seen[name] = true;
  57. let data = pkgData[name];
  58. let valid = true;
  59. // Ensure it is a dependency.
  60. if (!metaPackageData.dependencies[name]) {
  61. valid = false;
  62. metaPackageData.dependencies[name] = '^' + data.version;
  63. }
  64. // Ensure it is in index.ts
  65. if (index.indexOf(name) === -1) {
  66. valid = false;
  67. }
  68. lines.push(`import "${name}";`);
  69. if (!valid) {
  70. messages.push(`Updated: ${name}`);
  71. }
  72. });
  73. // Make sure there are no extra deps.
  74. Object.keys(metaPackageData.dependencies).forEach(name => {
  75. if (!(name in seen)) {
  76. messages.push(`Removing dependency: ${name}`);
  77. delete metaPackageData.dependencies[name];
  78. }
  79. });
  80. // Write the files.
  81. if (messages.length > 0) {
  82. utils.writePackageData(metaPackageJson, metaPackageData);
  83. }
  84. let newIndex = lines.join('\n');
  85. if (newIndex !== index) {
  86. messages.push('Index changed');
  87. fs.writeFileSync(indexPath, lines.join('\n'));
  88. }
  89. return messages;
  90. }
  91. /**
  92. * Ensure the jupyterlab application package.
  93. */
  94. function ensureJupyterlab(): string[] {
  95. // Get the current version of JupyterLab
  96. let cmd = 'python setup.py --version';
  97. let version = String(childProcess.execSync(cmd)).trim();
  98. let basePath = path.resolve('.');
  99. let corePath = path.join(basePath, 'dev_mode', 'package.json');
  100. let corePackage = utils.readJSONFile(corePath);
  101. corePackage.jupyterlab.extensions = {};
  102. corePackage.jupyterlab.mimeExtensions = {};
  103. corePackage.jupyterlab.version = version;
  104. corePackage.jupyterlab.linkedPackages = {};
  105. corePackage.dependencies = {};
  106. let singletonPackages = corePackage.jupyterlab.singletonPackages;
  107. utils.getCorePaths().forEach(pkgPath => {
  108. let dataPath = path.join(pkgPath, 'package.json');
  109. let data: any;
  110. try {
  111. data = utils.readJSONFile(dataPath);
  112. } catch (e) {
  113. return;
  114. }
  115. if (data.private === true) {
  116. return;
  117. }
  118. // Make sure it is included as a dependency.
  119. corePackage.dependencies[data.name] = '^' + String(data.version);
  120. let relativePath = `../packages/${path.basename(pkgPath)}`;
  121. corePackage.jupyterlab.linkedPackages[data.name] = relativePath;
  122. // Add its dependencies to the core dependencies if they are in the
  123. // singleton packages.
  124. let deps = data.dependencies || {};
  125. for (let dep in deps) {
  126. if (singletonPackages.indexOf(dep) !== -1) {
  127. corePackage.dependencies[dep] = deps[dep];
  128. }
  129. }
  130. let jlab = data.jupyterlab;
  131. if (!jlab) {
  132. return;
  133. }
  134. // Handle extensions.
  135. ['extension', 'mimeExtension'].forEach(item => {
  136. let ext = jlab[item];
  137. if (ext === true) {
  138. ext = '';
  139. }
  140. if (typeof ext !== 'string') {
  141. return;
  142. }
  143. corePackage.jupyterlab[item + 's'][data.name] = ext;
  144. });
  145. });
  146. // Write the package.json back to disk.
  147. if (utils.writePackageData(corePath, corePackage)) {
  148. return ['Updated core'];
  149. }
  150. return [];
  151. }
  152. /**
  153. * Ensure the repo integrity.
  154. */
  155. export
  156. function ensureIntegrity(): boolean {
  157. let messages: { [key: string]: string[] } = {};
  158. // Pick up all the package versions.
  159. let paths = utils.getLernaPaths();
  160. // These two are not part of the workspaces but should be kept
  161. // in sync.
  162. paths.push('./jupyterlab/tests/mock_packages/extension');
  163. paths.push('./jupyterlab/tests/mock_packages/mimeextension');
  164. paths.forEach(pkgPath => {
  165. // Read in the package.json.
  166. let data: any;
  167. try {
  168. data = utils.readJSONFile(path.join(pkgPath, 'package.json'));
  169. } catch (e) {
  170. console.error(e);
  171. return;
  172. }
  173. pkgData[data.name] = data;
  174. pkgPaths[data.name] = pkgPath;
  175. pkgNames[pkgPath] = data.name;
  176. });
  177. // Validate each package.
  178. for (let name in pkgData) {
  179. let options = {
  180. pkgPath: pkgPaths[name],
  181. data: pkgData[name],
  182. depCache,
  183. missing: MISSING[name],
  184. unused: UNUSED[name]
  185. };
  186. let pkgMessages = ensurePackage(options);
  187. if (pkgMessages.length > 0) {
  188. messages[name] = pkgMessages;
  189. }
  190. }
  191. // Handle the top level package.
  192. let corePath = path.resolve('.', 'package.json');
  193. let coreData: any = utils.readJSONFile(corePath);
  194. if (utils.writePackageData(corePath, coreData)) {
  195. messages['top'] = ['Update package.json'];
  196. }
  197. // Handle the metapackage metapackage.
  198. let pkgMessages = ensureMetaPackage();
  199. if (pkgMessages.length > 0) {
  200. let pkgName ='@jupyterlab/metapackage';
  201. if (!messages[pkgName]) {
  202. messages[pkgName] = [];
  203. }
  204. messages[pkgName] = messages[pkgName].concat(pkgMessages);
  205. }
  206. // Handle the JupyterLab application top package.
  207. pkgMessages = ensureJupyterlab();
  208. if (pkgMessages.length > 0) {
  209. let pkgName = '@jupyterlab/application-top';
  210. if (!messages[pkgName]) {
  211. messages[pkgName] = [];
  212. }
  213. messages[pkgName] = messages[pkgName].concat(pkgMessages);
  214. }
  215. // Handle any messages.
  216. if (Object.keys(messages).length > 0) {
  217. console.log(JSON.stringify(messages, null, 2));
  218. if (process.env.TRAVIS_BRANCH || process.env.APPVEYOR) {
  219. console.log('\n\nPlease run `jlpm run integrity` locally and commit the changes');
  220. process.exit(1);
  221. }
  222. utils.run('jlpm install');
  223. console.log('\n\nPlease commit the changes by running:');
  224. console.log('git commit -a -m "Package integrity updates"');
  225. return false;
  226. }
  227. console.log('Repo integrity verified!');
  228. return true;
  229. }
  230. if (require.main === module) {
  231. ensureIntegrity();
  232. }