grafanaui.release.ts 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  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. }
  15. const promptBumpType = async () => {
  16. return inquirer.prompt<{ type: VersionBumpType }>([
  17. {
  18. type: 'list',
  19. message: 'Select version bump',
  20. name: 'type',
  21. choices: ['prerelease', 'patch', 'minor', 'major'],
  22. validate: answer => {
  23. if (answer.length < 1) {
  24. return 'You must choose something';
  25. }
  26. return true;
  27. },
  28. },
  29. ]);
  30. };
  31. const promptPrereleaseId = async (message = 'Is this a prerelease?', allowNo = true) => {
  32. return inquirer.prompt<{ id: string }>([
  33. {
  34. type: 'list',
  35. message: message,
  36. name: 'id',
  37. choices: allowNo ? ['no', 'alpha', 'beta'] : ['alpha', 'beta'],
  38. validate: answer => {
  39. if (answer.length < 1) {
  40. return 'You must choose something';
  41. }
  42. return true;
  43. },
  44. },
  45. ]);
  46. };
  47. const promptConfirm = async (message?: string) => {
  48. return inquirer.prompt<{ confirmed: boolean }>([
  49. {
  50. type: 'confirm',
  51. message: message || 'Is that correct?',
  52. name: 'confirmed',
  53. default: false,
  54. },
  55. ]);
  56. };
  57. const bumpVersion = (version: string) =>
  58. useSpinner<void>(`Saving version ${version} to package.json`, async () => {
  59. changeCwdToGrafanaUi();
  60. await execa('npm', ['version', version]);
  61. changeCwdToGrafanaUiDist();
  62. const pkg = require(`${process.cwd()}/package.json`);
  63. pkg.version = version;
  64. await savePackage({ path: `${process.cwd()}/package.json`, pkg });
  65. })();
  66. const publishPackage = (name: string, version: string) =>
  67. useSpinner<void>(`Publishing ${name} @ ${version} to npm registry...`, async () => {
  68. changeCwdToGrafanaUiDist();
  69. console.log(chalk.yellowBright.bold(`\nReview dist package.json before proceeding!\n`));
  70. const { confirmed } = await promptConfirm('Are you ready to publish to npm?');
  71. if (!confirmed) {
  72. process.exit();
  73. }
  74. await execa('npm', ['publish', '--access', 'public']);
  75. })();
  76. const ensureMasterBranch = async () => {
  77. const currentBranch = await execa.stdout('git', ['symbolic-ref', '--short', 'HEAD']);
  78. const status = await execa.stdout('git', ['status', '--porcelain']);
  79. if (currentBranch !== 'master' && status !== '') {
  80. console.error(chalk.red.bold('You need to be on clean master branch to release @grafana/ui'));
  81. process.exit(1);
  82. }
  83. };
  84. const releaseTaskRunner: TaskRunner<ReleaseTaskOptions> = async ({ publishToNpm, usePackageJsonVersion }) => {
  85. if (publishToNpm) {
  86. await ensureMasterBranch();
  87. }
  88. await execTask(buildTask)();
  89. let releaseConfirmed = false;
  90. let nextVersion;
  91. changeCwdToGrafanaUiDist();
  92. const pkg = require(`${process.cwd()}/package.json`);
  93. console.log(`Current version: ${pkg.version}`);
  94. do {
  95. if (!usePackageJsonVersion) {
  96. const { type } = await promptBumpType();
  97. console.log(type);
  98. if (type === 'prerelease') {
  99. const { id } = await promptPrereleaseId('What kind of prerelease?', false);
  100. nextVersion = semver.inc(pkg.version, type, id);
  101. } else {
  102. const { id } = await promptPrereleaseId();
  103. if (id !== 'no') {
  104. nextVersion = semver.inc(pkg.version, `pre${type}`, id);
  105. } else {
  106. nextVersion = semver.inc(pkg.version, type);
  107. }
  108. }
  109. } else {
  110. nextVersion = pkg.version;
  111. }
  112. console.log(chalk.yellowBright.bold(`You are going to release a new version of ${pkg.name}`));
  113. if (usePackageJsonVersion) {
  114. console.log(chalk.green(`Version based on package.json: `), chalk.bold.yellowBright(`${nextVersion}`));
  115. } else {
  116. console.log(chalk.green(`Version bump: ${pkg.version} ->`), chalk.bold.yellowBright(`${nextVersion}`));
  117. }
  118. const { confirmed } = await promptConfirm();
  119. releaseConfirmed = confirmed;
  120. } while (!releaseConfirmed);
  121. if (!usePackageJsonVersion) {
  122. await bumpVersion(nextVersion);
  123. }
  124. if (publishToNpm) {
  125. await publishPackage(pkg.name, nextVersion);
  126. console.log(chalk.green(`\nVersion ${nextVersion} of ${pkg.name} succesfully released!`));
  127. console.log(chalk.yellow(`\nUpdated @grafana/ui/package.json with version bump created - COMMIT THIS FILE!`));
  128. process.exit();
  129. } else {
  130. console.log(
  131. chalk.green(
  132. `\nVersion ${nextVersion} of ${pkg.name} succesfully prepared for release. See packages/grafana-ui/dist`
  133. )
  134. );
  135. console.log(chalk.green(`\nTo publish to npm registry run`), chalk.bold.blue(`npm run gui:publish`));
  136. }
  137. };
  138. export const releaseTask = new Task<ReleaseTaskOptions>();
  139. releaseTask.setName('@grafana/ui release');
  140. releaseTask.setRunner(releaseTaskRunner);