vmlive 1.0.13 → 1.0.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (2) hide show
  1. package/package.json +1 -1
  2. package/src/cli.js +95 -0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "vmlive",
3
- "version": "1.0.13",
3
+ "version": "1.0.14",
4
4
  "description": "Local development VM for custom Serverless PaaS",
5
5
  "type": "module",
6
6
  "bin": {
package/src/cli.js CHANGED
@@ -716,6 +716,98 @@ const runLlm = async () => {
716
716
  process.exit(1);
717
717
  }
718
718
  };
719
+ const runDb = async () => {
720
+ const subcommand = process.argv[3];
721
+ const envArg = process.argv[4];
722
+
723
+ if (subcommand !== 'push') {
724
+ console.error('\x1b[31m❌ Unknown DB command:\x1b[0m Must be `vmlive db push local` or `vmlive db push remote`');
725
+ process.exit(1);
726
+ }
727
+
728
+ if (envArg !== 'local' && envArg !== 'remote') {
729
+ console.error('\x1b[31m❌ Missing Target:\x1b[0m Please specify environment `local` or `remote`');
730
+ process.exit(1);
731
+ }
732
+
733
+ if (!fs.existsSync(CONFIG_PATH)) {
734
+ console.error('\x1b[31m❌ Project Not Found:\x1b[0m You must be inside a vmlive project folder.');
735
+ process.exit(1);
736
+ }
737
+
738
+ const vmConf = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf-8'));
739
+ const migrationsDir = path.resolve('migrations');
740
+
741
+ if (!fs.existsSync(migrationsDir)) {
742
+ console.log('\x1b[33m⚠ No migrations folder found.\x1b[0m Skipping.');
743
+ return;
744
+ }
745
+
746
+ const files = fs.readdirSync(migrationsDir)
747
+ .filter(f => f.endsWith('.sql'))
748
+ .sort();
749
+
750
+ if (files.length === 0) {
751
+ console.log('\x1b[32m✔ No pending migrations found.\x1b[0m');
752
+ return;
753
+ }
754
+
755
+ console.log(`\x1b[36mFound ${files.length} migration files. Pushing to [${envArg.toUpperCase()}]...\x1b[0m`);
756
+
757
+ let combinedSql = '';
758
+ for (const file of files) {
759
+ combinedSql += fs.readFileSync(path.join(migrationsDir, file), 'utf-8') + '\n';
760
+ }
761
+
762
+ if (envArg === 'local') {
763
+ console.log('\x1b[90mSyncing Local V8 Emulator D1...\x1b[0m');
764
+ const mf = new Emulator({
765
+ modules: true,
766
+ script: 'export default { fetch: () => new Response("ok") }',
767
+ d1Databases: ["DB"]
768
+ });
769
+ try {
770
+ const db = await mf.getD1Database("DB");
771
+ await db.exec(combinedSql);
772
+ console.log('\x1b[32m✔ Local Database push successful.\x1b[0m');
773
+ } catch(e) {
774
+ console.error('\n\x1b[31m❌ Local execute failed:\x1b[0m', e.message);
775
+ }
776
+ await mf.dispose();
777
+ } else {
778
+ const GATEKEEPER_URL = process.env.GATEKEEPER_URL || 'https://api.vm.live';
779
+ const configPath = path.join(os.homedir(), '.vm-config.json');
780
+ let jwtToken = process.env.VM_API_TOKEN;
781
+ if (!jwtToken && fs.existsSync(configPath)) {
782
+ jwtToken = JSON.parse(fs.readFileSync(configPath, 'utf-8')).token;
783
+ }
784
+
785
+ if (!jwtToken) {
786
+ console.error('\x1b[31m❌ Unauthorized.\x1b[0m Please run `vmlive login` first.');
787
+ process.exit(1);
788
+ }
789
+
790
+ const { projectId } = vmConf;
791
+ console.log(`\x1b[90mSyncing Cloudflare Edge D1 [${projectId}]...\x1b[0m`);
792
+
793
+ try {
794
+ const res = await fetch(`${GATEKEEPER_URL}/api/projects/${projectId}/database/migrate`, {
795
+ method: 'POST',
796
+ headers: { 'Authorization': `Bearer ${jwtToken}`, 'Content-Type': 'application/json' },
797
+ body: JSON.stringify({ sql: combinedSql })
798
+ });
799
+
800
+ if (res.ok) {
801
+ console.log('\n\x1b[32m✔ Cloudflare D1 Remote Pipeline successfully upgraded.\x1b[0m');
802
+ } else {
803
+ console.error('\n\x1b[31m❌ Remote Migration Error:\x1b[0m', res.status, await res.text());
804
+ }
805
+ } catch (err) {
806
+ console.error('\n\x1b[31m❌ Network Failure:\x1b[0m', err.message);
807
+ }
808
+ }
809
+ };
810
+
719
811
  const main = async () => {
720
812
  const command = process.argv[2];
721
813
  if (command === 'init') {
@@ -732,6 +824,8 @@ const main = async () => {
732
824
  await runUpdate();
733
825
  } else if (command === 'llm') {
734
826
  await runLlm();
827
+ } else if (command === 'db') {
828
+ await runDb();
735
829
  } else if (command === 'help' || !command) {
736
830
  console.log('\n\x1b[1m\x1b[36mvm.live\x1b[0m - Serverless Edge Compute Engine');
737
831
  console.log('Usage: vmlive <command>\n');
@@ -741,6 +835,7 @@ const main = async () => {
741
835
  console.log(' \x1b[32mdev\x1b[0m Start the local emulator and dashboard over localhost');
742
836
  console.log(' \x1b[32mlogin\x1b[0m Authenticate your local environment securely via Stripe/OAuth');
743
837
  console.log(' \x1b[32mdeploy\x1b[0m Upload your footprint to the vm.live edge platform');
838
+ console.log(' \x1b[32mdb\x1b[0m Manage Cloudflare D1 schema migrations (`db push local|remote`)');
744
839
  console.log(' \x1b[32mllm\x1b[0m Download the official AI Context Rules for agentic workflows');
745
840
  console.log(' \x1b[32mupdate\x1b[0m Upgrade this CLI engine to the latest published version');
746
841
  console.log(' \x1b[32mwhich\x1b[0m Display the currently active CLI version');