tablas y delete module
This commit is contained in:
@@ -38,19 +38,23 @@ export function registerCheckModuleUsageTool(server) {
|
|||||||
const apiError = handleApiResponse(response.data, 'check_module_usage');
|
const apiError = handleApiResponse(response.data, 'check_module_usage');
|
||||||
if (apiError) return apiError;
|
if (apiError) return apiError;
|
||||||
|
|
||||||
// Extract usage information
|
// El PHP devuelve { result, success, message }. Si el modulo NO esta
|
||||||
const usageData = response.data.data || response.data;
|
// en uso, message = "No encuentro el módulo en ninguna sección".
|
||||||
|
// Si esta en uso, message contiene HTML con las tablas/paginas.
|
||||||
|
const msg = (response.data?.message || "");
|
||||||
|
const inUse = !!msg && !msg.includes("No encuentro");
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content: [{
|
content: [{
|
||||||
type: "text", text: JSON.stringify({
|
type: "text", text: JSON.stringify({
|
||||||
success: true,
|
success: true,
|
||||||
moduleId: id,
|
moduleId: id,
|
||||||
usage: usageData,
|
inUse,
|
||||||
canDelete: !usageData || Object.keys(usageData).length === 0,
|
canDelete: !inUse,
|
||||||
message: Object.keys(usageData || {}).length === 0
|
message: inUse
|
||||||
? "Module is not used anywhere - safe to delete"
|
? "Module is in use — deletion denied. Inform the user which pages use it and stop. Do NOT attempt to remove it from pages."
|
||||||
: `Module is used in ${Object.keys(usageData || {}).length} location(s)`
|
: "Module is not used anywhere — safe to delete",
|
||||||
|
rawMessage: msg,
|
||||||
}, null, 2)
|
}, null, 2)
|
||||||
}],
|
}],
|
||||||
};
|
};
|
||||||
|
|||||||
53
mcp-server/tools/modules/delete.js
Normal file
53
mcp-server/tools/modules/delete.js
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { z } from "zod";
|
||||||
|
import { withAuth, getSessionCredentials, getApiClient } from "../../auth/index.js";
|
||||||
|
import { handleToolError, validateRequired, handleApiResponse } from "../helpers/errorHandler.js";
|
||||||
|
import { withAuthParams } from "../helpers/authSchema.js";
|
||||||
|
import { AcaiHttpClient } from "../helpers/acaiHttpClient.js";
|
||||||
|
|
||||||
|
export function registerDeleteModuleTool(server) {
|
||||||
|
server.tool(
|
||||||
|
"delete_module",
|
||||||
|
"Elimina un módulo del proyecto. Borra la carpeta completa del módulo (template/estandar/modulos/{moduleId}/). OBLIGATORIO: llama a check_module_usage ANTES. Si el módulo está en uso (inUse=true), DENIEGA el borrado e informa al usuario de las páginas donde se usa. NO intentes quitar el módulo de las páginas por tu cuenta — solo el usuario puede decidir eso.",
|
||||||
|
withAuthParams({
|
||||||
|
moduleId: z.string().describe("ID del módulo a eliminar (nombre de la carpeta)"),
|
||||||
|
}),
|
||||||
|
{ readOnlyHint: false, destructiveHint: true },
|
||||||
|
withAuth(async ({ moduleId }, extra) => {
|
||||||
|
try {
|
||||||
|
const validationError = validateRequired({ moduleId }, ['moduleId'], 'delete_module');
|
||||||
|
if (validationError) return validationError;
|
||||||
|
|
||||||
|
const credentials = await getSessionCredentials(extra.sessionId);
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
action_ws: "deleteModule",
|
||||||
|
fileName: moduleId,
|
||||||
|
token: credentials.token,
|
||||||
|
tokenHash: credentials.tokenHash
|
||||||
|
};
|
||||||
|
|
||||||
|
const response = await AcaiHttpClient.postViewerFunctions(
|
||||||
|
await getApiClient(extra.sessionId),
|
||||||
|
payload
|
||||||
|
);
|
||||||
|
|
||||||
|
// Check for API errors (ej: módulo en uso)
|
||||||
|
const apiError = handleApiResponse(response.data, 'delete_module');
|
||||||
|
if (apiError) return apiError;
|
||||||
|
|
||||||
|
return {
|
||||||
|
content: [{
|
||||||
|
type: "text",
|
||||||
|
text: JSON.stringify({
|
||||||
|
success: true,
|
||||||
|
moduleId,
|
||||||
|
message: `Módulo "${moduleId}" eliminado correctamente.`
|
||||||
|
}, null, 2)
|
||||||
|
}],
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
return handleToolError(error, 'delete_module', { moduleId });
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import { registerCheckModuleTool } from './check.js';
|
import { registerCheckModuleTool } from './check.js';
|
||||||
import { registerCheckModuleUsageTool } from './checkUsage.js';
|
import { registerCheckModuleUsageTool } from './checkUsage.js';
|
||||||
import { registerCompileModuleTool } from './compile.js';
|
import { registerCompileModuleTool } from './compile.js';
|
||||||
|
import { registerDeleteModuleTool } from './delete.js';
|
||||||
import { canEditCode } from '../helpers/roleCheck.js';
|
import { canEditCode } from '../helpers/roleCheck.js';
|
||||||
|
|
||||||
export function registerModuleTools(server) {
|
export function registerModuleTools(server) {
|
||||||
@@ -8,5 +9,6 @@ export function registerModuleTools(server) {
|
|||||||
registerCheckModuleUsageTool(server);
|
registerCheckModuleUsageTool(server);
|
||||||
if (canEditCode()) {
|
if (canEditCode()) {
|
||||||
registerCompileModuleTool(server);
|
registerCompileModuleTool(server);
|
||||||
|
registerDeleteModuleTool(server);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,7 @@
|
|||||||
// TODO: adaptar create, delete, fields, list, schema para Docker local
|
import { registerListTablesTool } from './list.js';
|
||||||
// import { registerListTablesTool } from './list.js';
|
import { registerGetTableSchemaTool } from './schema.js';
|
||||||
// import { registerGetTableSchemaTool, registerUpdateTableSchemaTool } from './schema.js';
|
|
||||||
// import { registerGetTableTemplatesTool } from './templates.js';
|
|
||||||
// import { registerCreateTableTool } from './create.js';
|
|
||||||
// import { registerDeleteTableTool } from './delete.js';
|
|
||||||
// import { registerEditTableFieldTool, registerDeleteTableFieldTool } from './fields.js';
|
|
||||||
|
|
||||||
export function registerTableTools(server) {
|
export function registerTableTools(server) {
|
||||||
// registerListTablesTool(server);
|
registerListTablesTool(server);
|
||||||
// registerGetTableSchemaTool(server);
|
registerGetTableSchemaTool(server);
|
||||||
// registerUpdateTableSchemaTool(server);
|
|
||||||
// registerGetTableTemplatesTool(server);
|
|
||||||
// registerCreateTableTool(server);
|
|
||||||
// registerDeleteTableTool(server);
|
|
||||||
// registerEditTableFieldTool(server);
|
|
||||||
// registerDeleteTableFieldTool(server);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
65
mcp-server/tools/tables/iniParser.js
Normal file
65
mcp-server/tools/tables/iniParser.js
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
const BOOL_FIELDS = new Set([
|
||||||
|
'isSystemField', 'isRequired', 'isAdmin', 'adminOnly', 'hidden',
|
||||||
|
'_detailPage', '_disableAdd', '_disableDelete'
|
||||||
|
]);
|
||||||
|
const INT_FIELDS = new Set(['order', 'menuOrder', 'maxRecords', 'maxUploads']);
|
||||||
|
|
||||||
|
function stripPhpLine(ini) {
|
||||||
|
return ini.replace(/^<\?php.*?\?>\s*\n?/, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function coerceValue(key, value) {
|
||||||
|
if (BOOL_FIELDS.has(key)) return value === '1';
|
||||||
|
if (INT_FIELDS.has(key)) { const n = parseInt(value, 10); return isNaN(n) ? 0 : n; }
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseIniMeta(iniContent) {
|
||||||
|
const clean = stripPhpLine(iniContent);
|
||||||
|
const meta = {};
|
||||||
|
for (const line of clean.split('\n')) {
|
||||||
|
const trimmed = line.trim();
|
||||||
|
if (!trimmed || trimmed.startsWith(';') || trimmed.startsWith('#')) continue;
|
||||||
|
if (trimmed.startsWith('[')) break;
|
||||||
|
const m = trimmed.match(/^([^\s=]+)\s*=\s*(.*)$/);
|
||||||
|
if (m) {
|
||||||
|
const key = m[1];
|
||||||
|
const value = m[2].trim().replace(/^"|"$/g, '');
|
||||||
|
meta[key] = coerceValue(key, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return meta;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function parseIniSchema(iniContent) {
|
||||||
|
const clean = stripPhpLine(iniContent);
|
||||||
|
const result = {};
|
||||||
|
const fields = {};
|
||||||
|
let currentSection = null;
|
||||||
|
|
||||||
|
for (const line of clean.split('\n')) {
|
||||||
|
const trimmed = line.trim();
|
||||||
|
if (!trimmed || trimmed.startsWith(';') || trimmed.startsWith('#')) continue;
|
||||||
|
|
||||||
|
const sectionMatch = trimmed.match(/^\[(.+)\]$/);
|
||||||
|
if (sectionMatch) {
|
||||||
|
currentSection = sectionMatch[1];
|
||||||
|
fields[currentSection] = {};
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const kvMatch = trimmed.match(/^([^\s=]+)\s*=\s*(.*)$/);
|
||||||
|
if (kvMatch) {
|
||||||
|
const key = kvMatch[1];
|
||||||
|
const value = kvMatch[2].trim().replace(/^"|"$/g, '');
|
||||||
|
if (currentSection) {
|
||||||
|
fields[currentSection][key] = coerceValue(key, value);
|
||||||
|
} else {
|
||||||
|
result[key] = coerceValue(key, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result.fields = fields;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
@@ -1,65 +1,43 @@
|
|||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { withAuth, getSessionCredentials } from "../../auth/index.js";
|
import { withAuth, getSessionCredentials, getApiClient } from "../../auth/index.js";
|
||||||
import { handleToolError, handleApiResponse } from "../helpers/errorHandler.js";
|
import { handleToolError, handleApiResponse } from "../helpers/errorHandler.js";
|
||||||
import { AcaiHttpClient } from "../helpers/acaiHttpClient.js";
|
import { AcaiHttpClient } from "../helpers/acaiHttpClient.js";
|
||||||
import { withAuthParams } from "../helpers/authSchema.js";
|
import { withAuthParams } from "../helpers/authSchema.js";
|
||||||
|
import { parseIniMeta } from "./iniParser.js";
|
||||||
|
|
||||||
export function registerListTablesTool(server) {
|
export function registerListTablesTool(server) {
|
||||||
server.tool(
|
server.tool(
|
||||||
"list_tables",
|
"list_tables",
|
||||||
"List all database tables/schemas and General Sections (tables with 'enlace' field) in the system. Table names returned here are WITHOUT the 'cms_' prefix — use them as-is in all other tool calls. The primary key for all tables is 'num', never 'id'.",
|
"List all database tables in the project. Table names are WITHOUT 'cms_' prefix — use them as-is in all other tool calls. Primary key is 'num', never 'id'.",
|
||||||
withAuthParams({
|
withAuthParams({}),
|
||||||
withoutEnlace: z.boolean().default(true).describe("If true, include all tables, not only the ones that are general sections with 'enlace' field"),
|
|
||||||
}),
|
|
||||||
{ readOnlyHint: true, destructiveHint: false },
|
{ readOnlyHint: true, destructiveHint: false },
|
||||||
withAuth(async ({ withoutEnlace }, extra) => {
|
withAuth(async (params, extra) => {
|
||||||
try {
|
try {
|
||||||
console.error(`[list_tables] Tool called with sessionId: ${extra.sessionId}`);
|
const credentials = await getSessionCredentials(extra.sessionId);
|
||||||
console.error(`[list_tables] Getting credentials for session...`);
|
const payload = {
|
||||||
|
action_ws: "getTableSchemas",
|
||||||
const creds = await getSessionCredentials(extra.sessionId);
|
token: credentials.token,
|
||||||
console.error(`[list_tables] Credentials: website=${creds.website}, hasToken=${!!creds.token}, profileName=${creds.profileName}`);
|
tokenHash: credentials.tokenHash,
|
||||||
|
|
||||||
if (!creds.token) {
|
|
||||||
console.error(`[list_tables] ERROR: No token found for session ${extra.sessionId}!`);
|
|
||||||
return {
|
|
||||||
content: [{
|
|
||||||
type: "text",
|
|
||||||
text: JSON.stringify({
|
|
||||||
error: "No authentication token found for this session. Please login first using login_client tool.",
|
|
||||||
sessionId: extra.sessionId,
|
|
||||||
profileName: creds.profileName
|
|
||||||
}, null, 2)
|
|
||||||
}],
|
|
||||||
isError: true
|
|
||||||
};
|
};
|
||||||
}
|
const response = await AcaiHttpClient.postViewerFunctions(
|
||||||
|
await getApiClient(extra.sessionId),
|
||||||
const response = await AcaiHttpClient.saasPostRequest(
|
payload
|
||||||
{
|
|
||||||
action: 'getSchemaTables',
|
|
||||||
type: 'menu'
|
|
||||||
},
|
|
||||||
creds.token
|
|
||||||
);
|
);
|
||||||
|
const apiError = handleApiResponse(response.data, 'list_tables');
|
||||||
|
if (apiError) return apiError;
|
||||||
|
|
||||||
if (!response.data.success) {
|
const schemas = response.data.schemas || {};
|
||||||
|
const tables = Object.entries(schemas).map(([filename, iniContent]) => {
|
||||||
|
const tableName = filename.replace('.ini.php', '');
|
||||||
|
const meta = parseIniMeta(iniContent);
|
||||||
return {
|
return {
|
||||||
content: [{ type: "text", text: "Error getting tables: " + JSON.stringify(response.data) }],
|
tableName,
|
||||||
isError: true,
|
menuName: meta.menuName || tableName,
|
||||||
|
menuType: meta.menuType || 'multi',
|
||||||
|
enlace: meta.enlace || null,
|
||||||
|
hasBuilder: iniContent.includes('[builder]'),
|
||||||
};
|
};
|
||||||
}
|
});
|
||||||
|
|
||||||
// Filter tables based on withoutEnlace parameter
|
|
||||||
const tables = response.data.data.filter(schema =>
|
|
||||||
withoutEnlace ? true : !!schema.enlace
|
|
||||||
).map(table => ({
|
|
||||||
name: table.menuName,
|
|
||||||
tableName: table.tableName,
|
|
||||||
order: table.menuOrder,
|
|
||||||
enlace: table.enlace,
|
|
||||||
hasBuilder: !!table.builder
|
|
||||||
}));
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content: [{ type: "text", text: JSON.stringify(tables, null, 2) }],
|
content: [{ type: "text", text: JSON.stringify(tables, null, 2) }],
|
||||||
@@ -70,5 +48,3 @@ export function registerListTablesTool(server) {
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,184 +1,65 @@
|
|||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { withAuth, getApiClient, getSessionCredentials, getCommonParams } from "../../auth/index.js";
|
import { withAuth, getSessionCredentials, getApiClient } from "../../auth/index.js";
|
||||||
import { normalizeSchemaForSave, mergeTableSchemas } from "../../utils/fieldHelpers.js";
|
|
||||||
import { handleToolError, validateRequired, handleApiResponse } from "../helpers/errorHandler.js";
|
import { handleToolError, validateRequired, handleApiResponse } from "../helpers/errorHandler.js";
|
||||||
import { AcaiHttpClient } from "../helpers/acaiHttpClient.js";
|
import { AcaiHttpClient } from "../helpers/acaiHttpClient.js";
|
||||||
import { withAuthParams } from "../helpers/authSchema.js";
|
import { withAuthParams } from "../helpers/authSchema.js";
|
||||||
|
import { parseIniSchema } from "./iniParser.js";
|
||||||
|
|
||||||
export function registerGetTableSchemaTool(server) {
|
export function registerGetTableSchemaTool(server) {
|
||||||
server.tool(
|
server.tool(
|
||||||
"get_table_schema",
|
"get_table_schema",
|
||||||
"Get the schema of a database table. Tables WITHOUT 'cms_' prefix. Primary key is 'num', NEVER 'id'. Use minimal=true for just field names + types (saves tokens).",
|
"Get the full schema of a database table, including all fields and their types/metadata. Table names WITHOUT 'cms_' prefix. Primary key is 'num', never 'id'. Use minimal=true for compact output (saves tokens).",
|
||||||
withAuthParams({
|
withAuthParams({
|
||||||
tableName: z.string().describe("Name of the table to get schema for (without 'cms_' prefix)"),
|
tableName: z.string().describe("Table name without cms_ prefix"),
|
||||||
minimal: z.boolean().optional().describe("If true, returns only field names and types (compact). Default: false (full schema with all metadata)."),
|
minimal: z.boolean().optional().describe("If true, returns only field names + types + labels"),
|
||||||
}),
|
}),
|
||||||
{ readOnlyHint: true, destructiveHint: false },
|
{ readOnlyHint: true, destructiveHint: false },
|
||||||
withAuth(async ({ tableName, minimal }, extra) => {
|
withAuth(async ({ tableName, minimal }, extra) => {
|
||||||
try {
|
try {
|
||||||
// Validate required parameters
|
|
||||||
const validationError = validateRequired({ tableName }, ['tableName'], 'get_table_schema');
|
const validationError = validateRequired({ tableName }, ['tableName'], 'get_table_schema');
|
||||||
if (validationError) return validationError;
|
if (validationError) return validationError;
|
||||||
|
|
||||||
const credentials = await getSessionCredentials(extra.sessionId);
|
const credentials = await getSessionCredentials(extra.sessionId);
|
||||||
const response = await AcaiHttpClient.saasPostRequest(
|
const payload = {
|
||||||
{
|
action_ws: "getTableSchemas",
|
||||||
id: tableName
|
tableName,
|
||||||
},
|
token: credentials.token,
|
||||||
credentials.token
|
tokenHash: credentials.tokenHash,
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.data.success) {
|
|
||||||
return {
|
|
||||||
content: [{ type: "text", text: "Error getting schema: " + JSON.stringify(response.data) }],
|
|
||||||
isError: true,
|
|
||||||
};
|
};
|
||||||
}
|
const response = await AcaiHttpClient.postViewerFunctions(
|
||||||
|
await getApiClient(extra.sessionId),
|
||||||
|
payload
|
||||||
|
);
|
||||||
|
const apiError = handleApiResponse(response.data, 'get_table_schema');
|
||||||
|
if (apiError) return apiError;
|
||||||
|
|
||||||
// Find the specific table
|
const schemas = response.data.schemas || {};
|
||||||
const table = response.data.data;
|
const filename = tableName + '.ini.php';
|
||||||
|
const iniContent = schemas[filename];
|
||||||
if (!table) {
|
if (!iniContent) {
|
||||||
return {
|
return {
|
||||||
content: [{ type: "text", text: `Table '${tableName}' not found` }],
|
content: [{ type: "text", text: `Table '${tableName}' not found` }],
|
||||||
isError: true,
|
isError: true,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// Minimal mode: return only field names, types, and key metadata
|
const parsed = parseIniSchema(iniContent);
|
||||||
|
|
||||||
if (minimal) {
|
if (minimal) {
|
||||||
const minimalSchema = {};
|
const minimalSchema = { menuName: parsed.menuName, menuType: parsed.menuType, enlace: parsed.enlace };
|
||||||
for (const [key, value] of Object.entries(table)) {
|
const minFields = {};
|
||||||
if (value && typeof value === 'object' && value.type) {
|
for (const [key, value] of Object.entries(parsed.fields || {})) {
|
||||||
const field = { type: value.type };
|
minFields[key] = { type: value.type };
|
||||||
if (value.label) field.label = value.label;
|
if (value.label) minFields[key].label = value.label;
|
||||||
if (value.optionsType) field.optionsType = value.optionsType;
|
|
||||||
if (value.optionsTablename) field.optionsTablename = value.optionsTablename;
|
|
||||||
if (value.isRequired) field.isRequired = value.isRequired;
|
|
||||||
minimalSchema[key] = field;
|
|
||||||
} else if (typeof value !== 'object') {
|
|
||||||
// Keep top-level scalar metadata (menuName, menuType, enlace, etc.)
|
|
||||||
minimalSchema[key] = value;
|
|
||||||
}
|
}
|
||||||
}
|
minimalSchema.fields = minFields;
|
||||||
return {
|
return { content: [{ type: "text", text: JSON.stringify(minimalSchema, null, 2) }] };
|
||||||
content: [{ type: "text", text: JSON.stringify(minimalSchema, null, 2) }],
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return { content: [{ type: "text", text: JSON.stringify(parsed, null, 2) }] };
|
||||||
content: [{ type: "text", text: JSON.stringify(table, null, 2) }],
|
|
||||||
};
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return handleToolError(error, 'get_table_schema', { tableName });
|
return handleToolError(error, 'get_table_schema', { tableName });
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function registerUpdateTableSchemaTool(server) {
|
|
||||||
server.tool(
|
|
||||||
"update_table_schema",
|
|
||||||
`Update table-level metadata (menuName, menuOrder, enlace, seo_metas). NOT for field operations — use edit_table_field instead.
|
|
||||||
|
|
||||||
Tables WITHOUT 'cms_' prefix. 2-step process: saves to SAAS server, then triggers website schema update.`,
|
|
||||||
withAuthParams({
|
|
||||||
tableName: z.string().describe("Name of the table to update"),
|
|
||||||
schema: z.object({}).passthrough().describe("Schema object with fields objects ( like reference schema table ) to add or update. By default, this is merged with the existing schema."),
|
|
||||||
overwrite: z.boolean().optional().describe("If true, replaces the ENTIRE schema with the provided one (deleting missing fields). If false (default), merges with existing schema."),
|
|
||||||
}),
|
|
||||||
{ readOnlyHint: false, destructiveHint: false },
|
|
||||||
withAuth(async ({ tableName, schema, overwrite = false }, extra) => {
|
|
||||||
try {
|
|
||||||
// Validate required parameters
|
|
||||||
const validationError = validateRequired({ tableName, schema }, ['tableName', 'schema'], 'update_table_schema');
|
|
||||||
if (validationError) return validationError;
|
|
||||||
|
|
||||||
let schemaToSave;
|
|
||||||
|
|
||||||
const credentials = await getSessionCredentials(extra.sessionId);
|
|
||||||
|
|
||||||
if (overwrite) {
|
|
||||||
// If overwrite is true, use the provided schema directly
|
|
||||||
schemaToSave = { ...schema };
|
|
||||||
} else {
|
|
||||||
// Step 1: Fetch current schema to preserve existing fields
|
|
||||||
const getResponse = await AcaiHttpClient.saasPostRequest(
|
|
||||||
{
|
|
||||||
id: tableName
|
|
||||||
},
|
|
||||||
credentials.token
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!getResponse.data.success) {
|
|
||||||
return {
|
|
||||||
content: [{ type: "text", text: "Error fetching current schema: " + JSON.stringify(getResponse.data) }],
|
|
||||||
isError: true,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const currentTable = getResponse.data.data;
|
|
||||||
|
|
||||||
if (!currentTable) {
|
|
||||||
return {
|
|
||||||
content: [{ type: "text", text: `Table '${tableName}' not found. Please create it first using create_table.` }],
|
|
||||||
isError: true,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Step 2: Merge new schema into existing schema
|
|
||||||
schemaToSave = mergeTableSchemas(currentTable, schema);
|
|
||||||
}
|
|
||||||
|
|
||||||
normalizeSchemaForSave(schemaToSave);
|
|
||||||
|
|
||||||
// Remove tableName from schema (as done in frontend)
|
|
||||||
delete schemaToSave.tableName;
|
|
||||||
|
|
||||||
// Step 3: Save merged schema to SAAS server (PUT request)
|
|
||||||
const saasResponse = await AcaiHttpClient.saasPutRequest(
|
|
||||||
{
|
|
||||||
action: "saveSchema",
|
|
||||||
schema: schemaToSave,
|
|
||||||
id: tableName,
|
|
||||||
},
|
|
||||||
credentials.token
|
|
||||||
);
|
|
||||||
|
|
||||||
// SAAS returns {success: true} not {result: true}
|
|
||||||
if (!saasResponse.data.success && !saasResponse.data.result) {
|
|
||||||
return {
|
|
||||||
content: [{ type: "text", text: "Error saving schema to SAAS: " + JSON.stringify(saasResponse.data) }],
|
|
||||||
isError: true,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Step 4: Trigger schema update on website
|
|
||||||
const client = await getApiClient(extra.sessionId);
|
|
||||||
const updateResponse = await client.post("/cms/lib/viewer_functions.php", await getCommonParams(extra.sessionId, {
|
|
||||||
action_ws: "updateAllSchemas",
|
|
||||||
tokenHash: credentials.tokenHash
|
|
||||||
}));
|
|
||||||
|
|
||||||
// Check for website update errors
|
|
||||||
let updateError = handleApiResponse(updateResponse.data, 'update_table_schema');
|
|
||||||
if (updateError) return updateError;
|
|
||||||
|
|
||||||
return {
|
|
||||||
content: [{
|
|
||||||
type: "text",
|
|
||||||
text: JSON.stringify({
|
|
||||||
success: true,
|
|
||||||
message: overwrite ? "Schema overwritten successfully" : "Schema updated successfully (merged with existing fields)",
|
|
||||||
mergedFields: Object.keys(schemaToSave),
|
|
||||||
saasResponse: saasResponse.data,
|
|
||||||
webResponse: updateResponse.data
|
|
||||||
}, null, 2)
|
|
||||||
}],
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
return handleToolError(error, 'update_table_schema', { tableName, overwrite });
|
|
||||||
}
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|||||||
Reference in New Issue
Block a user