grafanaui.release.ts 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. import execa from 'execa';
  2. import { execTask } from '../utils/execTask';
  3. import { changeCwdToGrafanaUiDist, changeCwdToGrafanaUi, restoreCwd } 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, clean } 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. await execa('npm', ['publish', '--access', 'public']);
  76. })();
  77. const ensureMasterBranch = async () => {
  78. const currentBranch = await execa.stdout('git', ['symbolic-ref', '--short', 'HEAD']);
  79. const status = await execa.stdout('git', ['status', '--porcelain']);
  80. if (currentBranch !== 'master' && status !== '') {
  81. console.error(chalk.red.bold('You need to be on clean master branch to release @grafana/ui'));
  82. process.exit(1);
  83. }
  84. };
  85. const prepareVersionCommitAndPush = async (version: string) =>
  86. useSpinner<void>('Commiting and pushing @grafana/ui version update', async () => {
  87. await execa.stdout('git', ['commit', '-a', '-m', `Upgrade @grafana/ui version to v${version}`]);
  88. await execa.stdout('git', ['push']);
  89. })();
  90. const releaseTaskRunner: TaskRunner<ReleaseTaskOptions> = async ({
  91. publishToNpm,
  92. usePackageJsonVersion,
  93. createVersionCommit,
  94. }) => {
  95. changeCwdToGrafanaUi();
  96. await clean(); // Clean previous build if exists
  97. restoreCwd();
  98. if (publishToNpm) {
  99. // TODO: Ensure release branch
  100. // When need to update this when we star keeping @grafana/ui releases in sync with core
  101. await ensureMasterBranch();
  102. }
  103. runChecksAndTests();
  104. await execTask(buildTask)();
  105. let releaseConfirmed = false;
  106. let nextVersion;
  107. changeCwdToGrafanaUiDist();
  108. const pkg = require(`${process.cwd()}/package.json`);
  109. console.log(`Current version: ${pkg.version}`);
  110. do {
  111. if (!usePackageJsonVersion) {
  112. const { type } = await promptBumpType();
  113. console.log(type);
  114. if (type === 'prerelease') {
  115. const { id } = await promptPrereleaseId('What kind of prerelease?', false);
  116. nextVersion = semver.inc(pkg.version, type, id);
  117. } else {
  118. const { id } = await promptPrereleaseId();
  119. if (id !== 'no') {
  120. nextVersion = semver.inc(pkg.version, `pre${type}`, id);
  121. } else {
  122. nextVersion = semver.inc(pkg.version, type);
  123. }
  124. }
  125. } else {
  126. nextVersion = pkg.version;
  127. }
  128. console.log(chalk.yellowBright.bold(`You are going to release a new version of ${pkg.name}`));
  129. if (usePackageJsonVersion) {
  130. console.log(chalk.green(`Version based on package.json: `), chalk.bold.yellowBright(`${nextVersion}`));
  131. } else {
  132. console.log(chalk.green(`Version bump: ${pkg.version} ->`), chalk.bold.yellowBright(`${nextVersion}`));
  133. }
  134. const { confirmed } = await promptConfirm();
  135. releaseConfirmed = confirmed;
  136. } while (!releaseConfirmed);
  137. if (!usePackageJsonVersion) {
  138. await bumpVersion(nextVersion);
  139. }
  140. if (createVersionCommit) {
  141. await prepareVersionCommitAndPush(nextVersion);
  142. }
  143. if (publishToNpm) {
  144. console.log(chalk.yellowBright.bold(`\nReview dist package.json before proceeding!\n`));
  145. const { confirmed } = await promptConfirm('Are you ready to publish to npm?');
  146. if (!confirmed) {
  147. process.exit();
  148. }
  149. await publishPackage(pkg.name, nextVersion);
  150. console.log(chalk.green(`\nVersion ${nextVersion} of ${pkg.name} succesfully released!`));
  151. console.log(chalk.yellow(`\nUpdated @grafana/ui/package.json with version bump created.`));
  152. process.exit();
  153. } else {
  154. console.log(
  155. chalk.green(
  156. `\nVersion ${nextVersion} of ${pkg.name} succesfully prepared for release. See packages/grafana-ui/dist`
  157. )
  158. );
  159. console.log(chalk.green(`\nTo publish to npm registry run`), chalk.bold.blue(`npm run gui:publish`));
  160. }
  161. };
  162. export const releaseTask = new Task<ReleaseTaskOptions>();
  163. releaseTask.setName('@grafana/ui release');
  164. releaseTask.setRunner(releaseTaskRunner);