@neetru/cli 2.7.5 → 2.8.0
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.
- package/CHANGELOG.md +220 -220
- package/README.md +137 -137
- package/dist/cli-kit/format.d.ts +49 -0
- package/dist/cli-kit/format.js +88 -0
- package/dist/cli-kit/format.js.map +1 -0
- package/dist/cli-kit/glyphs.d.ts +22 -0
- package/dist/cli-kit/glyphs.js +22 -0
- package/dist/cli-kit/glyphs.js.map +1 -0
- package/dist/cli-kit/index.d.ts +13 -0
- package/dist/cli-kit/index.js +12 -0
- package/dist/cli-kit/index.js.map +1 -0
- package/dist/cli-kit/palette.d.ts +10 -0
- package/dist/cli-kit/palette.js +36 -0
- package/dist/cli-kit/palette.js.map +1 -0
- package/dist/commands/ai.js +8 -8
- package/dist/commands/autocomplete.js +34 -34
- package/dist/commands/db.d.ts +87 -7
- package/dist/commands/db.js +697 -126
- package/dist/commands/db.js.map +1 -1
- package/dist/commands/deploy.d.ts +5 -0
- package/dist/commands/deploy.js +68 -0
- package/dist/commands/deploy.js.map +1 -1
- package/dist/commands/dev.d.ts +68 -0
- package/dist/commands/dev.js +345 -0
- package/dist/commands/dev.js.map +1 -0
- package/dist/commands/init.js +121 -121
- package/dist/commands/new.d.ts +6 -0
- package/dist/commands/new.js +31 -10
- package/dist/commands/new.js.map +1 -1
- package/dist/commands/products-db.d.ts +1 -1
- package/dist/commands/products-db.js +17 -4
- package/dist/commands/products-db.js.map +1 -1
- package/dist/index.js +258 -42
- package/dist/index.js.map +1 -1
- package/dist/lib/ai/context.js +90 -90
- package/dist/lib/db-local/db-json.d.ts +63 -0
- package/dist/lib/db-local/db-json.js +189 -0
- package/dist/lib/db-local/db-json.js.map +1 -0
- package/dist/lib/db-local/env.d.ts +26 -0
- package/dist/lib/db-local/env.js +64 -0
- package/dist/lib/db-local/env.js.map +1 -0
- package/dist/lib/db-local/fingerprint.d.ts +8 -0
- package/dist/lib/db-local/fingerprint.js +28 -0
- package/dist/lib/db-local/fingerprint.js.map +1 -0
- package/dist/lib/db-local/index.d.ts +15 -0
- package/dist/lib/db-local/index.js +14 -0
- package/dist/lib/db-local/index.js.map +1 -0
- package/dist/lib/db-pipeline/build-deps.d.ts +14 -0
- package/dist/lib/db-pipeline/build-deps.js +158 -0
- package/dist/lib/db-pipeline/build-deps.js.map +1 -0
- package/dist/lib/db-pipeline/errors.d.ts +29 -0
- package/dist/lib/db-pipeline/errors.js +29 -0
- package/dist/lib/db-pipeline/errors.js.map +1 -0
- package/dist/lib/db-pipeline/index.d.ts +26 -0
- package/dist/lib/db-pipeline/index.js +25 -0
- package/dist/lib/db-pipeline/index.js.map +1 -0
- package/dist/lib/db-pipeline/pipeline.d.ts +13 -0
- package/dist/lib/db-pipeline/pipeline.js +119 -0
- package/dist/lib/db-pipeline/pipeline.js.map +1 -0
- package/dist/lib/db-pipeline/rehearse.d.ts +99 -0
- package/dist/lib/db-pipeline/rehearse.js +219 -0
- package/dist/lib/db-pipeline/rehearse.js.map +1 -0
- package/dist/lib/db-pipeline/types.d.ts +112 -0
- package/dist/lib/db-pipeline/types.js +20 -0
- package/dist/lib/db-pipeline/types.js.map +1 -0
- package/package.json +63 -62
- package/templates/auth/callback.ts +22 -22
- package/templates/auth/sign-in.tsx +41 -41
- package/templates/billing/checkout.ts +22 -22
- package/templates/billing/page.tsx +43 -43
- package/templates/support/ticket-form.tsx +68 -68
- package/templates/usage/track.ts +30 -30
- package/templates/users/profile.tsx +43 -43
package/package.json
CHANGED
|
@@ -1,62 +1,63 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "@neetru/cli",
|
|
3
|
-
"version": "2.
|
|
4
|
-
"type": "module",
|
|
5
|
-
"description": "Neetru Developer Kit — scaffold, AI assistant, publish, build e deploy de produtos SaaS Neetru via Core (VM-based agents)",
|
|
6
|
-
"keywords": [
|
|
7
|
-
"neetru",
|
|
8
|
-
"saas",
|
|
9
|
-
"cli",
|
|
10
|
-
"devkit",
|
|
11
|
-
"scaffold",
|
|
12
|
-
"deploy",
|
|
13
|
-
"agent",
|
|
14
|
-
"vm"
|
|
15
|
-
],
|
|
16
|
-
"homepage": "https://core.neetru.com",
|
|
17
|
-
"repository": {
|
|
18
|
-
"type": "git",
|
|
19
|
-
"url": "https://github.com/Neetru/neetru-core.git",
|
|
20
|
-
"directory": "cli"
|
|
21
|
-
},
|
|
22
|
-
"author": "Neetru Inc. <neetrucode@gmail.com>",
|
|
23
|
-
"license": "MIT",
|
|
24
|
-
"bin": {
|
|
25
|
-
"neetru": "./dist/index.js"
|
|
26
|
-
},
|
|
27
|
-
"main": "./dist/index.js",
|
|
28
|
-
"files": [
|
|
29
|
-
"dist",
|
|
30
|
-
"templates",
|
|
31
|
-
"README.md",
|
|
32
|
-
"CHANGELOG.md"
|
|
33
|
-
],
|
|
34
|
-
"scripts": {
|
|
35
|
-
"build": "tsc",
|
|
36
|
-
"dev": "tsx src/index.ts",
|
|
37
|
-
"start": "node dist/index.js",
|
|
38
|
-
"prepublishOnly": "npm run build"
|
|
39
|
-
},
|
|
40
|
-
"dependencies": {
|
|
41
|
-
"@anthropic-ai/sdk": "^0.96.0",
|
|
42
|
-
"
|
|
43
|
-
"
|
|
44
|
-
"
|
|
45
|
-
"
|
|
46
|
-
"
|
|
47
|
-
"
|
|
48
|
-
"
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
"@types/
|
|
53
|
-
"
|
|
54
|
-
"
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
}
|
|
1
|
+
{
|
|
2
|
+
"name": "@neetru/cli",
|
|
3
|
+
"version": "2.8.0",
|
|
4
|
+
"type": "module",
|
|
5
|
+
"description": "Neetru Developer Kit — scaffold, AI assistant, publish, build e deploy de produtos SaaS Neetru via Core (VM-based agents)",
|
|
6
|
+
"keywords": [
|
|
7
|
+
"neetru",
|
|
8
|
+
"saas",
|
|
9
|
+
"cli",
|
|
10
|
+
"devkit",
|
|
11
|
+
"scaffold",
|
|
12
|
+
"deploy",
|
|
13
|
+
"agent",
|
|
14
|
+
"vm"
|
|
15
|
+
],
|
|
16
|
+
"homepage": "https://core.neetru.com",
|
|
17
|
+
"repository": {
|
|
18
|
+
"type": "git",
|
|
19
|
+
"url": "https://github.com/Neetru/neetru-core.git",
|
|
20
|
+
"directory": "cli"
|
|
21
|
+
},
|
|
22
|
+
"author": "Neetru Inc. <neetrucode@gmail.com>",
|
|
23
|
+
"license": "MIT",
|
|
24
|
+
"bin": {
|
|
25
|
+
"neetru": "./dist/index.js"
|
|
26
|
+
},
|
|
27
|
+
"main": "./dist/index.js",
|
|
28
|
+
"files": [
|
|
29
|
+
"dist",
|
|
30
|
+
"templates",
|
|
31
|
+
"README.md",
|
|
32
|
+
"CHANGELOG.md"
|
|
33
|
+
],
|
|
34
|
+
"scripts": {
|
|
35
|
+
"build": "tsc",
|
|
36
|
+
"dev": "tsx src/index.ts",
|
|
37
|
+
"start": "node dist/index.js",
|
|
38
|
+
"prepublishOnly": "npm run build"
|
|
39
|
+
},
|
|
40
|
+
"dependencies": {
|
|
41
|
+
"@anthropic-ai/sdk": "^0.96.0",
|
|
42
|
+
"@neetru/db-classifier": "^0.1.0",
|
|
43
|
+
"chalk": "^5.4.1",
|
|
44
|
+
"commander": "^12.1.0",
|
|
45
|
+
"conf": "^13.0.1",
|
|
46
|
+
"inquirer": "^10.2.2",
|
|
47
|
+
"openai": "^4.77.0",
|
|
48
|
+
"ora": "^8.1.1",
|
|
49
|
+
"zod": "^3.24.2"
|
|
50
|
+
},
|
|
51
|
+
"devDependencies": {
|
|
52
|
+
"@types/inquirer": "^9.0.7",
|
|
53
|
+
"@types/node": "^22.0.0",
|
|
54
|
+
"tsx": "^4.22.0",
|
|
55
|
+
"typescript": "^5.9.3"
|
|
56
|
+
},
|
|
57
|
+
"engines": {
|
|
58
|
+
"node": ">=20"
|
|
59
|
+
},
|
|
60
|
+
"publishConfig": {
|
|
61
|
+
"access": "public"
|
|
62
|
+
}
|
|
63
|
+
}
|
|
@@ -1,22 +1,22 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Template — `src/lib/neetru/auth/callback.ts` (gerado por `neetru add auth`).
|
|
3
|
-
*
|
|
4
|
-
* Handler de callback OIDC (auth code → id_token). Use em `app/oauth/callback/route.ts`.
|
|
5
|
-
*/
|
|
6
|
-
import { NextRequest, NextResponse } from 'next/server';
|
|
7
|
-
|
|
8
|
-
export const runtime = 'nodejs';
|
|
9
|
-
|
|
10
|
-
export async function GET(request: NextRequest) {
|
|
11
|
-
const { searchParams } = new URL(request.url);
|
|
12
|
-
const code = searchParams.get('code');
|
|
13
|
-
const state = searchParams.get('state');
|
|
14
|
-
|
|
15
|
-
if (!code) {
|
|
16
|
-
return NextResponse.json({ error: 'missing_code' }, { status: 400 });
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
// Trocar code → id_token contra auth.neetru.com/token
|
|
20
|
-
// Implementação detalhada em docs/PLATFORM_AUTH.md §4
|
|
21
|
-
return NextResponse.redirect(state ?? '/');
|
|
22
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Template — `src/lib/neetru/auth/callback.ts` (gerado por `neetru add auth`).
|
|
3
|
+
*
|
|
4
|
+
* Handler de callback OIDC (auth code → id_token). Use em `app/oauth/callback/route.ts`.
|
|
5
|
+
*/
|
|
6
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
7
|
+
|
|
8
|
+
export const runtime = 'nodejs';
|
|
9
|
+
|
|
10
|
+
export async function GET(request: NextRequest) {
|
|
11
|
+
const { searchParams } = new URL(request.url);
|
|
12
|
+
const code = searchParams.get('code');
|
|
13
|
+
const state = searchParams.get('state');
|
|
14
|
+
|
|
15
|
+
if (!code) {
|
|
16
|
+
return NextResponse.json({ error: 'missing_code' }, { status: 400 });
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
// Trocar code → id_token contra auth.neetru.com/token
|
|
20
|
+
// Implementação detalhada em docs/PLATFORM_AUTH.md §4
|
|
21
|
+
return NextResponse.redirect(state ?? '/');
|
|
22
|
+
}
|
|
@@ -1,41 +1,41 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Template — `src/lib/neetru/auth/sign-in.tsx` (gerado por `neetru add auth`).
|
|
3
|
-
*
|
|
4
|
-
* Botão Google → redireciona pra auth.neetru.com via SDK (`client.auth.signIn`).
|
|
5
|
-
*
|
|
6
|
-
* Customização: ajuste `redirectUri` se hospedar fora do domínio default.
|
|
7
|
-
*/
|
|
8
|
-
'use client';
|
|
9
|
-
|
|
10
|
-
import { useState } from 'react';
|
|
11
|
-
import { createNeetruClient } from '@neetru/sdk';
|
|
12
|
-
|
|
13
|
-
const client = createNeetruClient({
|
|
14
|
-
apiKey: process.env.NEXT_PUBLIC_NEETRU_API_KEY,
|
|
15
|
-
productId: process.env.NEXT_PUBLIC_NEETRU_PRODUCT_ID,
|
|
16
|
-
});
|
|
17
|
-
|
|
18
|
-
export function SignInButton() {
|
|
19
|
-
const [loading, setLoading] = useState(false);
|
|
20
|
-
|
|
21
|
-
async function handleSignIn() {
|
|
22
|
-
setLoading(true);
|
|
23
|
-
try {
|
|
24
|
-
await client.auth.signIn();
|
|
25
|
-
} catch (err) {
|
|
26
|
-
console.error('signIn failed', err);
|
|
27
|
-
setLoading(false);
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
return (
|
|
32
|
-
<button
|
|
33
|
-
type="button"
|
|
34
|
-
onClick={handleSignIn}
|
|
35
|
-
disabled={loading}
|
|
36
|
-
className="inline-flex items-center justify-center px-4 py-2 border bg-white hover:bg-zinc-50 text-zinc-900 font-medium"
|
|
37
|
-
>
|
|
38
|
-
{loading ? 'Redirecionando...' : 'Entrar com Neetru'}
|
|
39
|
-
</button>
|
|
40
|
-
);
|
|
41
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Template — `src/lib/neetru/auth/sign-in.tsx` (gerado por `neetru add auth`).
|
|
3
|
+
*
|
|
4
|
+
* Botão Google → redireciona pra auth.neetru.com via SDK (`client.auth.signIn`).
|
|
5
|
+
*
|
|
6
|
+
* Customização: ajuste `redirectUri` se hospedar fora do domínio default.
|
|
7
|
+
*/
|
|
8
|
+
'use client';
|
|
9
|
+
|
|
10
|
+
import { useState } from 'react';
|
|
11
|
+
import { createNeetruClient } from '@neetru/sdk';
|
|
12
|
+
|
|
13
|
+
const client = createNeetruClient({
|
|
14
|
+
apiKey: process.env.NEXT_PUBLIC_NEETRU_API_KEY,
|
|
15
|
+
productId: process.env.NEXT_PUBLIC_NEETRU_PRODUCT_ID,
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
export function SignInButton() {
|
|
19
|
+
const [loading, setLoading] = useState(false);
|
|
20
|
+
|
|
21
|
+
async function handleSignIn() {
|
|
22
|
+
setLoading(true);
|
|
23
|
+
try {
|
|
24
|
+
await client.auth.signIn();
|
|
25
|
+
} catch (err) {
|
|
26
|
+
console.error('signIn failed', err);
|
|
27
|
+
setLoading(false);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
return (
|
|
32
|
+
<button
|
|
33
|
+
type="button"
|
|
34
|
+
onClick={handleSignIn}
|
|
35
|
+
disabled={loading}
|
|
36
|
+
className="inline-flex items-center justify-center px-4 py-2 border bg-white hover:bg-zinc-50 text-zinc-900 font-medium"
|
|
37
|
+
>
|
|
38
|
+
{loading ? 'Redirecionando...' : 'Entrar com Neetru'}
|
|
39
|
+
</button>
|
|
40
|
+
);
|
|
41
|
+
}
|
|
@@ -1,22 +1,22 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Template — `src/lib/neetru/billing/checkout.ts` (gerado por `neetru add billing`).
|
|
3
|
-
*
|
|
4
|
-
* Inicia checkout Stripe via Core. Server route handler.
|
|
5
|
-
*/
|
|
6
|
-
import { NextRequest, NextResponse } from 'next/server';
|
|
7
|
-
|
|
8
|
-
export const runtime = 'nodejs';
|
|
9
|
-
|
|
10
|
-
export async function POST(request: NextRequest) {
|
|
11
|
-
const body = await request.json();
|
|
12
|
-
const slug = body?.slug as string | undefined;
|
|
13
|
-
if (!slug) {
|
|
14
|
-
return NextResponse.json({ error: 'slug required' }, { status: 400 });
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
// Hit Core REST: POST /api/v1/billing/checkout
|
|
18
|
-
// (Sprint 9 expõe via SDK direto)
|
|
19
|
-
return NextResponse.json({
|
|
20
|
-
redirectUrl: `https://minhaconta.neetru.com/checkout?slug=${slug}`,
|
|
21
|
-
});
|
|
22
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Template — `src/lib/neetru/billing/checkout.ts` (gerado por `neetru add billing`).
|
|
3
|
+
*
|
|
4
|
+
* Inicia checkout Stripe via Core. Server route handler.
|
|
5
|
+
*/
|
|
6
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
7
|
+
|
|
8
|
+
export const runtime = 'nodejs';
|
|
9
|
+
|
|
10
|
+
export async function POST(request: NextRequest) {
|
|
11
|
+
const body = await request.json();
|
|
12
|
+
const slug = body?.slug as string | undefined;
|
|
13
|
+
if (!slug) {
|
|
14
|
+
return NextResponse.json({ error: 'slug required' }, { status: 400 });
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
// Hit Core REST: POST /api/v1/billing/checkout
|
|
18
|
+
// (Sprint 9 expõe via SDK direto)
|
|
19
|
+
return NextResponse.json({
|
|
20
|
+
redirectUrl: `https://minhaconta.neetru.com/checkout?slug=${slug}`,
|
|
21
|
+
});
|
|
22
|
+
}
|
|
@@ -1,43 +1,43 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Template — `src/lib/neetru/billing/page.tsx` (gerado por `neetru add billing`).
|
|
3
|
-
*
|
|
4
|
-
* Página simples: lista planos + botão checkout via SDK.
|
|
5
|
-
*/
|
|
6
|
-
'use client';
|
|
7
|
-
|
|
8
|
-
import { useEffect, useState } from 'react';
|
|
9
|
-
import { createNeetruClient, type Product } from '@neetru/sdk';
|
|
10
|
-
|
|
11
|
-
const client = createNeetruClient({
|
|
12
|
-
apiKey: process.env.NEXT_PUBLIC_NEETRU_API_KEY,
|
|
13
|
-
});
|
|
14
|
-
|
|
15
|
-
export default function BillingPage() {
|
|
16
|
-
const [products, setProducts] = useState<Product[]>([]);
|
|
17
|
-
useEffect(() => {
|
|
18
|
-
client.catalog
|
|
19
|
-
.list()
|
|
20
|
-
.then(({ products }) => setProducts(products))
|
|
21
|
-
.catch((err) => console.error(err));
|
|
22
|
-
}, []);
|
|
23
|
-
|
|
24
|
-
return (
|
|
25
|
-
<main className="p-8">
|
|
26
|
-
<h1 className="text-3xl font-bold mb-6">Planos</h1>
|
|
27
|
-
<div className="grid gap-4 md:grid-cols-3">
|
|
28
|
-
{products.map((p) => (
|
|
29
|
-
<article key={p.slug} className="border p-6">
|
|
30
|
-
<h2 className="text-xl font-semibold">{p.name}</h2>
|
|
31
|
-
<p className="text-sm text-zinc-600 mt-1">{p.tagline}</p>
|
|
32
|
-
<a
|
|
33
|
-
href={`/checkout?slug=${p.slug}`}
|
|
34
|
-
className="inline-block mt-4 px-4 py-2 bg-zinc-900 text-white"
|
|
35
|
-
>
|
|
36
|
-
Assinar
|
|
37
|
-
</a>
|
|
38
|
-
</article>
|
|
39
|
-
))}
|
|
40
|
-
</div>
|
|
41
|
-
</main>
|
|
42
|
-
);
|
|
43
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Template — `src/lib/neetru/billing/page.tsx` (gerado por `neetru add billing`).
|
|
3
|
+
*
|
|
4
|
+
* Página simples: lista planos + botão checkout via SDK.
|
|
5
|
+
*/
|
|
6
|
+
'use client';
|
|
7
|
+
|
|
8
|
+
import { useEffect, useState } from 'react';
|
|
9
|
+
import { createNeetruClient, type Product } from '@neetru/sdk';
|
|
10
|
+
|
|
11
|
+
const client = createNeetruClient({
|
|
12
|
+
apiKey: process.env.NEXT_PUBLIC_NEETRU_API_KEY,
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
export default function BillingPage() {
|
|
16
|
+
const [products, setProducts] = useState<Product[]>([]);
|
|
17
|
+
useEffect(() => {
|
|
18
|
+
client.catalog
|
|
19
|
+
.list()
|
|
20
|
+
.then(({ products }) => setProducts(products))
|
|
21
|
+
.catch((err) => console.error(err));
|
|
22
|
+
}, []);
|
|
23
|
+
|
|
24
|
+
return (
|
|
25
|
+
<main className="p-8">
|
|
26
|
+
<h1 className="text-3xl font-bold mb-6">Planos</h1>
|
|
27
|
+
<div className="grid gap-4 md:grid-cols-3">
|
|
28
|
+
{products.map((p) => (
|
|
29
|
+
<article key={p.slug} className="border p-6">
|
|
30
|
+
<h2 className="text-xl font-semibold">{p.name}</h2>
|
|
31
|
+
<p className="text-sm text-zinc-600 mt-1">{p.tagline}</p>
|
|
32
|
+
<a
|
|
33
|
+
href={`/checkout?slug=${p.slug}`}
|
|
34
|
+
className="inline-block mt-4 px-4 py-2 bg-zinc-900 text-white"
|
|
35
|
+
>
|
|
36
|
+
Assinar
|
|
37
|
+
</a>
|
|
38
|
+
</article>
|
|
39
|
+
))}
|
|
40
|
+
</div>
|
|
41
|
+
</main>
|
|
42
|
+
);
|
|
43
|
+
}
|
|
@@ -1,68 +1,68 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Template — `src/lib/neetru/support/ticket-form.tsx` (gerado por `neetru add support`).
|
|
3
|
-
*
|
|
4
|
-
* Form de suporte: cria ticket via SDK.
|
|
5
|
-
*/
|
|
6
|
-
'use client';
|
|
7
|
-
|
|
8
|
-
import { useState } from 'react';
|
|
9
|
-
import { createNeetruClient } from '@neetru/sdk';
|
|
10
|
-
|
|
11
|
-
const client = createNeetruClient({
|
|
12
|
-
apiKey: process.env.NEXT_PUBLIC_NEETRU_API_KEY,
|
|
13
|
-
});
|
|
14
|
-
|
|
15
|
-
export function SupportTicketForm() {
|
|
16
|
-
const [subject, setSubject] = useState('');
|
|
17
|
-
const [message, setMessage] = useState('');
|
|
18
|
-
const [submitting, setSubmitting] = useState(false);
|
|
19
|
-
const [result, setResult] = useState<string | null>(null);
|
|
20
|
-
|
|
21
|
-
async function handleSubmit(e: React.FormEvent) {
|
|
22
|
-
e.preventDefault();
|
|
23
|
-
setSubmitting(true);
|
|
24
|
-
try {
|
|
25
|
-
const ticket = await client.support.createTicket({ subject, message });
|
|
26
|
-
setResult(`Ticket criado: ${ticket.id}`);
|
|
27
|
-
setSubject('');
|
|
28
|
-
setMessage('');
|
|
29
|
-
} catch (err) {
|
|
30
|
-
setResult('Falha ao criar ticket.');
|
|
31
|
-
console.error(err);
|
|
32
|
-
} finally {
|
|
33
|
-
setSubmitting(false);
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
return (
|
|
38
|
-
<form onSubmit={handleSubmit} className="grid gap-3 max-w-md">
|
|
39
|
-
<label className="grid gap-1">
|
|
40
|
-
<span className="text-sm font-medium">Assunto</span>
|
|
41
|
-
<input
|
|
42
|
-
required
|
|
43
|
-
value={subject}
|
|
44
|
-
onChange={(e) => setSubject(e.target.value)}
|
|
45
|
-
className="border px-3 py-2"
|
|
46
|
-
/>
|
|
47
|
-
</label>
|
|
48
|
-
<label className="grid gap-1">
|
|
49
|
-
<span className="text-sm font-medium">Mensagem</span>
|
|
50
|
-
<textarea
|
|
51
|
-
required
|
|
52
|
-
rows={5}
|
|
53
|
-
value={message}
|
|
54
|
-
onChange={(e) => setMessage(e.target.value)}
|
|
55
|
-
className="border px-3 py-2"
|
|
56
|
-
/>
|
|
57
|
-
</label>
|
|
58
|
-
<button
|
|
59
|
-
type="submit"
|
|
60
|
-
disabled={submitting}
|
|
61
|
-
className="px-4 py-2 bg-zinc-900 text-white"
|
|
62
|
-
>
|
|
63
|
-
{submitting ? 'Enviando...' : 'Enviar'}
|
|
64
|
-
</button>
|
|
65
|
-
{result && <p className="text-sm">{result}</p>}
|
|
66
|
-
</form>
|
|
67
|
-
);
|
|
68
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Template — `src/lib/neetru/support/ticket-form.tsx` (gerado por `neetru add support`).
|
|
3
|
+
*
|
|
4
|
+
* Form de suporte: cria ticket via SDK.
|
|
5
|
+
*/
|
|
6
|
+
'use client';
|
|
7
|
+
|
|
8
|
+
import { useState } from 'react';
|
|
9
|
+
import { createNeetruClient } from '@neetru/sdk';
|
|
10
|
+
|
|
11
|
+
const client = createNeetruClient({
|
|
12
|
+
apiKey: process.env.NEXT_PUBLIC_NEETRU_API_KEY,
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
export function SupportTicketForm() {
|
|
16
|
+
const [subject, setSubject] = useState('');
|
|
17
|
+
const [message, setMessage] = useState('');
|
|
18
|
+
const [submitting, setSubmitting] = useState(false);
|
|
19
|
+
const [result, setResult] = useState<string | null>(null);
|
|
20
|
+
|
|
21
|
+
async function handleSubmit(e: React.FormEvent) {
|
|
22
|
+
e.preventDefault();
|
|
23
|
+
setSubmitting(true);
|
|
24
|
+
try {
|
|
25
|
+
const ticket = await client.support.createTicket({ subject, message });
|
|
26
|
+
setResult(`Ticket criado: ${ticket.id}`);
|
|
27
|
+
setSubject('');
|
|
28
|
+
setMessage('');
|
|
29
|
+
} catch (err) {
|
|
30
|
+
setResult('Falha ao criar ticket.');
|
|
31
|
+
console.error(err);
|
|
32
|
+
} finally {
|
|
33
|
+
setSubmitting(false);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
return (
|
|
38
|
+
<form onSubmit={handleSubmit} className="grid gap-3 max-w-md">
|
|
39
|
+
<label className="grid gap-1">
|
|
40
|
+
<span className="text-sm font-medium">Assunto</span>
|
|
41
|
+
<input
|
|
42
|
+
required
|
|
43
|
+
value={subject}
|
|
44
|
+
onChange={(e) => setSubject(e.target.value)}
|
|
45
|
+
className="border px-3 py-2"
|
|
46
|
+
/>
|
|
47
|
+
</label>
|
|
48
|
+
<label className="grid gap-1">
|
|
49
|
+
<span className="text-sm font-medium">Mensagem</span>
|
|
50
|
+
<textarea
|
|
51
|
+
required
|
|
52
|
+
rows={5}
|
|
53
|
+
value={message}
|
|
54
|
+
onChange={(e) => setMessage(e.target.value)}
|
|
55
|
+
className="border px-3 py-2"
|
|
56
|
+
/>
|
|
57
|
+
</label>
|
|
58
|
+
<button
|
|
59
|
+
type="submit"
|
|
60
|
+
disabled={submitting}
|
|
61
|
+
className="px-4 py-2 bg-zinc-900 text-white"
|
|
62
|
+
>
|
|
63
|
+
{submitting ? 'Enviando...' : 'Enviar'}
|
|
64
|
+
</button>
|
|
65
|
+
{result && <p className="text-sm">{result}</p>}
|
|
66
|
+
</form>
|
|
67
|
+
);
|
|
68
|
+
}
|
package/templates/usage/track.ts
CHANGED
|
@@ -1,30 +1,30 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Template — `src/lib/neetru/usage/track.ts` (gerado por `neetru add usage`).
|
|
3
|
-
*
|
|
4
|
-
* Helper para reportar uso de resources metered ao Core.
|
|
5
|
-
*
|
|
6
|
-
* @example
|
|
7
|
-
* ```ts
|
|
8
|
-
* import { reportApiCall } from '@/lib/neetru/usage/track';
|
|
9
|
-
* await reportApiCall();
|
|
10
|
-
* ```
|
|
11
|
-
*/
|
|
12
|
-
import { createNeetruClient } from '@neetru/sdk';
|
|
13
|
-
|
|
14
|
-
const client = createNeetruClient({
|
|
15
|
-
apiKey: process.env.NEETRU_API_KEY,
|
|
16
|
-
productId: process.env.NEETRU_PRODUCT_ID,
|
|
17
|
-
tenantId: process.env.NEETRU_TENANT_ID,
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
export async function reportApiCall(qty = 1): Promise<void> {
|
|
21
|
-
try {
|
|
22
|
-
await client.usage.report('api_calls', qty);
|
|
23
|
-
} catch (err) {
|
|
24
|
-
console.warn('usage.report failed', err);
|
|
25
|
-
}
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
export async function checkApiCallLimit() {
|
|
29
|
-
return client.usage.check('api_calls');
|
|
30
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Template — `src/lib/neetru/usage/track.ts` (gerado por `neetru add usage`).
|
|
3
|
+
*
|
|
4
|
+
* Helper para reportar uso de resources metered ao Core.
|
|
5
|
+
*
|
|
6
|
+
* @example
|
|
7
|
+
* ```ts
|
|
8
|
+
* import { reportApiCall } from '@/lib/neetru/usage/track';
|
|
9
|
+
* await reportApiCall();
|
|
10
|
+
* ```
|
|
11
|
+
*/
|
|
12
|
+
import { createNeetruClient } from '@neetru/sdk';
|
|
13
|
+
|
|
14
|
+
const client = createNeetruClient({
|
|
15
|
+
apiKey: process.env.NEETRU_API_KEY,
|
|
16
|
+
productId: process.env.NEETRU_PRODUCT_ID,
|
|
17
|
+
tenantId: process.env.NEETRU_TENANT_ID,
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
export async function reportApiCall(qty = 1): Promise<void> {
|
|
21
|
+
try {
|
|
22
|
+
await client.usage.report('api_calls', qty);
|
|
23
|
+
} catch (err) {
|
|
24
|
+
console.warn('usage.report failed', err);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export async function checkApiCallLimit() {
|
|
29
|
+
return client.usage.check('api_calls');
|
|
30
|
+
}
|