1
0

utils.mjs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  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((member) =>
  57. (getCargo(member).lib?.['crate-type'] ?? []).includes('cdylib')
  58. );
  59. }
  60. export function getCargo(folder) {
  61. return parseToml(
  62. fs.readFileSync(
  63. path.join(workingDirectory, folder ? folder : '.', 'Cargo.toml'),
  64. 'utf8'
  65. )
  66. );
  67. }
  68. export function getCargoMetadata(folder) {
  69. const cargo = getCargo(folder);
  70. return folder ? cargo?.package?.metadata : cargo?.workspace?.metadata;
  71. }
  72. export function getSolanaVersion() {
  73. return getCargoMetadata()?.cli?.solana;
  74. }
  75. export function getToolchain(operation) {
  76. return getCargoMetadata()?.toolchains?.[operation];
  77. }
  78. export function getToolchainArgument(operation) {
  79. const channel = getToolchain(operation);
  80. return channel ? `+${channel}` : '';
  81. }
  82. export function cliArguments() {
  83. return process.argv.slice(3);
  84. }
  85. export function popArgument(args, arg) {
  86. const index = args.indexOf(arg);
  87. if (index >= 0) {
  88. args.splice(index, 1);
  89. }
  90. return index >= 0;
  91. }
  92. export function partitionArguments(args, delimiter) {
  93. const index = args.indexOf(delimiter);
  94. return index >= 0
  95. ? [args.slice(0, index), args.slice(index + 1)]
  96. : [args, []];
  97. }
  98. export async function getInstalledSolanaVersion() {
  99. try {
  100. const { stdout } = await $`solana --version`.quiet();
  101. return stdout.match(/(\d+\.\d+\.\d+)/)?.[1];
  102. } catch (error) {
  103. return '';
  104. }
  105. }