grafanaui.release.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. import execa from 'execa';
  2. import { execTask } from '../utils/execTask';
  3. import { changeCwdToGrafanaUiDist, changeCwdToGrafanaUi } from '../utils/cwd';
  4. import semver from 'semver';
  5. import inquirer from 'inquirer';
  6. import chalk from 'chalk';
  7. import { useSpinner } from '../utils/useSpinner';
  8. import { savePackage, buildTask } from './grafanaui.build';
  9. import { TaskRunner, Task } from './task';
  10. type VersionBumpType = 'prerelease' | 'patch' | 'minor' | 'major';
  11. interface ReleaseTaskOptions {
  12. publishToNpm: boolean;
  13. usePackageJsonVersion: boolean;
  14. createVersionCommit: boolean;
  15. }
  16. const promptBumpType = async () => {
  17. return inquirer.prompt<{ type: VersionBumpType }>([
  18. {
  19. type: 'list',
  20. message: 'Select version bump',
  21. name: 'type',
  22. choices: ['prerelease', 'patch', 'minor', 'major'],
  23. validate: answer => {
  24. if (answer.length < 1) {
  25. return 'You must choose something';
  26. }
  27. return true;
  28. },
  29. },
  30. ]);
  31. };
  32. const promptPrereleaseId = async (message = 'Is this a prerelease?', allowNo = true) => {
  33. return inquirer.prompt<{ id: string }>([
  34. {
  35. type: 'list',
  36. message: message,
  37. name: 'id',
  38. choices: allowNo ? ['no', 'alpha', 'beta'] : ['alpha', 'beta'],
  39. validate: answer => {
  40. if (answer.length < 1) {
  41. return 'You must choose something';
  42. }
  43. return true;
  44. },
  45. },
  46. ]);
  47. };
  48. const promptConfirm = async (message?: string) => {
  49. return inquirer.prompt<{ confirmed: boolean }>([
  50. {
  51. type: 'confirm',
  52. message: message || 'Is that correct?',
  53. name: 'confirmed',
  54. default: false,
  55. },
  56. ]);
  57. };
  58. // Since Grafana core depends on @grafana/ui highly, we run full check before release
  59. const runChecksAndTests = async () =>
  60. useSpinner<void>(`Running checks and tests`, async () => {
  61. await execa('npm', ['run', 'test']);
  62. })();
  63. const bumpVersion = (version: string) =>
  64. useSpinner<void>(`Saving version ${version} to package.json`, async () => {
  65. changeCwdToGrafanaUi();
  66. await execa('npm', ['version', version]);
  67. changeCwdToGrafanaUiDist();
  68. const pkg = require(`${process.cwd()}/package.json`);
  69. pkg.version = version;
  70. await savePackage({ path: `${process.cwd()}/package.json`, pkg });
  71. })();
  72. const publishPackage = (name: string, version: string) =>
  73. useSpinner<void>(`Publishing ${name} @ ${version} to npm registry...`, async () => {
  74. changeCwdToGrafanaUiDist();
  75. console.log(chalk.yellowBright.bold(`\nReview dist package.json before proceeding!\n`));
  76. const { confirmed } = await promptConfirm('Are you ready to publish to npm?');
  77. if (!confirmed) {
  78. process.exit();
  79. }
  80. await execa('npm', ['publish', '--access', 'public']);
  81. })();
  82. const ensureMasterBranch = async () => {
  83. const currentBranch = await execa.stdout('git', ['symbolic-ref', '--short', 'HEAD']);
  84. const status = await execa.stdout('git', ['status', '--porcelain']);
  85. if (currentBranch !== 'master' && status !== '') {
  86. console.error(chalk.red.bold('You need to be on clean master branch to release @grafana/ui'));
  87. process.exit(1);
  88. }
  89. };
  90. const prepareVersionCommitAndPush = async (version: string) =>
  91. useSpinner<void>('Commiting and pushing @grafana/ui version update', async () => {
  92. await execa.stdout('git', ['commit', '-a', '-m', `Upgrade @grafana/ui version to v${version}`]);
  93. await execa.stdout('git', ['push']);
  94. })();
  95. const releaseTaskRunner: TaskRunner<ReleaseTaskOptions> = async ({
  96. publishToNpm,
  97. usePackageJsonVersion,
  98. createVersionCommit,
  99. }) => {
  100. await runChecksAndTests();
  101. if (publishToNpm) {
  102. // TODO: Ensure release branch
  103. // When need to update this when we star keeping @grafana/ui releases in sync with core
  104. await ensureMasterBranch();
  105. }
  106. await execTask(buildTask)();
  107. let releaseConfirmed = false;
  108. let nextVersion;
  109. changeCwdToGrafanaUiDist();
  110. const pkg = require(`${process.cwd()}/package.json`);
  111. console.log(`Current version: ${pkg.version}`);
  112. do {
  113. if (!usePackageJsonVersion) {
  114. const { type } = await promptBumpType();
  115. console.log(type);
  116. if (type === 'prerelease') {
  117. const { id } = await promptPrereleaseId('What kind of prerelease?', false);
  118. nextVersion = semver.inc(pkg.version, type, id);
  119. } else {
  120. const { id } = await promptPrereleaseId();
  121. if (id !== 'no') {
  122. nextVersion = semver.inc(pkg.version, `pre${type}`, id);
  123. } else {
  124. nextVersion = semver.inc(pkg.version, type);
  125. }
  126. }
  127. } else {
  128. nextVersion = pkg.version;
  129. }
  130. console.log(chalk.yellowBright.bold(`You are going to release a new version of ${pkg.name}`));
  131. if (usePackageJsonVersion) {
  132. console.log(chalk.green(`Version based on package.json: `), chalk.bold.yellowBright(`${nextVersion}`));
  133. } else {
  134. console.log(chalk.green(`Version bump: ${pkg.version} ->`), chalk.bold.yellowBright(`${nextVersion}`));
  135. }
  136. const { confirmed } = await promptConfirm();
  137. releaseConfirmed = confirmed;
  138. } while (!releaseConfirmed);
  139. if (!usePackageJsonVersion) {
  140. await bumpVersion(nextVersion);
  141. }
  142. if (createVersionCommit) {
  143. await prepareVersionCommitAndPush(nextVersion);
  144. }
  145. if (publishToNpm) {
  146. await publishPackage(pkg.name, nextVersion);
  147. console.log(chalk.green(`\nVersion ${nextVersion} of ${pkg.name} succesfully released!`));
  148. console.log(chalk.yellow(`\nUpdated @grafana/ui/package.json with version bump created.`));
  149. process.exit();
  150. } else {
  151. console.log(
  152. chalk.green(
  153. `\nVersion ${nextVersion} of ${pkg.name} succesfully prepared for release. See packages/grafana-ui/dist`
  154. )
  155. );
  156. console.log(chalk.green(`\nTo publish to npm registry run`), chalk.bold.blue(`npm run gui:publish`));
  157. }
  158. };
  159. export const releaseTask = new Task<ReleaseTaskOptions>();
  160. releaseTask.setName('@grafana/ui release');
  161. releaseTask.setRunner(releaseTaskRunner);