utils.mjs 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. import 'zx/globals';
  2. import { parse as parseToml } from '@iarna/toml';
  3. process.env.FORCE_COLOR = 3;
  4. process.env.CARGO_TERM_COLOR = 'always';
  5. export const workingDirectory = (await $`pwd`.quiet()).toString().trim();
  6. export function getAllProgramIdls() {
  7. return getAllProgramFolders().map((folder) =>
  8. path.join(workingDirectory, folder, 'idl.json')
  9. );
  10. }
  11. export function getExternalProgramOutputDir() {
  12. const config = getCargoMetadata()?.solana?.['external-programs-output'];
  13. return path.join(workingDirectory, config ?? 'target/deploy');
  14. }
  15. export function getExternalProgramAddresses() {
  16. const addresses = getProgramFolders().flatMap(
  17. (folder) => getCargoMetadata(folder)?.solana?.['program-dependencies'] ?? []
  18. );
  19. return Array.from(new Set(addresses));
  20. }
  21. export function getExternalAccountAddresses() {
  22. const addresses = getProgramFolders().flatMap(
  23. (folder) => getCargoMetadata(folder)?.solana?.['account-dependencies'] ?? []
  24. );
  25. return Array.from(new Set(addresses));
  26. }
  27. let didWarnAboutMissingPrograms = false;
  28. export function getProgramFolders() {
  29. let programs;
  30. if (process.env.PROGRAMS) {
  31. try {
  32. programs = JSON.parse(process.env.PROGRAMS);
  33. } catch (error) {
  34. programs = process.env.PROGRAMS.split(/\s+/);
  35. }
  36. } else {
  37. programs = getAllProgramFolders();
  38. }
  39. const filteredPrograms = programs.filter((program) =>
  40. fs.existsSync(path.join(workingDirectory, program))
  41. );
  42. if (
  43. filteredPrograms.length !== programs.length &&
  44. !didWarnAboutMissingPrograms
  45. ) {
  46. didWarnAboutMissingPrograms = true;
  47. programs
  48. .filter((program) => !filteredPrograms.includes(program))
  49. .forEach((program) => {
  50. echo(chalk.yellow(`Program not found: ${workingDirectory}/${program}`));
  51. });
  52. }
  53. return filteredPrograms;
  54. }
  55. export function getAllProgramFolders() {
  56. return getCargo().workspace.members.filter(
  57. (member) =>
  58. (getCargo(member).lib?.['crate-type'] ?? []).includes('cdylib') &&
  59. // Exclude the pinocchio-token-program crate.
  60. getCargo(member).package?.name !== 'pinocchio-token-program'
  61. );
  62. }
  63. export function getCargo(folder) {
  64. return parseToml(
  65. fs.readFileSync(
  66. path.join(workingDirectory, folder ? folder : '.', 'Cargo.toml'),
  67. 'utf8'
  68. )
  69. );
  70. }
  71. export function getCargoMetadata(folder) {
  72. const cargo = getCargo(folder);
  73. return folder ? cargo?.package?.metadata : cargo?.workspace?.metadata;
  74. }
  75. export function getSolanaVersion() {
  76. return getCargoMetadata()?.cli?.solana;
  77. }
  78. export function getToolchain(operation) {
  79. return getCargoMetadata()?.toolchains?.[operation];
  80. }
  81. export function getToolchainArgument(operation) {
  82. const channel = getToolchain(operation);
  83. return channel ? `+${channel}` : '';
  84. }
  85. export function cliArguments() {
  86. return process.argv.slice(3);
  87. }
  88. export function popArgument(args, arg) {
  89. const index = args.indexOf(arg);
  90. if (index >= 0) {
  91. args.splice(index, 1);
  92. }
  93. return index >= 0;
  94. }
  95. export function partitionArguments(args, delimiter) {
  96. const index = args.indexOf(delimiter);
  97. return index >= 0
  98. ? [args.slice(0, index), args.slice(index + 1)]
  99. : [args, []];
  100. }
  101. export async function getInstalledSolanaVersion() {
  102. try {
  103. const { stdout } = await $`solana --version`.quiet();
  104. return stdout.match(/(\d+\.\d+\.\d+)/)?.[1];
  105. } catch (error) {
  106. return '';
  107. }
  108. }