@mailmodo/cli 0.0.11 → 0.0.12-beta.pr14.20

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.
@@ -3,6 +3,7 @@ import { input } from '@inquirer/prompts';
3
3
  import chalk from 'chalk';
4
4
  import { BaseCommand } from '../../lib/base-command.js';
5
5
  import { API_ENDPOINTS, DNS_GUIDE_URL } from '../../lib/constants.js';
6
+ import { saveConfig } from '../../lib/config.js';
6
7
  import { saveYaml } from '../../lib/yaml-config.js';
7
8
  export default class Domain extends BaseCommand {
8
9
  static description = 'Set up and verify your sending domain';
@@ -24,22 +25,22 @@ export default class Domain extends BaseCommand {
24
25
  };
25
26
  async run() {
26
27
  const { flags } = await this.parse(Domain);
27
- await this.ensureAuth();
28
+ const config = await this.ensureAuth();
28
29
  if (flags.verify) {
29
- await this.verifyDomain(flags.json);
30
+ await this.verifyDomain(flags.json, config);
30
31
  return;
31
32
  }
32
33
  if (flags.status) {
33
- await this.showDomainStatus(flags.json);
34
+ await this.showDomainStatus(flags.json, config);
34
35
  return;
35
36
  }
36
- await this.setupDomain(flags);
37
+ await this.setupDomain(flags, config);
37
38
  }
38
39
  /**
39
40
  * Interactive domain setup: collects domain, sender email, and business address,
40
41
  * then calls the API to retrieve the required DNS records.
41
42
  */
42
- async setupDomain(flags) {
43
+ async setupDomain(flags, config) {
43
44
  const yamlConfig = await this.ensureYaml();
44
45
  this.log(`\n ${'─'.repeat(53)}`);
45
46
  this.log(` ${chalk.bold('DOMAIN SETUP')}`);
@@ -84,6 +85,7 @@ export default class Domain extends BaseCommand {
84
85
  yamlConfig.project.fromEmail = senderEmail;
85
86
  yamlConfig.project.address = address;
86
87
  await saveYaml(yamlConfig);
88
+ await saveConfig({ ...config, domain });
87
89
  const records = response.data?.dnsRecords || [];
88
90
  if (flags.json) {
89
91
  this.log(JSON.stringify({ dnsRecords: records, domain }, null, 2));
@@ -104,15 +106,21 @@ export default class Domain extends BaseCommand {
104
106
  message: "Press Enter once you've added the records, or 'skip'.",
105
107
  });
106
108
  if (action.toLowerCase() !== 'skip') {
107
- await this.verifyDomain(false);
109
+ await this.verifyDomain(false, { ...config, domain });
108
110
  }
109
111
  }
110
112
  }
111
113
  /**
112
114
  * Calls the domain verification API and displays pass/fail for each DNS record.
113
115
  */
114
- async verifyDomain(jsonOutput) {
115
- const response = await this.withApiSpinner({ json: jsonOutput, text: ' Checking DNS...' }, () => this.apiClient.get(API_ENDPOINTS.DOMAIN_VERIFY));
116
+ async verifyDomain(jsonOutput, config) {
117
+ if (!config.domain) {
118
+ this.error(`No domain configured. Run ${chalk.cyan('mailmodo domain')} to set up your sending domain.`);
119
+ }
120
+ const domain = config.domain;
121
+ const response = await this.withApiSpinner({ json: jsonOutput, text: ' Checking DNS...' }, () => this.apiClient.get(API_ENDPOINTS.DOMAIN_VERIFY, {
122
+ domain,
123
+ }));
116
124
  if (!response.ok) {
117
125
  this.handleApiError(response);
118
126
  }
@@ -121,16 +129,16 @@ export default class Domain extends BaseCommand {
121
129
  this.log(JSON.stringify({ dkim, dmarc, spf }, null, 2));
122
130
  return;
123
131
  }
124
- this.log(` SPF ${spf === 'pass' ? chalk.green('✓') : chalk.red('✗ Not found')}`);
125
- this.log(` DKIM ${dkim === 'pass' ? chalk.green('✓') : chalk.red('✗ Not found')}`);
126
- this.log(` DMARC ${dmarc === 'pass' ? chalk.green('✓') : chalk.red('✗ Not found')}`);
127
- const allPassed = spf === 'pass' && dkim === 'pass' && dmarc === 'pass';
132
+ this.log(` SPF ${spf ? chalk.green('✓') : chalk.red('✗ Not found')}`);
133
+ this.log(` DKIM ${dkim ? chalk.green('✓') : chalk.red('✗ Not found')}`);
134
+ this.log(` DMARC ${dmarc ? chalk.green('✓') : chalk.red('✗ Not found')}`);
135
+ const allPassed = spf && dkim && dmarc;
128
136
  if (allPassed) {
129
137
  this.log(`\n ${chalk.green('✓')} Domain verified.\n`);
130
138
  }
131
139
  else {
132
140
  this.log(`\n ${chalk.yellow('Some records failed.')}`);
133
- if (dkim !== 'pass') {
141
+ if (!dkim) {
134
142
  this.log(`\n DKIM common mistakes:`);
135
143
  this.log(` - Using TXT instead of CNAME record type`);
136
144
  this.log(` - Including the full domain in the Host field`);
@@ -144,8 +152,14 @@ export default class Domain extends BaseCommand {
144
152
  * Displays domain health metrics including verification status,
145
153
  * bounce rate, and spam complaint rate.
146
154
  */
147
- async showDomainStatus(jsonOutput) {
148
- const response = await this.withApiSpinner({ json: jsonOutput, text: ' Loading domain status...' }, () => this.apiClient.get(API_ENDPOINTS.DOMAIN_STATUS));
155
+ async showDomainStatus(jsonOutput, config) {
156
+ if (!config.domain) {
157
+ this.error(`No domain configured. Run ${chalk.cyan('mailmodo domain')} to set up your sending domain.`);
158
+ }
159
+ const domain = config.domain;
160
+ const response = await this.withApiSpinner({ json: jsonOutput, text: ' Loading domain status...' }, () => this.apiClient.get(API_ENDPOINTS.DOMAIN_STATUS, {
161
+ domain,
162
+ }));
149
163
  if (!response.ok) {
150
164
  this.handleApiError(response);
151
165
  }
@@ -2,7 +2,7 @@ import { Flags } from '@oclif/core';
2
2
  import { editor, input, select } from '@inquirer/prompts';
3
3
  import chalk from 'chalk';
4
4
  import { BaseCommand } from '../../lib/base-command.js';
5
- import { API_ENDPOINTS, DEFAULT_BRAND_COLOR } from '../../lib/constants.js';
5
+ import { API_ENDPOINTS, DEFAULT_BRAND_COLOR, DEFAULT_MONTHLY_CAP, } from '../../lib/constants.js';
6
6
  import { saveTemplate, saveYaml, } from '../../lib/yaml-config.js';
7
7
  function isValidUrl(value) {
8
8
  try {
@@ -106,13 +106,31 @@ export default class Init extends BaseCommand {
106
106
  analysisPayload = JSON.parse(editedAnalysis);
107
107
  }
108
108
  }
109
- const generateResponse = await this.withApiSpinner({
110
- json: flags.json,
111
- text: ' Generating email templates...',
112
- }, () => this.apiClient.post(API_ENDPOINTS.GENERATE, {
113
- analysis: analysisPayload,
114
- productUrl,
115
- }));
109
+ // TODO: Remove dummy data once the Generate API is ready.
110
+ const GENERATE_API_READY = false;
111
+ const generateResponse = GENERATE_API_READY
112
+ ? await this.withApiSpinner({
113
+ json: flags.json,
114
+ text: ' Generating email templates...',
115
+ }, () => this.apiClient.post(API_ENDPOINTS.GENERATE, {
116
+ analysis: analysisPayload,
117
+ productUrl,
118
+ }))
119
+ : {
120
+ ok: true,
121
+ status: 200,
122
+ data: {
123
+ emails: analysisPayload.recommendedEmails.map((rec) => ({
124
+ ctaText: 'Get Started',
125
+ html: `<html><body><h1>${analysisPayload.productName}</h1><p>This is a placeholder email for <strong>${rec.id}</strong>.</p><a href="{{cta_url}}">Get Started</a></body></html>`,
126
+ id: rec.id,
127
+ plainHtml: `<html><body><p>This is a placeholder email for ${rec.id}.</p><a href="{{cta_url}}">Get Started</a></body></html>`,
128
+ previewText: `Welcome — ${rec.goal}`,
129
+ subject: `${analysisPayload.productName}: ${rec.goal}`,
130
+ suggestedCtaPath: '/dashboard',
131
+ })),
132
+ },
133
+ };
116
134
  if (!generateResponse.ok) {
117
135
  this.handleApiError(generateResponse);
118
136
  }
@@ -140,6 +158,7 @@ export default class Init extends BaseCommand {
140
158
  fromEmail: '',
141
159
  fromName: `Team ${analysisPayload.productName}`,
142
160
  logoUrl: analysisPayload.brand?.logoUrl || '',
161
+ monthlyCap: DEFAULT_MONTHLY_CAP,
143
162
  name: analysisPayload.productName,
144
163
  replyTo: '',
145
164
  type: analysisPayload.pricingModel,
@@ -1,6 +1,7 @@
1
1
  export interface MailmodoConfig {
2
2
  accountName?: string;
3
3
  apiKey: string;
4
+ domain?: string;
4
5
  email?: string;
5
6
  freeRemaining?: number;
6
7
  }
@@ -1,4 +1,4 @@
1
- export declare const API_BASE_URL: string;
1
+ export declare const API_BASE_URL = "https://app-vertex-debug.azurewebsites.net";
2
2
  export declare const API_ENDPOINTS: Readonly<{
3
3
  ANALYTICS: "/analytics";
4
4
  ANALYZE: "/analyze";
@@ -23,5 +23,6 @@ export declare const DOCS_URL = "https://mailmodo.com/docs/cli";
23
23
  export declare const DNS_GUIDE_URL = "https://mailmodo.com/docs/dns";
24
24
  export declare const PREVIEW_PORT = 3421;
25
25
  export declare const DEFAULT_BRAND_COLOR = "#1A56DB";
26
+ export declare const DEFAULT_MONTHLY_CAP = 5;
26
27
  export declare const TEMPLATES_DIR = "mailmodo";
27
28
  export declare const YAML_FILE = "mailmodo.yaml";
@@ -1,6 +1,7 @@
1
1
  /** Set by `bin/dev.js` when running the CLI locally (tsx bootstrap). */
2
2
  const DEV_API_BASE_URL = 'https://app-vertex-debug.azurewebsites.net';
3
- const PRODUCTION_API_BASE_URL = 'https://api.mailmodo.com';
3
+ // const PRODUCTION_API_BASE_URL = 'https://api.mailmodo.com';
4
+ const PRODUCTION_API_BASE_URL = 'https://app-vertex-debug.azurewebsites.net';
4
5
  export const API_BASE_URL = process.env.MAILMODO_DEV_TSX
5
6
  ? DEV_API_BASE_URL
6
7
  : PRODUCTION_API_BASE_URL;
@@ -28,5 +29,6 @@ export const DOCS_URL = 'https://mailmodo.com/docs/cli';
28
29
  export const DNS_GUIDE_URL = 'https://mailmodo.com/docs/dns';
29
30
  export const PREVIEW_PORT = 3421;
30
31
  export const DEFAULT_BRAND_COLOR = '#1A56DB';
32
+ export const DEFAULT_MONTHLY_CAP = 5;
31
33
  export const TEMPLATES_DIR = 'mailmodo';
32
34
  export const YAML_FILE = 'mailmodo.yaml';
@@ -205,13 +205,19 @@
205
205
  "index.js"
206
206
  ]
207
207
  },
208
- "emails": {
208
+ "edit": {
209
209
  "aliases": [],
210
- "args": {},
211
- "description": "List and view configured email sequences",
210
+ "args": {
211
+ "id": {
212
+ "description": "Email ID to edit",
213
+ "name": "id",
214
+ "required": true
215
+ }
216
+ },
217
+ "description": "Edit an email using AI-assisted natural language changes",
212
218
  "examples": [
213
- "<%= config.bin %> emails",
214
- "<%= config.bin %> emails --json"
219
+ "<%= config.bin %> edit welcome",
220
+ "<%= config.bin %> edit welcome --change \"make subject more urgent\" --yes"
215
221
  ],
216
222
  "flags": {
217
223
  "json": {
@@ -226,11 +232,18 @@
226
232
  "name": "yes",
227
233
  "allowNo": false,
228
234
  "type": "boolean"
235
+ },
236
+ "change": {
237
+ "description": "Natural language description of the change",
238
+ "name": "change",
239
+ "hasDynamicHelp": false,
240
+ "multiple": false,
241
+ "type": "option"
229
242
  }
230
243
  },
231
244
  "hasDynamicHelp": false,
232
245
  "hiddenAliases": [],
233
- "id": "emails",
246
+ "id": "edit",
234
247
  "pluginAlias": "@mailmodo/cli",
235
248
  "pluginName": "@mailmodo/cli",
236
249
  "pluginType": "core",
@@ -240,23 +253,17 @@
240
253
  "relativePath": [
241
254
  "dist",
242
255
  "commands",
243
- "emails",
256
+ "edit",
244
257
  "index.js"
245
258
  ]
246
259
  },
247
- "edit": {
260
+ "emails": {
248
261
  "aliases": [],
249
- "args": {
250
- "id": {
251
- "description": "Email ID to edit",
252
- "name": "id",
253
- "required": true
254
- }
255
- },
256
- "description": "Edit an email using AI-assisted natural language changes",
262
+ "args": {},
263
+ "description": "List and view configured email sequences",
257
264
  "examples": [
258
- "<%= config.bin %> edit welcome",
259
- "<%= config.bin %> edit welcome --change \"make subject more urgent\" --yes"
265
+ "<%= config.bin %> emails",
266
+ "<%= config.bin %> emails --json"
260
267
  ],
261
268
  "flags": {
262
269
  "json": {
@@ -271,18 +278,11 @@
271
278
  "name": "yes",
272
279
  "allowNo": false,
273
280
  "type": "boolean"
274
- },
275
- "change": {
276
- "description": "Natural language description of the change",
277
- "name": "change",
278
- "hasDynamicHelp": false,
279
- "multiple": false,
280
- "type": "option"
281
281
  }
282
282
  },
283
283
  "hasDynamicHelp": false,
284
284
  "hiddenAliases": [],
285
- "id": "edit",
285
+ "id": "emails",
286
286
  "pluginAlias": "@mailmodo/cli",
287
287
  "pluginName": "@mailmodo/cli",
288
288
  "pluginType": "core",
@@ -292,7 +292,7 @@
292
292
  "relativePath": [
293
293
  "dist",
294
294
  "commands",
295
- "edit",
295
+ "emails",
296
296
  "index.js"
297
297
  ]
298
298
  },
@@ -618,5 +618,5 @@
618
618
  ]
619
619
  }
620
620
  },
621
- "version": "0.0.11"
621
+ "version": "0.0.12-beta.pr14.20"
622
622
  }
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@mailmodo/cli",
3
3
  "description": "Email lifecycle automation for the AI-native builder generation.",
4
- "version": "0.0.11",
4
+ "version": "0.0.12-beta.pr14.20",
5
5
  "author": "provishalk",
6
6
  "bin": {
7
7
  "mailmodo": "bin/run.js"
@@ -24,13 +24,17 @@
24
24
  "@types/chai": "^4",
25
25
  "@types/js-yaml": "^4.0.9",
26
26
  "@types/mocha": "^10",
27
- "@types/node": "^18",
27
+ "@types/node": "^18.19.130",
28
28
  "chai": "^4",
29
29
  "eslint": "^9",
30
30
  "eslint-config-oclif": "^6",
31
31
  "eslint-config-prettier": "^10",
32
+ "eslint-plugin-unused-imports": "^4.4.1",
33
+ "husky": "^9.1.7",
34
+ "lint-staged": "^16.4.0",
32
35
  "mocha": "^11",
33
36
  "oclif": "^4",
37
+ "prettier": "^3.8.2",
34
38
  "shx": "^0.3.3",
35
39
  "ts-node": "^10",
36
40
  "tsx": "^4.21.0",
@@ -69,7 +73,15 @@
69
73
  "posttest": "npm run lint",
70
74
  "prepack": "oclif manifest && oclif readme",
71
75
  "test": "mocha --forbid-only \"test/**/*.test.ts\"",
72
- "version": "oclif readme && git add README.md"
76
+ "version": "oclif readme && git add README.md",
77
+ "prepare": "husky"
73
78
  },
74
- "types": "dist/index.d.ts"
79
+ "types": "dist/index.d.ts",
80
+ "lint-staged": {
81
+ "*.{ts,tsx,js,jsx,mjs,cjs}": [
82
+ "prettier --write",
83
+ "eslint --fix"
84
+ ],
85
+ "*.{json,md,yaml,yml}": "prettier --write"
86
+ }
75
87
  }