@lark-apaas/openclaw-extension-miaoda-coding 1.0.11 → 1.0.12
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/dist/index.mjs +24 -8
- package/package.json +7 -5
- package/skills/miaoda-coding/SKILL.md +52 -10
package/dist/index.mjs
CHANGED
|
@@ -1,11 +1,27 @@
|
|
|
1
|
-
import{createRequire as e}from"node:module";import{execFile as t,spawn as n}from"node:child_process";import r,{readFileSync as i}from"node:fs";import a,{dirname as o,join as s,resolve as c}from"node:path";import{appendFile as l,mkdir as u,readFile as d,readdir as f,rm as p,writeFile as m}from"node:fs/promises";import{fileURLToPath as h}from"node:url";var g=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),_=e(import.meta.url);function v(){return c(process.env.OPENCLAW_STATE_DIR||process.cwd())}function y(e){return s(e,`.agent`)}function b(e,t){return s(e,t)}function x(e){let t=typeof e.workspaceDir==`string`&&e.workspaceDir!==``,n=typeof e.agentDir==`string`&&e.agentDir!==``;if(!t||!n)throw Error(`resolveToolPaths: workspaceDir and agentDir are required in tool factory context`);return{workspaceDir:e.workspaceDir,agentDir:e.agentDir,agentId:e.agentId,sessionKey:e.sessionKey}}var S=g(((e,t)=>{var n=Object.defineProperty,r=Object.getOwnPropertyDescriptor,i=Object.getOwnPropertyNames,a=Object.prototype.hasOwnProperty,o=(e,t)=>{for(var r in t)n(e,r,{get:t[r],enumerable:!0})},s=(e,t,o,s)=>{if(t&&typeof t==`object`||typeof t==`function`)for(let c of i(t))!a.call(e,c)&&c!==o&&n(e,c,{get:()=>t[c],enumerable:!(s=r(t,c))||s.enumerable});return e},c=e=>s(n({},`__esModule`,{value:!0}),e),l={};o(l,{DEFAULT_CLOCK_TOLERANCE_SEC:()=>f,DEFAULT_JWT_EXPIRE_TIME_MS:()=>d,FileTokenProvider:()=>w,HttpClient:()=>L,HttpError:()=>P,generateJWTToken:()=>m,parseJWTTokenWithVerify:()=>h,registerPlatformPlugin:()=>k,resolvePlatformBaseURL:()=>O}),t.exports=c(l);var u=_(`crypto`),d=1800*1e3,f=60,p={alg:`HS256`,typ:`JWT`};function m(e,t){let n=Math.floor(Date.now()/1e3),r={...e,iss:e.access_key,iat:n,nbf:n,exp:n+Math.floor(t.expireTimeMs/1e3),jti:(0,u.randomUUID)()},i=v(JSON.stringify(p)),a=v(JSON.stringify(r));return`${i}.${a}.${g(`${i}.${a}`,t.secretKey)}`}function h(e,t,n){let r=e.split(`.`);if(r.length!==3)throw Error(`invalid JWT token format`);let[i,a,o]=r;if(JSON.parse(y(i)).alg!==`HS256`)throw Error(`unsupported JWT alg`);let s=g(`${i}.${a}`,t.secretKey),c=Buffer.from(s),l=Buffer.from(o);if(c.length!==l.length||!(0,u.timingSafeEqual)(c,l))throw Error(`JWT signature verification failed`);let d=JSON.parse(y(a));if(!n?.skipExpiration){let e=n?.clockTolerance??f,t=Math.floor(Date.now()/1e3);if(d.exp!==void 0&&d.exp+e<t)throw Error(`JWT token expired at ${new Date(d.exp*1e3).toISOString()}`);if(d.nbf!==void 0&&d.nbf-e>t)throw Error(`JWT token not yet valid, will be valid at ${new Date(d.nbf*1e3).toISOString()}`);if(d.iat!==void 0&&d.iat-e>t)throw Error(`JWT token issued in the future at ${new Date(d.iat*1e3).toISOString()}`)}return d}function g(e,t){return(0,u.createHmac)(`sha256`,t).update(e).digest(`base64`).replace(/=/g,``).replace(/\+/g,`-`).replace(/\//g,`_`)}function v(e){return Buffer.from(e).toString(`base64`).replace(/=/g,``).replace(/\+/g,`-`).replace(/\//g,`_`)}function y(e){let t=(4-(e.length%4||4))%4,n=`${e}${`=`.repeat(t)}`.replace(/-/g,`+`).replace(/_/g,`/`);return Buffer.from(n,`base64`).toString(`utf8`)}var b=class{cache=new Map;config;constructor(e){this.config={refreshBeforeMs:300*1e3,...e}}getToken(e){let t=this.getCacheKey(e),n=this.cache.get(t),r=Date.now();if(n&&n.expiresAtMs-r>this.config.refreshBeforeMs)return n.token;let i=m(e,this.config),a=r+this.config.expireTimeMs;return this.cache.set(t,{token:i,expiresAtMs:a}),i}clearCache(){this.cache.clear()}clearCacheFor(e){let t=this.getCacheKey(e);this.cache.delete(t)}getCacheKey(e){return[e.access_key,e.tenant_id?.toString()||``,e.user_id||``,e.app_id||``,e.app_env||``,e.sandbox_id||``].join(`:`)}getCacheStats(){let e=Date.now(),t=0,n=0;for(let r of this.cache.values())r.expiresAtMs>e?t++:n++;return{total:this.cache.size,valid:t,expired:n}}},x=_(`fs`),S=300*1e3,C=[`/home/gem/workspace/.force/openclaw/miaoda-provider-key`],w=class{cache=new Map;paths;refreshBeforeMs;constructor(e){this.paths=e?.paths??C,this.refreshBeforeMs=e?.refreshBeforeMs??S}getToken(){let e=Date.now();for(let t of this.paths){let n=this.cache.get(t);if(n&&n.expiresAtMs-e>this.refreshBeforeMs)return{token:n.token,accessKey:n.accessKey};let r=this.readAndParse(t);if(r&&(this.cache.set(t,r),r.expiresAtMs>e))return{token:r.token,accessKey:r.accessKey};if(n&&n.expiresAtMs>e)return{token:n.token,accessKey:n.accessKey}}return null}clearCache(){this.cache.clear()}readAndParse(e){try{let t=(0,x.readFileSync)(e,`utf-8`).trim();if(!t)return null;let n=this.decodePayload(t);return n?.exp?{token:t,accessKey:n.access_key??``,expiresAtMs:n.exp*1e3}:null}catch{return null}}decodePayload(e){try{let t=e.split(`.`);if(t.length!==3)return null;let n=(4-(t[1].length%4||4))%4,r=`${t[1]}${`=`.repeat(n)}`.replace(/-/g,`+`).replace(/_/g,`/`);return JSON.parse(Buffer.from(r,`base64`).toString(`utf8`))}catch{return null}}},T=`FORCE_AUTHN_INNERAPI_DOMAIN`,E=`FORCE_AUTHN_ACCESS_KEY`,D=`FORCE_AUTHN_ACCESS_SECRET`;function O(e){if(!e?.enabled)return e?.baseURL;if(e.baseURL)return e.baseURL;let t=e.domainEnv||T,n=process.env[t];if(!n)throw Error(`\u5E73\u53F0\u6A21\u5F0F\u9700\u8981\u57FA\u7840\u57DF\u540D\uFF0C\u8BF7\u8BBE\u7F6E\u73AF\u5883\u53D8\u91CF ${t}`);return n}function k(e,t){if(!t.enabled)return;A(`platform.defaultClaims`,t.defaultClaims);let n;t.tokenProvider?.type===`file`&&(n=new w({paths:t.tokenProvider.paths,refreshBeforeMs:t.refreshBeforeMs}));let r=t.accessKeyEnv||E,i=t.secretKeyEnv||D,a=t.accessKey??(process.env[r]||``),o=new b({accessKey:a,secretKey:t.secretKey??(process.env[i]||``),expireTimeMs:t.expireTimeMs??d,refreshBeforeMs:t.refreshBeforeMs});return e.request.use(e=>{if(n){let t=n.getToken();if(t){let n={...e.headers,Authorization:`Bearer ${t.token}`,"x-api-key":t.accessKey};return{...e,headers:n}}}A(`request.platformAuth.customClaims`,e.platformAuth?.customClaims);let r={...t.defaultClaims||{},...e.platformAuth?.customClaims||{},access_key:a},i=o.getToken(r),s={...e.headers,Authorization:`Bearer ${i}`,"x-api-key":a};return{...e,headers:s}}),o}function A(e,t){if(t&&Object.prototype.hasOwnProperty.call(t,`access_key`))throw Error(`${e} \u4E0D\u5141\u8BB8\u8BBE\u7F6E access_key`)}var j=class{interceptors=[];use(e,t){return this.interceptors.push({onFulfilled:e,onRejected:t}),this.interceptors.length-1}eject(e){this.interceptors[e]&&(this.interceptors[e]=null)}clear(){this.interceptors=[]}forEach(e){this.interceptors.forEach(t=>{t!==null&&e(t)})}};function M(e){if(typeof e!=`object`||!e)return!1;let t=Object.getPrototypeOf(e);return t===Object.prototype||t===null}function N(e){if(!e)return{};if(e instanceof Headers){let t={};return e.forEach((e,n)=>{t[n]=e}),t}if(Array.isArray(e))return e.reduce((e,[t,n])=>(e[t]=n,e),{});let t={};return Object.entries(e).forEach(([e,n])=>{t[e]=Array.isArray(n)?n.join(`,`):n}),t}var P=class e extends Error{isHttpError=!0;response;config;constructor(t,n,r){super(r||`Request failed with status ${t?.status||`unknown`}`),this.name=`HttpError`,this.response=t,this.config=F(n),Object.setPrototypeOf(this,e.prototype)}};function F(e){let{headers:t,...n}=e;return{...n,headers:I(t)}}function I(e){if(!e)return;let t=N(e),n={},r=[`authorization`,`x-api-key`,`cookie`,`x-secret`];for(let[e,i]of Object.entries(t)){let t=e.toLowerCase();r.includes(t)?n[e]=`[REDACTED]`:n[e]=i}return n}var L=class{defaultConfig;securityConfig;interceptors={request:new j,response:new j};constructor(e){let{platform:t,security:n,...r}=e||{};this.defaultConfig={timeout:5e3,...r};let i=n?.strictMode??!1;this.securityConfig={allowedProtocols:i?[`http:`,`https:`]:null,maxResponseSize:i?50*1024*1024:0,strictMode:i,...n},t?.enabled&&(this.defaultConfig.baseURL||(this.defaultConfig.baseURL=O(t)),k(this.interceptors,t))}async get(e,t){return this.request({...t,url:e,method:`GET`})}async post(e,t,n){let r={...n?.headers},i=t;return M(t)&&(i=JSON.stringify(t),Object.keys(r).some(e=>e.toLowerCase()===`content-type`)||(r[`Content-Type`]=`application/json`)),this.request({...n,url:e,method:`POST`,body:i,headers:r})}async put(e,t,n){let r={...n?.headers},i=t;return M(t)&&(i=JSON.stringify(t),Object.keys(r).some(e=>e.toLowerCase()===`content-type`)||(r[`Content-Type`]=`application/json`)),this.request({...n,url:e,method:`PUT`,body:i,headers:r})}async delete(e,t){return this.request({...t,url:e,method:`DELETE`})}async patch(e,t,n){let r={...n?.headers},i=t;return M(t)&&(i=JSON.stringify(t),Object.keys(r).some(e=>e.toLowerCase()===`content-type`)||(r[`Content-Type`]=`application/json`)),this.request({...n,url:e,method:`PATCH`,body:i,headers:r})}async request(e){let t=N(this.defaultConfig.headers),n=N(e.headers),r={};for(let e in t)r[e.toLowerCase()]={key:e,value:t[e]};for(let e in n)r[e.toLowerCase()]={key:e,value:n[e]};let i={};for(let e in r){let{key:t,value:n}=r[e];i[t]=n}let a={...this.defaultConfig,...e,headers:i};try{a=await this.runRequestInterceptors(a)}catch(e){return Promise.reject(e)}a.headers=N(a.headers);let o=this.buildUrl(a.url,a.params),s=a.timeout||this.defaultConfig.timeout||5e3,c=new AbortController,l=setTimeout(()=>c.abort(),s),u=a.signal;u&&(u.aborted?(clearTimeout(l),c.abort()):u.addEventListener(`abort`,()=>c.abort(),{once:!0}));let{platformAuth:d,params:f,timeout:p,baseURL:m,url:h,...g}=a;try{let e=await fetch(o,{...g,signal:c.signal});if(clearTimeout(l),this.securityConfig.maxResponseSize>0){let t=e.headers.get(`content-length`);if(t){let e=parseInt(t,10);if(e>this.securityConfig.maxResponseSize)throw Error(`Response size ${e} bytes exceeds limit of ${this.securityConfig.maxResponseSize} bytes`)}}if(!e.ok){let t=new P(e,a);return this.runResponseInterceptors(Promise.reject(t))}return this.runResponseInterceptors(Promise.resolve(e))}catch(e){clearTimeout(l);let t=e instanceof P?e:new P(void 0,a,e.name===`AbortError`?`Request aborted`:e.message);return this.runResponseInterceptors(Promise.reject(t))}}runRequestInterceptors(e){let t=Promise.resolve(e),n=[];this.interceptors.request.forEach(e=>{n.push(e)});for(let e of n)t=t.then(e.onFulfilled,e.onRejected);return t}async runResponseInterceptors(e){let t=e,n=[];this.interceptors.response.forEach(e=>{n.push(e)});for(let e of n)t=t.then(e.onFulfilled,e.onRejected);return t}buildUrl(e,t){let n=this.defaultConfig.baseURL,r;r=!n||/^https?:\/\//i.test(e)?e:n.replace(/\/+$/,``)+`/`+e.replace(/^\/+/,``);let i=new URL(r),{allowedProtocols:a}=this.securityConfig;if(a&&!a.includes(i.protocol))throw Error(`Protocol ${i.protocol} is not allowed. Allowed: ${a.join(`, `)}`);if(t)for(let[e,n]of Object.entries(t))n!=null&&i.searchParams.set(e,String(n));return i.href}}}))();let C;function w(){if(C)return C;try{let e=c(o(h(import.meta.url)),`..`,`package.json`);C=JSON.parse(i(e,`utf-8`))}catch(e){console.warn(`[miaoda-coding] failed to read package.json: ${e instanceof Error?e.message:e}`),C={}}return C}function T(){let e=process.env.FORCE_AUTHN_INNERAPI_DOMAIN;if(!e)throw Error(`studio-client: 缺少环境变量,需要: FORCE_AUTHN_INNERAPI_DOMAIN`);return{apiUrl:e}}const E=new Set([`k_st_ec_400002683`]);var D=class extends Error{userFacing=!0;constructor(e,t,n){super(e),this.statusCode=t,this.logId=n,this.name=`UserFacingError`}};function O(e){return!!e&&typeof e==`object`&&!Array.isArray(e)&&Object.values(e).every(e=>typeof e==`string`)}function k(...e){for(let t of e)if(typeof t==`string`&&t)return t}async function*A(e){let t=new TextDecoder,n=``,r;for await(let i of e){n+=t.decode(i,{stream:!0});let e=n.split(`
|
|
1
|
+
import{createRequire as e}from"node:module";import{execFile as t,spawn as n}from"node:child_process";import{randomUUID as r}from"node:crypto";import i,{readFileSync as a}from"node:fs";import o,{dirname as s,join as c,resolve as l}from"node:path";import{appendFile as u,mkdir as d,readFile as f,readdir as p,rm as m,writeFile as h}from"node:fs/promises";import{fileURLToPath as g}from"node:url";var _=Object.create,v=Object.defineProperty,y=Object.getOwnPropertyDescriptor,b=Object.getOwnPropertyNames,x=Object.getPrototypeOf,S=Object.prototype.hasOwnProperty,C=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),w=(e,t,n,r)=>{if(t&&typeof t==`object`||typeof t==`function`)for(var i=b(t),a=0,o=i.length,s;a<o;a++)s=i[a],!S.call(e,s)&&s!==n&&v(e,s,{get:(e=>t[e]).bind(null,s),enumerable:!(r=y(t,s))||r.enumerable});return e},T=(e,t,n)=>(n=e==null?{}:_(x(e)),w(t||!e||!e.__esModule?v(n,`default`,{value:e,enumerable:!0}):n,e)),E=e(import.meta.url);function D(){return l(process.env.OPENCLAW_STATE_DIR||process.cwd())}function O(e){return c(e,`.agent`)}function k(e,t){return c(e,`app`,t)}function A(e,t){return c(e,t)}function j(e){let t=typeof e.workspaceDir==`string`&&e.workspaceDir!==``,n=typeof e.agentDir==`string`&&e.agentDir!==``;if(!t||!n)throw Error(`resolveToolPaths: workspaceDir and agentDir are required in tool factory context`);return{workspaceDir:e.workspaceDir,agentDir:e.agentDir,agentId:e.agentId,sessionKey:e.sessionKey}}var M=C(((e,t)=>{var n=Object.defineProperty,r=Object.getOwnPropertyDescriptor,i=Object.getOwnPropertyNames,a=Object.prototype.hasOwnProperty,o=(e,t)=>{for(var r in t)n(e,r,{get:t[r],enumerable:!0})},s=(e,t,o,s)=>{if(t&&typeof t==`object`||typeof t==`function`)for(let c of i(t))!a.call(e,c)&&c!==o&&n(e,c,{get:()=>t[c],enumerable:!(s=r(t,c))||s.enumerable});return e},c=e=>s(n({},`__esModule`,{value:!0}),e),l={};o(l,{DEFAULT_CLOCK_TOLERANCE_SEC:()=>f,DEFAULT_JWT_EXPIRE_TIME_MS:()=>d,FileTokenProvider:()=>C,HttpClient:()=>F,HttpError:()=>P,generateJWTToken:()=>m,parseJWTTokenWithVerify:()=>h,registerPlatformPlugin:()=>k,resolvePlatformBaseURL:()=>O}),t.exports=c(l);var u=E(`crypto`),d=1800*1e3,f=60,p={alg:`HS256`,typ:`JWT`};function m(e,t){let n=Math.floor(Date.now()/1e3),r={...e,iss:e.access_key,iat:n,nbf:n,exp:n+Math.floor(t.expireTimeMs/1e3),jti:(0,u.randomUUID)()},i=_(JSON.stringify(p)),a=_(JSON.stringify(r));return`${i}.${a}.${g(`${i}.${a}`,t.secretKey)}`}function h(e,t,n){let r=e.split(`.`);if(r.length!==3)throw Error(`invalid JWT token format`);let[i,a,o]=r;if(JSON.parse(v(i)).alg!==`HS256`)throw Error(`unsupported JWT alg`);let s=g(`${i}.${a}`,t.secretKey),c=Buffer.from(s),l=Buffer.from(o);if(c.length!==l.length||!(0,u.timingSafeEqual)(c,l))throw Error(`JWT signature verification failed`);let d=JSON.parse(v(a));if(!n?.skipExpiration){let e=n?.clockTolerance??f,t=Math.floor(Date.now()/1e3);if(d.exp!==void 0&&d.exp+e<t)throw Error(`JWT token expired at ${new Date(d.exp*1e3).toISOString()}`);if(d.nbf!==void 0&&d.nbf-e>t)throw Error(`JWT token not yet valid, will be valid at ${new Date(d.nbf*1e3).toISOString()}`);if(d.iat!==void 0&&d.iat-e>t)throw Error(`JWT token issued in the future at ${new Date(d.iat*1e3).toISOString()}`)}return d}function g(e,t){return(0,u.createHmac)(`sha256`,t).update(e).digest(`base64`).replace(/=/g,``).replace(/\+/g,`-`).replace(/\//g,`_`)}function _(e){return Buffer.from(e).toString(`base64`).replace(/=/g,``).replace(/\+/g,`-`).replace(/\//g,`_`)}function v(e){let t=(4-(e.length%4||4))%4,n=`${e}${`=`.repeat(t)}`.replace(/-/g,`+`).replace(/_/g,`/`);return Buffer.from(n,`base64`).toString(`utf8`)}var y=class{cache=new Map;config;constructor(e){this.config={refreshBeforeMs:300*1e3,...e}}getToken(e){let t=this.getCacheKey(e),n=this.cache.get(t),r=Date.now();if(n&&n.expiresAtMs-r>this.config.refreshBeforeMs)return n.token;let i=m(e,this.config),a=r+this.config.expireTimeMs;return this.cache.set(t,{token:i,expiresAtMs:a}),i}clearCache(){this.cache.clear()}clearCacheFor(e){let t=this.getCacheKey(e);this.cache.delete(t)}getCacheKey(e){return[e.access_key,e.tenant_id?.toString()||``,e.user_id||``,e.app_id||``,e.app_env||``,e.sandbox_id||``].join(`:`)}getCacheStats(){let e=Date.now(),t=0,n=0;for(let r of this.cache.values())r.expiresAtMs>e?t++:n++;return{total:this.cache.size,valid:t,expired:n}}},b=E(`fs`),x=300*1e3,S=[`/home/gem/workspace/.force/openclaw/miaoda-provider-key`],C=class{cache=new Map;paths;refreshBeforeMs;constructor(e){this.paths=e?.paths??S,this.refreshBeforeMs=e?.refreshBeforeMs??x}getToken(){let e=Date.now();for(let t of this.paths){let n=this.cache.get(t);if(n&&n.expiresAtMs-e>this.refreshBeforeMs)return{token:n.token,accessKey:n.accessKey};let r=this.readAndParse(t);if(r&&(this.cache.set(t,r),r.expiresAtMs>e))return{token:r.token,accessKey:r.accessKey};if(n&&n.expiresAtMs>e)return{token:n.token,accessKey:n.accessKey}}return null}clearCache(){this.cache.clear()}readAndParse(e){try{let t=(0,b.readFileSync)(e,`utf-8`).trim();if(!t)return null;let n=this.decodePayload(t);return n?.exp?{token:t,accessKey:n.access_key??``,expiresAtMs:n.exp*1e3}:null}catch{return null}}decodePayload(e){try{let t=e.split(`.`);if(t.length!==3)return null;let n=(4-(t[1].length%4||4))%4,r=`${t[1]}${`=`.repeat(n)}`.replace(/-/g,`+`).replace(/_/g,`/`);return JSON.parse(Buffer.from(r,`base64`).toString(`utf8`))}catch{return null}}},w=`FORCE_AUTHN_INNERAPI_DOMAIN`,T=`FORCE_AUTHN_ACCESS_KEY`,D=`FORCE_AUTHN_ACCESS_SECRET`;function O(e){if(!e?.enabled)return e?.baseURL;if(e.baseURL)return e.baseURL;let t=e.domainEnv||w,n=process.env[t];if(!n)throw Error(`\u5E73\u53F0\u6A21\u5F0F\u9700\u8981\u57FA\u7840\u57DF\u540D\uFF0C\u8BF7\u8BBE\u7F6E\u73AF\u5883\u53D8\u91CF ${t}`);return n}function k(e,t){if(!t.enabled)return;A(`platform.defaultClaims`,t.defaultClaims);let n;t.tokenProvider?.type===`file`&&(n=new C({paths:t.tokenProvider.paths,refreshBeforeMs:t.refreshBeforeMs}));let r=t.accessKeyEnv||T,i=t.secretKeyEnv||D,a=t.accessKey??(process.env[r]||``),o=new y({accessKey:a,secretKey:t.secretKey??(process.env[i]||``),expireTimeMs:t.expireTimeMs??d,refreshBeforeMs:t.refreshBeforeMs});return e.request.use(e=>{if(n){let t=n.getToken();if(t){let n={...e.headers,Authorization:`Bearer ${t.token}`,"x-api-key":t.accessKey};return{...e,headers:n}}}A(`request.platformAuth.customClaims`,e.platformAuth?.customClaims);let r={...t.defaultClaims||{},...e.platformAuth?.customClaims||{},access_key:a},i=o.getToken(r),s={...e.headers,Authorization:`Bearer ${i}`,"x-api-key":a};return{...e,headers:s}}),o}function A(e,t){if(t&&Object.prototype.hasOwnProperty.call(t,`access_key`))throw Error(`${e} \u4E0D\u5141\u8BB8\u8BBE\u7F6E access_key`)}var j=class{interceptors=[];use(e,t){return this.interceptors.push({onFulfilled:e,onRejected:t}),this.interceptors.length-1}eject(e){this.interceptors[e]&&(this.interceptors[e]=null)}clear(){this.interceptors=[]}forEach(e){this.interceptors.forEach(t=>{t!==null&&e(t)})}};function M(e){if(typeof e!=`object`||!e)return!1;let t=Object.getPrototypeOf(e);return t===Object.prototype||t===null}function N(e){if(!e)return{};if(e instanceof Headers){let t={};return e.forEach((e,n)=>{t[n]=e}),t}if(Array.isArray(e))return e.reduce((e,[t,n])=>(e[t]=n,e),{});let t={};return Object.entries(e).forEach(([e,n])=>{t[e]=Array.isArray(n)?n.join(`,`):n}),t}var P=class e extends Error{isHttpError=!0;response;config;constructor(t,n,r){super(r||`Request failed with status ${t?.status||`unknown`}`),this.name=`HttpError`,this.response=t,this.config=ee(n),Object.setPrototypeOf(this,e.prototype)}};function ee(e){let{headers:t,...n}=e;return{...n,headers:te(t)}}function te(e){if(!e)return;let t=N(e),n={},r=[`authorization`,`x-api-key`,`cookie`,`x-secret`];for(let[e,i]of Object.entries(t)){let t=e.toLowerCase();r.includes(t)?n[e]=`[REDACTED]`:n[e]=i}return n}var F=class{defaultConfig;securityConfig;interceptors={request:new j,response:new j};constructor(e){let{platform:t,security:n,...r}=e||{};this.defaultConfig={timeout:5e3,...r};let i=n?.strictMode??!1;this.securityConfig={allowedProtocols:i?[`http:`,`https:`]:null,maxResponseSize:i?50*1024*1024:0,strictMode:i,...n},t?.enabled&&(this.defaultConfig.baseURL||(this.defaultConfig.baseURL=O(t)),k(this.interceptors,t))}async get(e,t){return this.request({...t,url:e,method:`GET`})}async post(e,t,n){let r={...n?.headers},i=t;return M(t)&&(i=JSON.stringify(t),Object.keys(r).some(e=>e.toLowerCase()===`content-type`)||(r[`Content-Type`]=`application/json`)),this.request({...n,url:e,method:`POST`,body:i,headers:r})}async put(e,t,n){let r={...n?.headers},i=t;return M(t)&&(i=JSON.stringify(t),Object.keys(r).some(e=>e.toLowerCase()===`content-type`)||(r[`Content-Type`]=`application/json`)),this.request({...n,url:e,method:`PUT`,body:i,headers:r})}async delete(e,t){return this.request({...t,url:e,method:`DELETE`})}async patch(e,t,n){let r={...n?.headers},i=t;return M(t)&&(i=JSON.stringify(t),Object.keys(r).some(e=>e.toLowerCase()===`content-type`)||(r[`Content-Type`]=`application/json`)),this.request({...n,url:e,method:`PATCH`,body:i,headers:r})}async request(e){let t=N(this.defaultConfig.headers),n=N(e.headers),r={};for(let e in t)r[e.toLowerCase()]={key:e,value:t[e]};for(let e in n)r[e.toLowerCase()]={key:e,value:n[e]};let i={};for(let e in r){let{key:t,value:n}=r[e];i[t]=n}let a={...this.defaultConfig,...e,headers:i};try{a=await this.runRequestInterceptors(a)}catch(e){return Promise.reject(e)}a.headers=N(a.headers);let o=this.buildUrl(a.url,a.params),s=a.timeout||this.defaultConfig.timeout||5e3,c=new AbortController,l=setTimeout(()=>c.abort(),s),u=a.signal;u&&(u.aborted?(clearTimeout(l),c.abort()):u.addEventListener(`abort`,()=>c.abort(),{once:!0}));let{platformAuth:d,params:f,timeout:p,baseURL:m,url:h,...g}=a;try{let e=await fetch(o,{...g,signal:c.signal});if(clearTimeout(l),this.securityConfig.maxResponseSize>0){let t=e.headers.get(`content-length`);if(t){let e=parseInt(t,10);if(e>this.securityConfig.maxResponseSize)throw Error(`Response size ${e} bytes exceeds limit of ${this.securityConfig.maxResponseSize} bytes`)}}if(!e.ok){let t=new P(e,a);return this.runResponseInterceptors(Promise.reject(t))}return this.runResponseInterceptors(Promise.resolve(e))}catch(e){clearTimeout(l);let t=e instanceof P?e:new P(void 0,a,e.name===`AbortError`?`Request aborted`:e.message);return this.runResponseInterceptors(Promise.reject(t))}}runRequestInterceptors(e){let t=Promise.resolve(e),n=[];this.interceptors.request.forEach(e=>{n.push(e)});for(let e of n)t=t.then(e.onFulfilled,e.onRejected);return t}async runResponseInterceptors(e){let t=e,n=[];this.interceptors.response.forEach(e=>{n.push(e)});for(let e of n)t=t.then(e.onFulfilled,e.onRejected);return t}buildUrl(e,t){let n=this.defaultConfig.baseURL,r;r=!n||/^https?:\/\//i.test(e)?e:n.replace(/\/+$/,``)+`/`+e.replace(/^\/+/,``);let i=new URL(r),{allowedProtocols:a}=this.securityConfig;if(a&&!a.includes(i.protocol))throw Error(`Protocol ${i.protocol} is not allowed. Allowed: ${a.join(`, `)}`);if(t)for(let[e,n]of Object.entries(t))n!=null&&i.searchParams.set(e,String(n));return i.href}}}))();let N;function P(){if(N)return N;try{let e=l(s(g(import.meta.url)),`..`,`package.json`);N=JSON.parse(a(e,`utf-8`))}catch(e){console.warn(`[miaoda-coding] failed to read package.json: ${e instanceof Error?e.message:e}`),N={}}return N}function ee(){let e=process.env.FORCE_AUTHN_INNERAPI_DOMAIN;if(!e)throw Error(`studio-client: 缺少环境变量,需要: FORCE_AUTHN_INNERAPI_DOMAIN`);return{apiUrl:e}}const te=new Set([`k_st_ec_400002683`]);var F=class extends Error{userFacing=!0;constructor(e,t,n){super(e),this.statusCode=t,this.logId=n,this.name=`UserFacingError`}};function ne(e){return!!e&&typeof e==`object`&&!Array.isArray(e)&&Object.values(e).every(e=>typeof e==`string`)}function re(...e){for(let t of e)if(typeof t==`string`&&t)return t}async function*ie(e){let t=new TextDecoder,n=``,r;for await(let i of e){n+=t.decode(i,{stream:!0});let e=n.split(`
|
|
2
2
|
|
|
3
3
|
`);n=e.pop();for(let t of e){let e=t.split(`
|
|
4
|
-
`),n=e.find(e=>e.startsWith(`id:`));n&&(r=n.slice(3).trim());let i=e.find(e=>e.startsWith(`data:`));if(!i)continue;let a=i.slice(5).trim();if(a){if(a===`[DONE]`){yield{data:null,id:r,done:!0};return}try{yield{data:JSON.parse(a),id:r,done:!1}}catch{}}}}}var
|
|
5
|
-
`)){let e=c.trim();if(a){if(e===`## 已完成`){o=!0,s=!1;continue}if(e.startsWith(`## 当前任务`)){s=!0,o=!1;continue}if(e===``)continue}if(o&&e)t.push(e);else if(s&&e){let t=e.match(/^\[([^\]]+)\]\s*(.*)/);if(t){let e=t[2];i=t[1],r=e,!a&&e.startsWith(`需求: `)&&(n=e.slice(4))}else n||=e}}return{history:t,taskDesc:n,lastStatus:r,lastTimestamp:i}}async function
|
|
4
|
+
`),n=e.find(e=>e.startsWith(`id:`));n&&(r=n.slice(3).trim());let i=e.find(e=>e.startsWith(`data:`));if(!i)continue;let a=i.slice(5).trim();if(a){if(a===`[DONE]`){yield{data:null,id:r,done:!0};return}try{yield{data:JSON.parse(a),id:r,done:!1}}catch{}}}}}var ae=class{#e;#t;constructor(){this.#t=ee();let e=P().headers;this.#e=new M.HttpClient({baseURL:this.#t.apiUrl,timeout:9e5,...process.env.FORCE_FRAMEWORK_ENVIRONMENT!==`online`&&ne(e)?{headers:e}:{},platform:{enabled:!0,tokenProvider:{type:`file`}}})}async createSubApp(e={}){let t=`${this.#t.apiUrl}/api/v1/studio/innerapi/openclaw/sub_app/create`,n={};e.message&&(n.message=e.message),e.agentId&&(n.agentId=e.agentId);let r;try{r=await this.#e.post(t,n,{headers:{"Content-Type":`application/json`}})}catch(e){if(e instanceof M.HttpError&&e.response){let t=e.response.headers.get(`x-tt-logid`)??``;throw Error(`createSubApp 请求失败 (logID: ${t}): ${e.response.status}`)}throw e}let i=r.headers?.get?.(`x-tt-logid`)??``,a=await r.json();if(a.BaseResp?.StatusCode!==0&&a.BaseResp?.StatusMessage)throw Error(`createSubApp 业务错误 (logID: ${i}): ${a.BaseResp.StatusMessage}`);if(a.error_msg&&a.status_code!==`0`)throw te.has(a.status_code)?new F(a.error_msg,a.status_code,i):Error(`createSubApp 服务错误 (logID: ${i}): ${a.error_msg}`);let o=re(a.appID,a.appId,a.app_id,a.data?.appID,a.data?.appId,a.data?.app_id),s=re(a.conversationID,a.conversationId,a.conversation_id,a.data?.conversationID,a.data?.conversationId,a.data?.conversation_id);if(!o){let e=Object.keys(a).join(`,`)||`(none)`,t=a.data&&typeof a.data==`object`&&Object.keys(a.data).join(`,`)||`(none)`;throw Error(`createSubApp 返回异常 (logID: ${i}): 缺少 appID,topLevelKeys=${e} dataKeys=${t}`)}return{appID:o,conversationID:s,logId:i}}async*streamChat(e){let t=`${this.#t.apiUrl}/api/v1/studio/innerapi/openclaw/sub_app/${e.appID}/stream_chat`,n={appID:e.appID,message:e.message};e.workDir&&(n.workDir=e.workDir),e.channelAppId&&(n.channelAppId=e.channelAppId);let r={"Content-Type":`application/json`,Accept:`text/event-stream`};e.lastEventID&&(r[`Last-Event-ID`]=e.lastEventID);let i;try{i=await this.#e.post(t,n,{headers:r})}catch(e){if(e instanceof M.HttpError&&e.response){let t=e.response.headers.get(`x-tt-logid`)??``,n=Error(`streamChat 请求失败 (logID: ${t}): ${e.response.status}`);throw n.logId=t,n}let t=e.cause?` cause: ${e.cause.message||e.cause}`:``,n=e.code?` code: ${e.code}`:``;throw Error(`streamChat 网络错误: ${e.message}${n}${t}`)}yield{__meta:!0,logId:i.headers?.get?.(`x-tt-logid`)??``},yield*ie(i.body)}};const I=`.spark`,L=`meta.json`;function oe(e){return c(e,I,L)}async function se(e){try{return JSON.parse(await f(oe(e),`utf8`))}catch{return null}}async function ce(e){if(!e)return null;try{return JSON.parse(await f(c(l(e),I,L),`utf8`))}catch{return null}}async function le(e,t,n){await d(c(e,I),{recursive:!0});let r=await se(e),i={...r??{},...t};if(i.features||=[],!r){let e=await ce(n?.rootDir);if(e)for(let[t,n]of Object.entries(e))t in i||(i[t]=n)}await h(oe(e),JSON.stringify(i,null,2),`utf8`)}function R(e){let t=e;return t?.data?.message??t?.Data?.Message??null}async function ue(e){let t=e.maxReconnect??2,n=e.maxTotalReconnect??10,r=e.reconnectDelayMs??1500,i,a=0,o=0,s=0,c,l;for(;;){let u=!1,d=!1,f=!1,p=i;try{for await(let t of e.createStream(p)){if(e.signal?.aborted)return{totalEvents:s,reconnectCount:o,lastEventId:i,finalText:c,logId:l,success:!1,error:`cancelled`};let n=t;if(n?.__meta){l=n.logId||l;continue}let r=n;if(r.id&&(i=r.id),r.id&&p&&r.id===p)continue;if(r.done)break;s++,a>0&&(a=0);let d=R(r.data);if(d?.type===`metadata`&&d?.status!==`in_progress`&&(u=!0),d?.role===`assistant`&&d?.type===`message`&&d?.status===`completed`&&typeof d?.content==`string`&&d.content&&(c=d.content),e.onEvent&&e.onEvent(r,{lastEventId:i,reconnectCount:o})===!1){f=!0;break}}}catch(e){if(!u){let t=e instanceof Error?e.message:String(e);if(/\b4[0-9]{2}\b/.test(t)||t.includes(`业务错误`))return{totalEvents:s,reconnectCount:o,lastEventId:i,finalText:c,logId:l,success:!1,error:t};d=!0}}if(f)break;if(s===0&&!u&&!d)return{totalEvents:s,reconnectCount:o,lastEventId:i,finalText:c,logId:l,success:!1,error:`stream returned no events`};if(u)break;if(d||i){if(a++,o++,a<=t&&o<=n&&i){if(e.onReconnect?.(o,i),await new Promise(t=>{let n=setTimeout(t,r);e.signal&&e.signal.addEventListener(`abort`,()=>{clearTimeout(n),t()},{once:!0})}),e.signal?.aborted)return{totalEvents:s,reconnectCount:o,lastEventId:i,finalText:c,logId:l,success:!1,error:`cancelled`};continue}let u=`stream disconnected, reconnect exhausted`;return e.onFailed?.(u),{totalEvents:s,reconnectCount:o,lastEventId:i,finalText:c,logId:l,success:!1,error:u}}break}return e.onComplete?.({totalEvents:s,reconnectCount:o,lastEventId:i}),{totalEvents:s,reconnectCount:o,lastEventId:i,finalText:c,logId:l,success:!0}}let de=``,fe=0;function pe(e){try{let t=me(e);if(!t)return null;let n=t.replace(/\s*\(.*\)$/,``).replace(/\s*:.*$/,``),r=Date.now();return n===de&&r-fe<3e4?null:(de=n,fe=r,t)}catch{return null}}function me(e){let t=e.role,n=e.tool_calls,r=e.content;if(e.finish_reason,t===`assistant`&&n?.length)for(let e of n){let t=e?.function?.name??e?.name,n={};try{let t=e?.function?.arguments??e?.arguments;n=typeof t==`string`?JSON.parse(t):t??{}}catch{}if(t===`bash`){let e=String(n.command??``);if(e.includes(`init`))return`初始化项目`;if(e.includes(`db sql`))return`数据库操作`;if(e.includes(`deploy`))return`部署中`}}return t===`tool`&&typeof r==`string`&&(r.startsWith(`执行 run_shell_command 工具失败`)||r.startsWith(`command exitcode:`))?`失败: ${r.slice(0,200)+(r.length>200?`...`:``)}`:null}const z=`.spark`,B=new Intl.DateTimeFormat(`en-CA`,{timeZone:`Asia/Shanghai`,year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1,timeZoneName:`longOffset`});function he(e){return c(e,z,`progress.txt`)}function V(){let e=Object.fromEntries(B.formatToParts(new Date).filter(e=>e.type!==`literal`).map(e=>[e.type,e.value])),t=e.timeZoneName===`GMT`?`+00:00`:e.timeZoneName.replace(/^GMT/,``);return`${e.year}-${e.month}-${e.day}T${e.hour}:${e.minute}:${e.second}${t}`}function ge(e){let t=[],n=``,r=``,i=``,a=e.includes(`## 已完成`)||e.includes(`## 当前任务`),o=!1,s=!a;for(let c of e.split(`
|
|
5
|
+
`)){let e=c.trim();if(a){if(e===`## 已完成`){o=!0,s=!1;continue}if(e.startsWith(`## 当前任务`)){s=!0,o=!1;continue}if(e===``)continue}if(o&&e)t.push(e);else if(s&&e){let t=e.match(/^\[([^\]]+)\]\s*(.*)/);if(t){let e=t[2];i=t[1],r=e,!a&&e.startsWith(`需求: `)&&(n=e.slice(4))}else n||=e}}return{history:t,taskDesc:n,lastStatus:r,lastTimestamp:i}}async function H(e,t){await d(c(e,z),{recursive:!0});let n=he(e),r=[];try{let e=await f(n,`utf8`);if(e.trim()){let t=ge(e);if(r.push(...t.history),t.taskDesc){let e=t.lastTimestamp||V(),n;n=t.lastStatus.startsWith(`失败:`)?t.lastStatus:t.lastStatus===`处理完成`?`成功`:t.lastStatus||`未知状态`,r.push(`[${e}] ${t.taskDesc} — ${n}`)}}}catch{}await h(n,`## 已完成\n\n${r.length?r.join(`
|
|
6
6
|
`)+`
|
|
7
|
-
`:``}\n## 当前任务\n\n${t}\n`,`utf8`)}async function
|
|
8
|
-
`)}`}
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
7
|
+
`:``}\n## 当前任务\n\n${t}\n`,`utf8`)}async function U(e,t){await u(he(e),`[${V()}] ${t}\n`,`utf8`)}async function*_e(e){let{cwd:t,rootDir:n,task:r,agentId:i,signal:a}=e;try{let o=new ae,s=e.requestText??r;await H(t,s.slice(0,200)+(s.length>200?`...`:``)),e.traceId&&await U(t,`trace_id: ${e.traceId}`),await U(t,`已接收需求`),yield{type:`progress`,message:`已接收需求`};let c=await se(t),l=!c?.appId,u=V();if(l){let e;try{e=await o.createSubApp({message:r,agentId:i})}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof F,i=e?.logId;try{await U(t,`失败: ${n}`)}catch{}yield{type:`failed`,error:n,phase:`create`,userFacing:r,logId:i};return}let{appID:a,conversationID:s,logId:l}=e;c={appId:a,conversationId:s,createdAt:u,updatedAt:u},await le(t,c,{rootDir:n}),await U(t,`应用已创建(appId: ${a})`),yield{type:`app_created`,appId:a,conversationId:s,logId:l}}await U(t,`正在处理`),yield{type:`progress`,message:`正在处理`};let d=!1,f=Date.now(),p=setInterval(()=>{U(t,`正在开发中(已 ${Math.floor((Date.now()-f)/6e4)} 分钟)`).catch(()=>{})},6e4),m;try{m=await ue({createStream:n=>o.streamChat({appID:c.appId,message:r,workDir:t,channelAppId:e.channelAppId,...n?{lastEventID:n}:{}}),onEvent:e=>{if(a?.aborted)return!1;let n=R(e.data),r=n?.delta?.type;!d&&n?.role===`assistant`&&(n?.status===`in_progress`||r===`text`)&&(d=!0,U(t,`正在输出结果`).catch(()=>{}));try{if(n?.status===`completed`){let e=pe(n);e&&U(t,e).catch(()=>{})}}catch{}},onReconnect:(t,n)=>{e.onReconnect?.(t,n)},onFailed:n=>{e.onStreamFailed?.(n),U(t,`失败: ${n}`).catch(()=>{})},signal:a})}finally{clearInterval(p)}if(!m.success){let e=m.error?.includes(`disconnected`)||m.error?.includes(`no events`);yield{type:`failed`,error:m.error||`stream failed`,logId:m.logId,phase:`stream`,retryable:e};return}let{finalText:h,logId:g}=m;if(d&&(yield{type:`progress`,message:`正在输出结果`,logId:g||void 0}),a?.aborted){await U(t,`失败: cancelled`),yield{type:`failed`,error:`cancelled`};return}let _=V();await le(t,{...c,updatedAt:_,finalText:h},{rootDir:n}),h&&await U(t,`Agent 回复: ${h.slice(0,200)+(h.length>200?`...`:``)}`),await U(t,`处理完成`),yield{type:`completed`,appId:c.appId,logId:g||void 0,finalText:h}}catch(e){let n=e instanceof Error?e.message:String(e);try{await U(t,`失败: ${n}`)}catch{}yield{type:`failed`,error:n,logId:e?.logId}}}const W=[`research`,`design`,`feedback`];function ve(e){return e.replace(/[^a-zA-Z0-9\u4e00-\u9fff-]/g,`-`).replace(/-+/g,`-`).replace(/^-|-$/g,``).slice(0,60)||`reference`}async function G(e){let t;try{t=await p(e)}catch{return 1}let n=0;for(let e of t){let t=e.match(/^(\d{3})-/);if(t){let e=parseInt(t[1],10);e>n&&(n=e)}}return n+1}async function ye(e){try{let t=await f(c(e,`reference`,`manifest.json`),`utf8`);return JSON.parse(t)}catch{return{updatedAt:new Date().toISOString(),files:[]}}}async function be(e,t){await d(c(e,`reference`),{recursive:!0}),await h(c(e,`reference`,`manifest.json`),JSON.stringify(t,null,2),`utf8`)}async function xe(e){let{appCwd:t,category:n,content:r,filename:i,mode:a=`append`}=e,o=c(t,`reference`,n);if(a===`replace`)try{await m(o,{recursive:!0,force:!0})}catch{}await d(o,{recursive:!0});let s=await G(o),l=`${n}/${`${String(s).padStart(3,`0`)}-${ve(i??`reference`)}.md`}`,u=`reference/${l}`;await h(c(t,u),r,`utf8`);let f=await ye(t);return a===`replace`&&(f.files=f.files.filter(e=>e.category!==n)),f.files.push({path:l,category:n,writtenAt:new Date().toISOString()}),f.updatedAt=new Date().toISOString(),await be(t,f),{relativePath:u,category:n,seq:s}}async function K(e){let t=c(e,`reference`),n;try{n=await p(t)}catch{return{totalFiles:0,categories:{}}}let r={},i=0;for(let e of n){if(!W.includes(e))continue;let n;try{n=(await p(c(t,e))).filter(e=>e.endsWith(`.md`)).sort()}catch{continue}n.length>0&&(r[e]={count:n.length,files:n},i+=n.length)}return{totalFiles:i,categories:r}}async function Se(e){return(await K(e)).totalFiles>0}function Ce(e){let t=e?.log,n=e?.timeoutMs??5e3,r=null,i=null,a=null;return{onSessionStart(e){t?.info?.(`db-schema: session_start sessionId=${e??`unknown`}, clearing cache`),r=null,i=null,a=e??null},async getSchemaSummary(e){return e&&a!==null&&e!==a?(t?.warn?.(`db-schema: session changed without session_start: ${a} -> ${e}, resetting cache`),r=null,i=null,a=e):e&&a===null&&(a=e),r===null?i||(i=(async()=>{t?.info?.(`db-schema: fetching (timeout=${n}ms)`);try{let{stdout:e}=await we(Te(),n),i=q(e);if(i.length===0)return t?.info?.(`db-schema: empty, skip injection for this session`),r=``,null;let a=Ee(i);return r=a,t?.info?.(`db-schema: cached ${i.length} table(s), ${a.length} chars`),a}catch(e){let n=e instanceof Error?e.message:String(e);return t?.warn?.(`db-schema: fetch failed (will not retry this session): ${n}`),r=``,null}finally{i=null}})(),i):r===``?null:r}}}function we(e,t){return new Promise((n,r)=>{let i=setTimeout(()=>r(Error(`timeout after ${t}ms`)),t);e.then(e=>{clearTimeout(i),n(e)},e=>{clearTimeout(i),r(e)})})}function Te(){return new Promise((e,n)=>{t(`npx`,[`-y`,`@lark-apaas/miaoda-data-cli`,`db`,`schema`,`--json`],{timeout:3e4,maxBuffer:10*1024*1024},(t,r,i)=>{if(t)return n(Error(`execFile failed: ${t.message} | stderr: ${i?.slice(0,500)??``} | stdout: ${r?.slice(0,500)??``}`));e({stdout:r,stderr:i})})})}function q(e){let t=e.trim();if(!t)return[];let n=JSON.parse(t);return Array.isArray(n)?n:n&&typeof n==`object`&&n.tableName?[n]:[]}function Ee(e){return`【当前数据库表结构】\n${e.map(e=>{let t=e.fields.map(e=>`${e.fieldName}(${e.type})`).join(`, `),n=e.comment?` (${e.comment})`:``;return`- ${e.tableName}${n}: ${t}`}).join(`
|
|
8
|
+
`)}`}function De(){return`dev`}var Oe=T(C((e=>{Object.defineProperty(e,`__esModule`,{value:!0});var t=10,n=1e3,r=function(e){return JSON.stringify({ev_type:`batch`,list:e})};function i(e){var i=e.transport,a=e.endpoint,o=e.size,s=o===void 0?t:o,c=e.wait,l=c===void 0?n:c,u=[],d=0,f,p,m={getSize:function(){return s},getWait:function(){return l},setSize:function(e){s=e},setWait:function(e){l=e},getEndpoint:function(){return a},setEndpoint:function(e){a=e},send:function(e){u.push(e),u.length>=s&&h.call(this),clearTimeout(d),d=setTimeout(h.bind(this),l)},flush:function(){clearTimeout(d),h.call(this)},getBatchData:function(){return u.length?r(u):``},clear:function(){clearTimeout(d),u=[]},fail:function(e){f=e},success:function(e){p=e}};function h(){if(u.length){var e=this.getBatchData();i.post({url:a,data:e,fail:function(t){f&&f(t,e)},success:function(){p&&p(e)}}),u=[]}}return m}
|
|
9
|
+
/*! *****************************************************************************
|
|
10
|
+
Copyright (c) Microsoft Corporation.
|
|
11
|
+
|
|
12
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
|
13
|
+
purpose with or without fee is hereby granted.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
16
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
17
|
+
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
18
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
19
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
|
20
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
21
|
+
PERFORMANCE OF THIS SOFTWARE.
|
|
22
|
+
***************************************************************************** */
|
|
23
|
+
var a=function(){return a=Object.assign||function(e){for(var t,n=1,r=arguments.length;n<r;n++)for(var i in t=arguments[n],t)Object.prototype.hasOwnProperty.call(t,i)&&(e[i]=t[i]);return e},a.apply(this,arguments)};function o(e){var t=typeof Symbol==`function`&&Symbol.iterator,n=t&&e[t],r=0;if(n)return n.call(e);if(e&&typeof e.length==`number`)return{next:function(){return e&&r>=e.length&&(e=void 0),{value:e&&e[r++],done:!e}}};throw TypeError(t?`Object is not iterable.`:`Symbol.iterator is not defined.`)}function s(e,t){var n=typeof Symbol==`function`&&e[Symbol.iterator];if(!n)return e;var r=n.call(e),i,a=[],o;try{for(;(t===void 0||t-- >0)&&!(i=r.next()).done;)a.push(i.value)}catch(e){o={error:e}}finally{try{i&&!i.done&&(n=r.return)&&n.call(r)}finally{if(o)throw o.error}}return a}function c(e,t,n){if(n||arguments.length===2)for(var r=0,i=t.length,a;r<i;r++)(a||!(r in t))&&(a||=Array.prototype.slice.call(t,0,r),a[r]=t[r]);return e.concat(a||Array.prototype.slice.call(t))}var l=function(){return{}};function u(e){return e}function d(e){return typeof e==`object`&&!!e}var f=Object.prototype;function p(e){return f.toString.call(e)===`[object Array]`}function m(e){return typeof e==`boolean`}function h(e){return typeof e==`number`}function g(e){return typeof e==`string`}function _(e,t){if(!p(e)||e.length===0)return!1;for(var n=0;n<e.length;){if(e[n]===t)return!0;n++}return!1}var v=function(e,t){if(!p(e))return e;var n=e.indexOf(t);if(n>=0){var r=e.slice();return r.splice(n,1),r}return e},y=function(e,t,n){for(var r,i,a=s(t.split(`.`)),o=a[0],c=a.slice(1);e&&c.length>0;)e=e[o],r=c,i=s(r),o=i[0],c=i.slice(1);if(e)return n(e,o)};function b(e){try{return g(e)?e:JSON.stringify(e)}catch(e){return`[FAILED_TO_STRINGIFY]:`+String(e)}}function x(){var e={},t={},n={set:function(r,i){return e[r]=i,t[r]=b(i),n},merge:function(r){return e=a(a({},e),r),Object.keys(r).forEach(function(e){t[e]=b(r[e])}),n},delete:function(r){return delete e[r],delete t[r],n},clear:function(){return e={},t={},n},get:function(e){return t[e]},toString:function(){return a({},t)}};return n}var S=function(){return``.padStart?function(e,t){return t===void 0&&(t=8),e.padStart(t,` `)}:function(e){return e}}(),C=0,w=function(){var e=[...arguments];console.error.apply(console,c([`[SDK]`,Date.now(),S(``+ C++)],s(e),!1))},T=0,E=function(){var e=[...arguments];console.warn.apply(console,c([`[SDK]`,Date.now(),S(``+ T++)],s(e),!1))},D=function(e){return Math.random()<Number(e)},O=function(e,t){return e<Number(t)},k=function(e){return function(t){for(var n=t,r=0;r<e.length&&n;r++)try{n=e[r](n)}catch(e){w(e)}return n}};function A(){for(var e=Array(16),t=0,n=0;n<16;n++)n&3||(t=Math.random()*4294967296),e[n]=t>>>((n&3)<<3)&255;return e}function j(e){for(var t=[],n=0;n<256;++n)t[n]=(n+256).toString(16).substr(1);var r=0,i=t;return[i[e[r++]],i[e[r++]],i[e[r++]],i[e[r++]],`-`,i[e[r++]],i[e[r++]],`-`,i[e[r++]],i[e[r++]],`-`,i[e[r++]],i[e[r++]],`-`,i[e[r++]],i[e[r++]],i[e[r++]],i[e[r++]],i[e[r++]],i[e[r++]]].join(``)}function M(){var e=A();return e[6]=e[6]&15|64,e[8]=e[8]&63|128,j(e)}function N(){var e=!1,t={},n=function(e){e.length&&e.forEach(function(e){try{e()}catch{}}),e.length=0},r=function(e){t[e]&&t[e].forEach(function(e){n(e[1])}),t[e]=void 0};return{set:function(r,i,a){t[r]?t[r].push([i,a]):t[r]=[[i,a]],e&&n(a)},has:function(e){return!!t[e]},remove:r,removeByEvType:function(e){Object.keys(t).forEach(function(r){t[r]&&t[r].forEach(function(t){t[0]===e&&n(t[1])})})},clear:function(){e=!0,Object.keys(t).forEach(function(e){r(e)})}}}var P=500,ee=50;function te(e,t,n){if(t.push(n),!(t.length<P)){var r=t.splice(0,ee);e.savePreStartDataToDb&&e.savePreStartDataToDb(r)}}function F(e){var t=e.getPreStartQueue();t.forEach(function(t){return e.build(t)}),t.length=0}var ne=[`init`,`start`,`config`,`beforeDestroy`,`provide`,`beforeReport`,`report`,`beforeBuild`,`build`,`beforeSend`,`send`,`beforeConfig`];function re(e){var t=e.builder,n=e.createSender,r=e.createDefaultConfig,i=e.createConfigManager,a=e.userConfigNormalizer,o=e.initConfigNormalizer,l=e.validateInitConfig,u,f,p={};ne.forEach(function(e){return p[e]=[]});var m=!1,h=!1,g=!1,y=[],b=[],x=N(),S={getBuilder:function(){return t},getSender:function(){return u},getPreStartQueue:function(){return y},init:function(e){if(m){E(`already inited`);return}if(e&&d(e)&&l(e)){var t=r(e);if(!t)throw Error(`defaultConfig missing`);var a=o(e);if(f=i(t),f.setConfig(a),f.onChange(function(){C(`config`)}),u=n(f.getConfig()),!u)throw Error(`sender missing`);m=!0,C(`init`,!0)}else throw Error(`invalid InitConfig, init failed`)},set:function(e){m&&e&&d(e)&&(C(`beforeConfig`,!1,e),f?.setConfig(e))},config:function(e){if(m)return e&&d(e)&&(C(`beforeConfig`,!1,e),f?.setConfig(a(e))),f?.getConfig()},provide:function(e,t){if(_(b,e)){E(`cannot provide `+e+`, reserved`);return}S[e]=t,C(`provide`,!1,e)},start:function(){m&&(h||f?.onReady(function(){h=!0,C(`start`,!0),F(S)}))},report:function(e){if(e){var t=k(p.beforeReport)(e);if(t){var n=k(p.report)(t);n&&(h?this.build(n):te(S,y,n))}}},build:function(e){if(h){var n=k(p.beforeBuild)(e);if(n){var r=t.build(n);if(r){var i=k(p.build)(r);i&&this.send(i)}}}},send:function(e){if(h){var t=k(p.beforeSend)(e);t&&(u.send(t),C(`send`,!1,t))}},destroy:function(){x.clear(),g=!0,y.length=0,C(`beforeDestroy`,!0)},on:function(e,t){if(e===`init`&&m||e===`start`&&h||e===`beforeDestroy`&&g)try{t()}catch{}else p[e]&&p[e].push(t)},off:function(e,t){p[e]&&(p[e]=v(p[e],t))},destroyAgent:x};return b=Object.keys(S),S;function C(e,t){t===void 0&&(t=!1);var n=[...arguments].slice(2);p[e].forEach(function(e){try{e.apply(void 0,c([],s(n),!1))}catch{}}),t&&(p[e].length=0)}}var ie=function(e){var t=x();e.provide(`context`,t),e.on(`report`,function(e){return e.extra||={},e.extra.context=t.toString(),e})};function ae(e,t){e.on(`init`,function(){var n=[],r=function(r){r.forEach(function(r){var i=r.name;_(n,i)||(n.push(i),r.setup(e),t&&t(i,r.setup),e.destroyAgent.set(i,i,[function(){n=v(n,i),r.tearDown&&r.tearDown()}]))})};e.provide(`applyIntegrations`,r);var i=e.config();i&&i.integrations&&r(i.integrations)})}var I=function(e){try{typeof window==`object`&&d(window)&&window.__SLARDAR_DEVTOOLS_GLOBAL_HOOK__&&window.__SLARDAR_DEVTOOLS_GLOBAL_HOOK__.push(e)}catch{}},L=function(){return Date.now()};function oe(){if(typeof window==`object`&&d(window))return window}var se=function(e){if(e)return e.__SLARDAR_REGISTRY__||={Slardar:{plugins:[],errors:[],subject:{}}},e.__SLARDAR_REGISTRY__.Slardar},ce=function(){var e=[...arguments],t=se(oe());t&&(t.errors||=[],t.errors.push(e))},le=function(e){var t;if(!(typeof window!=`object`||!window.__perfsee__)){var n={};return(t=Error.captureStackTrace)==null||t.call(Error,n,e),n.stack}},R=`custom`,ue=`event`,de=`log`,fe=function(e){if(!(!e||!d(e))&&!(!e.name||!g(e.name))){var t={name:e.name,type:ue};if(`metrics`in e&&d(e.metrics)){var n=e.metrics,r={};for(var i in n)h(n[i])&&(r[i]=n[i]);t.metrics=r}if(`categories`in e&&d(e.categories)){var a=e.categories,o={};for(var i in a)o[i]=b(a[i]);t.categories=o}return`attached_log`in e&&g(e.attached_log)&&(t.attached_log=e.attached_log),t}},pe=function(e){if(!(!e||!d(e))&&!(!e.content||!g(e.content))){var t=e.content,n={content:b(t),type:de,level:`info`};if(`level`in e&&(n.level=e.level),`extra`in e&&d(e.extra)){var r=e.extra,i={},a={};for(var o in r)h(r[o])?i[o]=r[o]:a[o]=b(r[o]);n.metrics=i,n.categories=a}return`attached_log`in e&&g(e.attached_log)&&(n.attached_log=e.attached_log),n}},me=function(e){var t=function(n){var r=fe(n);if(r){var i=le(t);i&&(r.stacks=i),e.report({ev_type:R,payload:r,extra:{timestamp:L()}})}};e.provide(`sendEvent`,t),e.provide(`sendLog`,function(t){var n=pe(t);n&&e.report({ev_type:R,payload:n,extra:{timestamp:L()}})})},z=function(e,t){var n=e.common||{};return n.sample_rate=t,e.common=n,e},B=function(e,t,n,r,i){return e?(function(e){return function(){return e}})(i(r,t)):function(){return n(t)}},he=function(e,t){return e.map(function(e){switch(t){case`number`:return Number(e);case`boolean`:return e===`1`;default:return String(e)}})},V=function(e,t,n){switch(n){case`eq`:return _(t,e);case`neq`:return!_(t,e);case`gt`:return e>t[0];case`gte`:return e>=t[0];case`lt`:return e<t[0];case`lte`:return e<=t[0];case`regex`:return!!e.match(new RegExp(t.join(`|`)));case`not_regex`:return!e.match(new RegExp(t.join(`|`)));default:return!1}},ge=function(e,t,n,r){var i=y(e,t,function(e,t){return e[t]});return i===void 0?!1:V(i,he(r,m(i)?`bool`:h(i)?`number`:`string`),n)},H=function(e,t){try{return t.type===`rule`?ge(e,t.field,t.op,t.values):t.type===`and`?t.children.every(function(t){return H(e,t)}):t.children.some(function(t){return H(e,t)})}catch(e){return ce(e),!1}},U=function(e,t,n,r,i,a){var o={};return Object.keys(e).forEach(function(s){var c=e[s],l=c.enable,u=c.sample_rate,d=c.conditional_sample_rules;l?(o[s]={enable:l,sample_rate:u,effectiveSampleRate:u*n,hit:B(t,u,r,i,a)},d&&(o[s].conditional_hit_rules=d.map(function(e){var o=e.sample_rate,s=e.filter;return{sample_rate:o,hit:B(t,o,r,i,a),effectiveSampleRate:o*n,filter:s}}))):o[s]={enable:l,hit:function(){return!1},sample_rate:0,effectiveSampleRate:0}}),o},_e=function(e,t,n,r,i){if(!t)return u;var a=t.sample_rate,o=t.include_users,s=t.sample_granularity,c=t.rules,l=t.r,d=l===void 0?Math.random():l;if(_(o,e))return function(e){return z(e,1)};var f=s===`session`,p=B(f,a,n,d,r),m=U(c,f,a,n,d,r);return function(e){if(!p())return f&&i[0](),!1;if(!(e.ev_type in m))return z(e,a);if(!m[e.ev_type].enable)return f&&i[1](e.ev_type),!1;if(e.common?.sample_rate)return e;var t=m[e.ev_type],n=t.conditional_hit_rules;if(n){for(var r=0;r<n.length;r++)if(H(e,n[r].filter))return n[r].hit()?z(e,n[r].effectiveSampleRate):!1}return t.hit()?z(e,t.effectiveSampleRate):(!(n&&n.length)&&f&&i[1](e.ev_type),!1)}},W=function(e){e.on(`start`,function(){var t=e.config(),n=t.userId,r=t.sample,i=_e(n,r,D,O,[function(){e.destroy()},function(t){e.destroyAgent.removeByEvType(t)}]);e.on(`build`,i)})},ve={build:function(e){return{ev_type:e.ev_type,payload:e.payload,common:a(a({},e.extra||{}),e.overrides||{})}}},G=`mon.zijieapi.com`,ye=G,be=`2.1.8`,xe=`SDK_BASE`,K=`/monitor_web/settings/browser-settings`,Se=`/monitor_browser/collect/batch/`,Ce={sample_rate:1,include_users:[],sample_granularity:`session`,rules:{}},we=20,Te=`session`;function q(e){var t,n,r=[`userId`,`deviceId`,`sessionId`,`env`];try{for(var i=o(r),a=i.next();!a.done;a=i.next()){var s=a.value;e[s]||delete e[s]}}catch(e){t={error:e}}finally{try{a&&!a.done&&(n=i.return)&&n.call(i)}finally{if(t)throw t.error}}return e}function Ee(e){return q(a({},e))}function De(e){return d(e)&&`bid`in e&&`transport`in e}function Oe(e){return q(a({},e))}function ke(e){if(!e)return{};var t=e.sample,n=e.timestamp,r=e.quota_rate,i=r===void 0?1:r;if(!t)return{};var a=t.sample_rate,o=t.sample_granularity,s=o===void 0?Te:o,c=t.include_users,l=t.rules,u=l===void 0?[]:l;return{sample:{include_users:c,sample_rate:a*i,sample_granularity:s,rules:u.reduce(function(e,t){var n=t.name;return e[n]={enable:t.enable,sample_rate:t.sample_rate,conditional_sample_rules:t.conditional_sample_rules},e},{})},serverTimestamp:n}}var Ae=function(e,t){return t===void 0&&(t=Se),(e&&e.indexOf(`//`)>=0?``:`https://`)+e+t},J=function(e,t){return t===void 0&&(t=K),(e&&e.indexOf(`//`)>=0?``:`https://`)+e+t},je=function(){return M()},Me=function(){return M()},Ne=function(e){return e+`_`+Date.now()},Y=function(){return M()},Pe=function(e){var t=e,n,r={},i,o,s=l,c=l;return{getConfig:function(){return t},setConfig:function(e){return r=a(a({},r),e||{}),u(),n||(n=e,t.useLocalConfig||!t.bid?(o={},s()):Fe(t.transport,t.domain,t.bid,function(e){i=e,d()})),t},onChange:function(e){c=e},onReady:function(e){s=e,o&&s()}};function u(){var n=a(a(a({},e),o||{}),r);n.sample=X(X(e.sample,o?.sample),r.sample),t=n,c()}function d(){o=ke(i),u(),s()}};function Fe(e,t,n,r){if(!e.get)return r({});e.get({withCredentials:!0,url:J(t)+`?bid=`+n+`&store=1`,success:function(e){r(e.data||{})},fail:function(){r({sample:{sample_rate:.001}})}})}function X(e,t){if(!e||!t)return e||t;var n=a(a({},e),t);return n.include_users=c(c([],s(e.include_users||[]),!1),s(t.include_users||[]),!1),n.rules=c(c([],s(Object.keys(e.rules||{})),!1),s(Object.keys(t.rules||{})),!1).reduce(function(n,r){return r in n||(r in(e.rules||{})&&r in(t.rules||{})?(n[r]=a(a({},e.rules[r]),t.rules[r]),n[r].conditional_sample_rules=c(c([],s(e.rules[r].conditional_sample_rules||[]),!1),s(t.rules[r].conditional_sample_rules||[]),!1)):n[r]=e.rules?.[r]||t.rules?.[r]),n},{}),n}var Ie=function(e,t){var n=t||{},r=n.version,i=n.name,o={url:``,protocol:``,domain:``,path:``,query:``,timestamp:Date.now(),sdk_version:r||be,sdk_name:i||xe};return a(a({},e),{extra:a(a({},o),e.extra||{})})},Z=function(e){e.on(`report`,function(t){return Ie(t,e.config())})},Q=function(e,t){var n={};return n.bid=t.bid,n.pid=t.pid,n.view_id=t.viewId,n.user_id=t.userId,n.device_id=t.deviceId,n.session_id=t.sessionId,n.release=t.release,n.env=t.env,a(a({},e),{extra:a(a({},n),e.extra||{})})},Le=function(e){e.on(`beforeBuild`,function(t){return Q(t,e.config())})};function Re(e){return i(e)}var ze=function(e){return{bid:``,pid:``,viewId:Ne(`_`),userId:je(),deviceId:Me(),sessionId:Y(),domain:G,release:``,env:`production`,sample:Ce,plugins:{},transport:{get:l,post:l},useLocalConfig:!1}},Be=function(e){var t=e===void 0?{}:e,n=t.createSender,r=n===void 0?function(e){return Re({size:we,endpoint:Ae(e.domain),transport:e.transport})}:n,i=t.builder,a=i===void 0?ve:i,o=t.createDefaultConfig,s=re({validateInitConfig:De,initConfigNormalizer:Ee,userConfigNormalizer:Oe,createSender:r,builder:a,createDefaultConfig:o===void 0?ze:o,createConfigManager:Pe});return ie(s),Le(s),Z(s),ae(s),I(s),s},$=function(e){e===void 0&&(e={});var t=Be(e);return W(t),me(t),t},Ve=$();e.BATCH_REPORT_PATH=Se,e.CustomPlugin=me,e.DEFAULT_SAMPLE_CONFIG=Ce,e.DEFAULT_SAMPLE_GRANULARITY=Te,e.DEFAULT_SENDER_SIZE=we,e.InjectConfigPlugin=Le,e.InjectEnvPlugin=Z,e.REPORT_DOMAIN=G,e.SDK_NAME=xe,e.SDK_VERSION=be,e.SETTINGS_DOMAIN=ye,e.SETTINGS_PATH=K,e.addConfigToReportEvent=Q,e.addEnvToSendEvent=Ie,e.builder=ve,e.createBaseClient=$,e.createConfigManager=Pe,e.createMinimalClient=Be,e.default=Ve,e.getDefaultConfig=ze,e.getDefaultDeviceId=Me,e.getDefaultSessionId=Y,e.getDefaultUserId=je,e.getReportUrl=Ae,e.getServerConfig=Fe,e.getSettingsUrl=J,e.getViewId=Ne,e.mergeSampleConfig=X,e.normalizeInitConfig=Ee,e.normalizeStrictFields=q,e.normalizeUserConfig=Oe,e.parseServerConfig=ke,e.validateInitConfig=De}))(),1);function ke(e){return!e||typeof e!=`object`||!(`default`in e)?e:e.default}function Ae(e){let t=ke(e);return ke(t)??t??e??{}}const J=Ae(Oe.default);let je=!1;function Me(){if(!je)try{let e=P();J.init({bid:`apaas_miaoda`,release:String(e.version??`unknown`),env:De(),transport:{get:({url:e,success:t,fail:n})=>fetch(e).then(e=>e.json()).then(e=>t?.(e)).catch(e=>n?.(e)),post:({url:e,data:t})=>{let n=typeof t==`string`?t:JSON.stringify(t);return fetch(e,{method:`POST`,headers:{"Content-Type":`application/json`},body:n}).catch(()=>{})}}}),J.start(),je=!0}catch{}}function Ne(e){if(e!==void 0)return String(e)}function Y(e){let t={source:`node-plugin`},n=e.traceId,r=e.projectId,i=Ne(e.isNew),a=e.appId,o=e.agentId,s=e.sessionKey;return n&&(t.trace_id=n),r&&(t.project_id=r),i&&(t.is_new=i),a&&(t.app_id=a),o&&(t.agent_id=o),e.domain&&(t.domain=e.domain),s&&(t.session_key=s),t}function Pe(e,t){let n={...e};for(let[e,r]of Object.entries(t??{}))r!=null&&(n[e]=typeof r==`boolean`?String(r):r);return n}function Fe(e){try{let t={},n=Y(e),r=e.durationMs,i=e.sseEventCount,a=e.reconnectCount,o=e.createDurationMs;typeof r==`number`&&(t.duration_ms=r),typeof i==`number`&&(t.sse_event_count=i),typeof a==`number`&&(t.reconnect_count=a),typeof o==`number`&&(t.create_duration_ms=o);let s={...n,status:e.status,is_new:n.is_new??`false`},c=e.errorType;e.phase&&(s.phase=e.phase),c&&(s.error_type=c.slice(0,100)),J.sendEvent?.({name:`miaoda_coding_task`,metrics:t,categories:s})}catch{}}function X(e){try{let t={},n=Y(e);typeof e.durationMs==`number`&&(t.duration_ms=e.durationMs);let r={...n,request_name:e.requestName,status:e.status};e.logId&&(r.log_id=e.logId),e.phase&&(r.phase=e.phase),e.errorType&&(r.error_type=e.errorType.slice(0,100)),J.sendEvent?.({name:`miaoda_coding_request`,metrics:t,categories:r})}catch{}}function Ie(e){try{if(!e.projectId)return;let t=Pe(Y(e),{event:e.event,phase:e.phase}),n=e.logIds;n?.length&&(t.log_ids=n.join(`,`)),J.sendLog?.({content:e.error,level:`error`,extra:t})}catch{}}function Z(e){try{let t=Pe(Y(e),{event:e.event,phase:e.phase,status:e.status,...e.extra});J.sendLog?.({content:e.message,level:e.level??`info`,extra:t})}catch{}}const Q=`这个需求更适合去妙搭主站继续完成,我帮你把入口放这里了:`;function Le(){return[`如果需求命中 miaoda-coding skill 中的不适用场景,直接输出最终给用户看的 Markdown 文案,不要解释原因,也不要给替代方案。`,`固定输出结构只有两段:第一段固定为“${Q}”,第二段是一条指向 miaoda.feishu.cn/home 的 Markdown 链接。`,`label 改成最贴合需求的一句话,例如“在妙搭创建电商平台”“在妙搭创建知识库应用”;query 需要是给 miaoda.feishu.cn/home 的需求摘要,尽量忠实保留用户原始需求。`,`不要输出任何内部标记或 XML/HTML 风格包裹内容;主路径必须直接输出最终文案。也不要输出 HTML <a> 标签、不要输出裸 URL、不要在链接前后补能力边界说明。`,`正确示例 1:`,`用户:帮我做一个带用户登录、支付、订单管理、后台审核和数据报表的大型电商平台`,`助手:`,Q,``,`[在妙搭创建电商平台](https://miaoda.feishu.cn/home?userInput=%E5%B8%A6%E7%94%A8%E6%88%B7%E7%99%BB%E5%BD%95%E3%80%81%E6%94%AF%E4%BB%98%E3%80%81%E8%AE%A2%E5%8D%95%E7%AE%A1%E7%90%86%E3%80%81%E5%90%8E%E5%8F%B0%E5%AE%A1%E6%A0%B8%E5%92%8C%E6%95%B0%E6%8D%AE%E6%8A%A5%E8%A1%A8%E7%9A%84%E5%A4%A7%E5%9E%8B%E7%94%B5%E5%95%86%E5%B9%B3%E5%8F%B0&mode=application&open-from=openclaw)`,``,`正确示例 2:`,`用户:做一个企业知识库问答系统,支持文档检索和智能问答`,`助手:`,Q,``,`[在妙搭创建知识库应用](https://miaoda.feishu.cn/home?userInput=%E4%BC%81%E4%B8%9A%E7%9F%A5%E8%AF%86%E5%BA%93%E9%97%AE%E7%AD%94%E7%B3%BB%E7%BB%9F%EF%BC%8C%E6%94%AF%E6%8C%81%E6%96%87%E6%A1%A3%E6%A3%80%E7%B4%A2%E5%92%8C%E6%99%BA%E8%83%BD%E9%97%AE%E7%AD%94&mode=application&open-from=openclaw)`,``,`错误示例:`,`这个需求超出了妙搭的能力范围,建议参考文档后再试。`,`错误原因:解释了能力边界,而且没有直接给出妙搭主站链接。`].join(`
|
|
24
|
+
`)}const Re=/^[a-z0-9][a-z0-9-]{0,63}$/,ze=new Set([`.`,`..`,`con`,`nul`,`prn`,`aux`]);function Be(e){let t=e?.config?.plugins?.entries?.[`openclaw-extension-miaoda-coding`]?.config??{};return{verbose:t?.verbose===!0,hooks:{allowPromptInjection:t?.hooks?.allowPromptInjection!==!1}}}function $(e,t){if(!e)return;let r=o.dirname(process.execPath),i=n(`openclaw`,[`message`,`send`,`--channel`,`feishu`,`--target`,e,`--message`,t],{stdio:[`ignore`,`ignore`,`pipe`],env:{...process.env,PATH:`${r}:${process.env.PATH||``}`}});i.stderr.on(`data`,()=>{}),i.on(`close`,()=>{}),i.unref()}function Ve(e){return String(e??``).replace(/[\u0000-\u0008\u000B\u000C\u000E-\u001F\u007F]/g,``).replace(/\s+/g,` `).trim()}function He(e){let t=String(e??``).replace(/[\u0000-\u0008\u000B\u000C\u000E-\u001F\u007F]/g,``).trim();if(!t)throw Error(`task 不能为空`);if(t.length>4e3)throw Error(`task 不能超过 4000 个字符`);return t}function Ue(e){let t=Ve(e);if(!We(t))throw Error(`project_id 不合法: "${t}"`);return t}function We(e){return Re.test(e)&&!ze.has(e)}function Ge(e){if(e==null||e===``)return;let t=Ve(e);if(t!==`none`){if(!/^(user|chat|thread):\S+$/.test(t))throw Error(`target 不合法`);return t}}function Ke(e,t){let n=k(e,t);if(i.existsSync(n))return n;let r=A(e,t);return i.existsSync(r)?r:n}function qe(e){let t=new Map,n=o.join(e,`app`);try{for(let e of i.readdirSync(n,{withFileTypes:!0}))!e.isDirectory()||!We(e.name)||t.set(e.name,o.join(n,e.name))}catch{}try{for(let n of i.readdirSync(e,{withFileTypes:!0}))!n.isDirectory()||n.name===`app`||!We(n.name)||t.has(n.name)||t.set(n.name,o.join(e,n.name))}catch{}return[...t.entries()].map(([e,t])=>({projectId:e,appCwd:t}))}function Je(e){try{return i.readFileSync(e,`utf8`).split(`
|
|
25
|
+
`).map(e=>e.trim()).filter(Boolean)}catch{return[]}}function Ye(e){return e.replace(/^\[[^\]]+\]\s*/,``)}function Xe(e){return e.replace(/应用已创建(appId:\s*[^)]+)/g,`应用已创建`)}function Ze(e){return e.slice(-3).map(Ye).map(Xe).join(` | `)||`暂无进度记录`}function Qe(e){let t=Ze(e);return t.includes(`失败:`)?`failed`:t.includes(`处理完成`)?`completed`:t===`暂无进度记录`?`pending`:`running`}function $e(e){try{let t=JSON.parse(i.readFileSync(o.join(e,`.spark`,`meta.json`),`utf8`));return typeof t?.appId==`string`&&t.appId.trim().length>0}catch{return!1}}var et={id:`openclaw-extension-miaoda-coding`,name:`妙搭`,description:`妙搭制品创建与修改`,register(e){let t=`[miaoda-coding]`,n={info:n=>e.logger.info?.(`${t} ${n}`),warn:n=>e.logger.warn?.(`${t} ${n}`),error:n=>e.logger.error?.(`${t} ${n}`)},a={verbose:!1,hooks:{allowPromptInjection:!0}},s=Ce({log:n});n.info(`plugin registered`),Me(),e.on(`session_start`,(e,t)=>{try{s.onSessionStart(e?.sessionId)}catch(e){n.warn(`session_start reset failed: ${String(e)}`)}});let c=new Set([`miaoda_coding`]),l=new Set([`miaoda_check_progress`]);e.on(`before_tool_call`,(e,t)=>{let r=e.toolName,i=t?.sessionKey??``,a=i.toLowerCase().split(`:`).includes(`subagent`);if(c.has(r)&&!a)return n.warn(`blocked ${r} from non-subagent session: ${i}`),{block:!0,blockReason:`${r} 只能由 subagent 调用。请通过 sessions_spawn(runtime="subagent", mode="run") 派活,不要直接调用此工具。读 miaoda-coding skill 了解完整流程。`};if(l.has(r)&&a)return n.warn(`blocked ${r} from subagent session: ${i}`),{block:!0,blockReason:`${r} 不能由 subagent 调用。subagent 不应直接操作参考资料或查询进度。`}}),e.on(`before_prompt_build`,async(e,t)=>{if(!Be(t).hooks.allowPromptInjection)return;let r={prependSystemContext:[`【硬规则 — 妙搭】`,`当用户要求创建或修改应用、网页、页面时,必须通过 sessions_spawn(runtime="subagent", mode="run") 交给妙搭。`,`读 miaoda-coding skill 后严格按指示操作。只传 runtime、mode、task 三个参数,不要传 streamTo/sandbox 等。`,Le(),`如果需求在支持范围内并且已经派活给 subagent,最终回复可以是正常交付文案:一条简短总结,外加预览链接;不要把“不适用场景”的主站引导规则误用到正常交付场景。`,`绝对不要自己写代码或用 exec 生成文件。如果 sessions_spawn 调用本身返回错误,去掉多余参数后重试。如果 subagent 执行失败需要重试,generation_request 填"继续"(不要重复发完整需求),详见 skill 中的重试流程。`,`如果对话中有需要传递给 code agent 的上下文(研究结果、用户偏好等),先调 miaoda_write_reference 写入 reference 目录,再 spawn。用户反悔或需要替换之前的上下文时,使用 miaoda_write_reference 的 mode=replace。妙搭只做制品生成,不做研究。`,`不要用 sessions_history 读取子 agent 的 session,查进度用 miaoda_check_progress tool(仅限建应用场景,数据库操作等不要用)。`,`subagent 完成后只发一条消息,严禁发多条。不要调 message tool。如果 result 的 hint 说预览链接已自动发送,则只发纯文字总结,不要再带任何链接;否则附上预览链接,格式用加粗 Markdown 链接 **[url](url)**,不要发裸 URL。`].join(` `)};try{let e=await s.getSchemaSummary(t?.sessionId);e&&(r.appendSystemContext=e)}catch(e){n.error(`db schema injection threw unexpected error: ${String(e)}`)}return r}),e.registerTool(e=>{let t=j(e);return{name:`list_projects`,label:`列出妙搭项目`,description:`列出已创建的妙搭项目,返回 project_id、appId、updatedAt、finalText、status 和进度摘要。修改已有项目时必须先调此工具。`,parameters:{type:`object`,properties:{}},async execute(){let e=[];try{e=qe(t.workspaceDir).map(({projectId:e,appCwd:t})=>{let n=o.join(t,`.spark`,`meta.json`),r=o.join(t,`.spark`,`progress.txt`),a=null;try{a=JSON.parse(i.readFileSync(n,`utf8`))}catch{}let s=Je(r);return{project_id:e,appId:a?.appId??null,updatedAt:a?.updatedAt??null,finalText:a?.finalText??null,status:Qe(s),progressSummary:Ze(s)}})}catch{}return{content:[{type:`text`,text:JSON.stringify({projects:e},null,2)}]}}}}),e.registerTool(e=>{let t=j(e);return{name:`miaoda_check_progress`,label:`查看妙搭应用生成进度`,description:`查看通过 miaoda_coding 创建的应用的生成进度。仅用于 sessions_spawn 派活建应用后查看进度,不适用于数据库操作等其他场景。`,parameters:{type:`object`,properties:{project_id:{type:`string`,description:`项目目录名(从 list_projects 获取)`}},required:[`project_id`]},async execute(e,n){try{let e=Ue(n.project_id),r=Ke(t.workspaceDir,e),a=o.join(r,`.spark`,`progress.txt`),s=i.readFileSync(a,`utf8`).trim().split(`
|
|
26
|
+
`).filter(Boolean),c=null;try{c=JSON.parse(i.readFileSync(o.join(r,`.spark`,`meta.json`),`utf8`))}catch{}let l=!!c?.finalText,u=s.slice(-20).map(Xe).join(`
|
|
27
|
+
`);return{content:[{type:`text`,text:JSON.stringify({status:`ok`,project_id:e,lines:s.length,progress:u,delivery_ready:l,finalTextAvailable:l,note:`miaoda_check_progress 只用于查看进度,不能根据 appId 或进度文本推导预览链接;最终链接只以 subagent completion result 中的 output/finalText 为准。`})}]}}catch{return{content:[{type:`text`,text:JSON.stringify({status:`no_progress`,project_id:n.project_id,message:`暂无进度记录`,delivery_ready:!1,finalTextAvailable:!1,note:`miaoda_check_progress 只用于查看进度,不能根据 appId 或进度文本推导预览链接;最终链接只以 subagent completion result 中的 output/finalText 为准。`})}]}}}}}),e.registerTool(e=>{let t=j(e);return{name:`miaoda_write_reference`,label:`写入妙搭参考资料`,description:`将参考资料写入妙搭项目的 reference 目录,供 code agent 在生成时自行查阅。按 category 分类写入。mode=append(默认)追加,mode=replace 替换该 category 全部内容(用于反悔/调整场景)。`,parameters:{type:`object`,properties:{project_id:{type:`string`,description:`项目目录名`},category:{type:`string`,enum:[`research`,`design`,`feedback`],description:`资料类别:research(调研结果)、design(设计要求)、feedback(用户反馈)`},content:{type:`string`,description:`参考资料内容(Markdown 格式,摘要+关键原文引用)`},filename:{type:`string`,description:`可选文件名(不含扩展名)`},mode:{type:`string`,enum:[`append`,`replace`],description:`写入模式:append(追加,默认)、replace(替换该 category 全部内容)`}},required:[`project_id`,`category`,`content`]},async execute(e,n){try{let e=Ue(n.project_id);if(!W.includes(n.category))throw Error(`category 不合法,必须是 ${W.join(`、`)} 之一`);let r=n.content?.trim();if(!r)throw Error(`content 不能为空`);if(r.length>5e4)throw Error(`content 不能超过 50000 个字符`);let i=n.mode===`replace`?`replace`:`append`,a=Ke(t.workspaceDir,e),o=await xe({appCwd:a,category:n.category,content:r,filename:n.filename,mode:i}),s=await K(a);return{content:[{type:`text`,text:JSON.stringify({status:`ok`,...o,summary:s},null,2)}]}}catch(e){return{content:[{type:`text`,text:JSON.stringify({status:`error`,error:e instanceof Error?e.message:String(e)})}]}}}}});let u=900*1e3;e.registerTool(e=>{let t=j(e),s=e.config?.channels?.feishu,c=e.agentAccountId,l=(c?s?.accounts?.[c]:void 0)?.appId??s?.appId;return n.info(`channelAppId resolved: ${l??`(none)`} (accountId=${c??`(none)`})`),{name:`miaoda_coding`,label:`妙搭`,description:`创建或修改妙搭应用/网页。只接受生成指令,不接受搜索/调研等任务。如有参考资料,应事先通过 miaoda_write_reference 写入 reference 目录。修改已有项目时必须先调 list_projects 获取 project_id。返回结构化 JSON(status/appId/finalText/output/meta)。`,parameters:{type:`object`,properties:{generation_request:{type:`string`,description:`生成指令(如'做一个学习网站,包含章节、题库、测验')`},project_id:{type:`string`,description:`本地项目目录名,仅允许小写字母、数字和短横线`},target:{type:`string`,description:`飞书消息推送目标:单聊用 user:ou_xxx,群聊用 chat:oc_xxx,非飞书渠道传 none`}},required:[`generation_request`,`project_id`,`target`]},async execute(s,c){let d=r();try{let r=Date.now(),s=He(c.generation_request),f=Ue(c.project_id),p=Ge(c.target),m=process.env.FORCE_AUTHN_INNERAPI_DOMAIN,h=Ke(t.workspaceDir,f);i.mkdirSync(h,{recursive:!0});let g=!$e(h),_={traceId:d,projectId:f,isNew:g,agentId:t.agentId,domain:m,sessionKey:e?.sessionKey};n.info(`miaoda_coding called (project=${f}, isNew=${g}, traceId=${d})`),Z({..._,event:`start`,message:`miaoda_coding started for ${f}`,extra:{target:p??`none`}}),g&&Z({..._,event:`create_start`,message:`createSubApp started for ${f}`});let v=await Se(h)?`工作目录下有 reference/ 目录,包含参考资料,请先查阅再开始工作。\n\n${s}`:s,y=o.join(h,`.agent`);if(!i.existsSync(y)){let e=O(D());if(i.existsSync(e))try{i.symlinkSync(e,y)}catch{}}let b=[],x,S=!1,C=!1,w=null,T=g?Date.now():void 0,E=g?void 0:Date.now(),k=new AbortController,A=setTimeout(()=>k.abort(),u);try{for await(let e of _e({cwd:h,rootDir:D(),task:v,requestText:s,traceId:d,agentId:t.agentId,channelAppId:l,signal:k.signal,onReconnect:(e,t)=>{n.warn(`stream reconnect (project=${f}, attempt=${e}, lastEventId=${t??`none`}, traceId=${d})`),Z({..._,event:`stream_reconnect`,message:`stream reconnect for ${f}`,level:`warn`,extra:{attempt:e,last_event_id:t??`none`}})},onStreamFailed:e=>{n.error(`stream failed (project=${f}, traceId=${d}): ${e}`),Z({..._,event:`stream_failed`,message:String(e),level:`error`})}})){if(`logId`in e&&e.logId&&b.push(e.logId),a.verbose&&p)switch(e.type){case`app_created`:$(p,`应用已创建(appId: ${e.appId})`);break;case`failed`:$(p,`处理失败: ${e.error}`);break}if(e.type===`completed`&&(w={appId:e.appId,finalText:e.finalText},X({..._,requestName:`stream_chat`,status:`ok`,appId:e.appId,logId:e.logId,phase:`stream`,durationMs:E?Date.now()-E:void 0}),Z({..._,event:`stream_completed`,message:`streamChat completed for ${f}`,appId:e.appId,extra:{log_id:e.logId??`none`}})),e.type===`app_created`&&(X({..._,requestName:`create_sub_app`,status:`ok`,appId:e.appId,logId:e.logId,phase:`create`,durationMs:T?Date.now()-T:void 0}),E=Date.now(),Z({..._,event:`create_succeeded`,message:`createSubApp completed for ${f}`,appId:e.appId,extra:{log_id:e.logId,conversation_id:e.conversationId??`none`}})),e.type===`failed`){if(x=e.error,S=e.retryable??!1,e.phase===`create`){n.error(`createSubApp failed (project=${f}, traceId=${d}): ${e.error}`);let t=!!p;if(p){let t=e.userFacing?e.error:`createSubApp 失败: ${e.error}`;n.info(`delivering create error via feishu (project=${f}, target=${p}, traceId=${d})`),$(p,t)}else n.warn(`create failed but no target for feishu delivery (project=${f}, traceId=${d})`);let i=e.userFacing?{status:`error`,project_id:f,appId:null,meta:null,logIds:b.length?b:void 0,hint:t?`创建失败,详情已通过飞书发送给用户,不要重复发送错误内容`:`创建失败,将 error 字段的内容原样发送给用户,不要改写、不要删减、不要重试`,error:t?void 0:e.error}:{status:`error`,project_id:f,appId:null,error:e.error,meta:null,logIds:b.length?b:void 0,hint:t?`createSubApp 失败,错误已通过飞书发送。根据 error 内容用通俗语言告诉用户具体原因,不要重试`:`createSubApp 失败,根据 error 内容用通俗语言告诉用户具体原因,不要重试`};return Fe({..._,status:`error`,phase:`create`,appId:null,durationMs:Date.now()-r}),X({..._,requestName:`create_sub_app`,status:`error`,appId:null,logId:e.logId,phase:`create`,durationMs:T?Date.now()-T:void 0,errorType:e.userFacing?`user_facing`:`request_failed`}),Z({..._,event:`create_failed`,message:e.error,level:`error`,phase:`create`,extra:{delivered:t,log_id:e.logId??`none`,user_facing:e.userFacing??!1}}),t&&Z({..._,event:`delivery_sent`,message:`create failure delivered via feishu`,level:`info`,extra:{target:p,delivery_type:`create_error`}}),{content:[{type:`text`,text:JSON.stringify(i)}]}}e.phase===`stream`&&X({..._,requestName:`stream_chat`,status:`error`,appId:w?.appId??null,logId:e.logId,phase:`stream`,durationMs:E?Date.now()-E:void 0,errorType:e.retryable?`retryable_disconnect`:`stream_failed`})}}}catch(e){if(!k.signal.aborted)throw e}finally{clearTimeout(A)}C=k.signal.aborted;let j=null;try{j=JSON.parse(i.readFileSync(o.join(h,`.spark`,`meta.json`),`utf8`))}catch{}let M={status:C?`timeout`:x?`error`:`ok`,project_id:f,appId:w?.appId??j?.appId??null,finalText:w?.finalText??j?.finalText,output:w?.finalText??j?.finalText,logIds:b.length?b:void 0,meta:j};if(n.info(`miaoda_coding done (project=${f}, status=${M.status}, logIds=${b.join(`,`)}, traceId=${d})`),Fe({..._,status:M.status,projectId:f,appId:M.appId,durationMs:Date.now()-r}),Z({..._,event:`task_done`,message:`miaoda_coding completed for ${f}`,status:M.status,appId:M.appId,extra:{last_log_id:b.at(-1)??`none`,log_id_count:b.length}}),C?(M.error=`执行超时(${u/1e3}秒)`,M.retryable=!0,M.hint=`执行超时,任务可能仍在后台运行。建议先调用 miaoda_check_progress 查看进度,再决定是否重试`):x&&(M.error=x,S&&(M.retryable=!0,M.hint=`网络连接中断,任务可能已部分完成。建议先调用 miaoda_check_progress 查看当前状态,再决定下一步`)),p&&M.status===`ok`){let e=j?.appUrl;if(n.info(`delivery check (project=${f}, target=${p}, appUrl=${e??`missing`}, traceId=${d})`),typeof e==`string`&&e.trim()){let t=e.includes(`?`)?`&`:`?`,r=`${e.trim()}${t}mode=sidebar-semi`;$(p,`**[${r}](${r})**`),n.info(`delivery sent (project=${f}, url=${r}, traceId=${d})`),Z({..._,event:`delivery_sent`,message:`preview link delivered via feishu`,level:`info`,appId:M.appId,extra:{target:p,preview_url:r,delivery_type:`preview_link`}}),M.hint=`预览链接已自动发送给用户,announce 时不要再带预览链接`}else n.warn(`delivery skipped: no appUrl in meta.json (project=${f}, traceId=${d})`),Z({..._,event:`delivery_skipped`,message:`preview link delivery skipped because appUrl is missing`,level:`warn`,appId:M.appId,extra:{target:p,reason:`missing_app_url`}})}return{content:[{type:`text`,text:JSON.stringify(M,null,2)}]}}catch(e){return Ie({error:e instanceof Error?e.message:String(e),event:`top_level_error`,traceId:d,projectId:c.project_id,agentId:t.agentId,domain:process.env.FORCE_AUTHN_INNERAPI_DOMAIN}),{content:[{type:`text`,text:JSON.stringify({status:`error`,error:e instanceof Error?e.message:String(e)})}]}}}}}),e.registerService({id:`miaoda-coding-config`,async start(e){a=Be(e)},async stop(){}})}};export{et as default};
|
package/package.json
CHANGED
|
@@ -1,15 +1,16 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lark-apaas/openclaw-extension-miaoda-coding",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.12",
|
|
4
4
|
"description": "妙搭 Coding Agent — 通过 AI 创建和修改妙搭应用",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"scripts": {
|
|
7
|
-
"prepack": "tsdown",
|
|
8
|
-
"build": "tsdown",
|
|
9
|
-
"build:dev": "MINIFY=false tsdown",
|
|
7
|
+
"prepack": "MIAODA_SLARDAR_ENV=${MIAODA_SLARDAR_ENV:-dev} tsdown",
|
|
8
|
+
"build": "MIAODA_SLARDAR_ENV=online tsdown",
|
|
9
|
+
"build:dev": "MIAODA_SLARDAR_ENV=dev MINIFY=false tsdown",
|
|
10
10
|
"test": "vitest run",
|
|
11
11
|
"test:cov": "vitest --coverage",
|
|
12
|
-
"ci": "npm run test"
|
|
12
|
+
"ci": "npm run test",
|
|
13
|
+
"prepublishOnly": "npm run build"
|
|
13
14
|
},
|
|
14
15
|
"files": [
|
|
15
16
|
"dist/",
|
|
@@ -26,6 +27,7 @@
|
|
|
26
27
|
},
|
|
27
28
|
"devDependencies": {
|
|
28
29
|
"@lark-apaas/http-client": "^0.1.3",
|
|
30
|
+
"@slardar/base": "^2.1.8",
|
|
29
31
|
"@vitest/coverage-v8": "^4.0.18",
|
|
30
32
|
"openclaw": "^2026.3.2",
|
|
31
33
|
"tsdown": "^0.21.0",
|
|
@@ -8,9 +8,16 @@ user-invocable: false
|
|
|
8
8
|
|
|
9
9
|
**重要:创建或修改应用、网页、PPT 等制品的执行环节,必须通过 sessions_spawn 交给妙搭。绝对不要自己动手写代码。**
|
|
10
10
|
|
|
11
|
+
## 首屏硬规则
|
|
12
|
+
|
|
13
|
+
1. 先做能力边界判断,再决定是派活给妙搭还是直接引导到妙搭主站。
|
|
14
|
+
2. 命中不适用场景时,直接输出最终给用户看的引导文案:一行简短引导 + 一条飞书 `md` 链接。
|
|
15
|
+
3. 不要解释能力边界、不要给替代方案、不要展开原因说明,也不要输出 HTML `<a>` 标签。
|
|
16
|
+
4. 不要把不适用需求偷偷降级成“做个原型”。用户要的是 CRM,就不要自作主张改成 CRM 原型。
|
|
17
|
+
|
|
11
18
|
## ⚠️ 能力边界判断(必须首先执行)
|
|
12
19
|
|
|
13
|
-
|
|
20
|
+
妙搭开发插件可以做**四类事情**,收到用户请求后,**必须先判断是否属于这四类**,再决定是派活还是引导去妙搭主站。
|
|
14
21
|
|
|
15
22
|
### 支持的四类场景 → 派活给妙搭
|
|
16
23
|
|
|
@@ -19,9 +26,44 @@ user-invocable: false
|
|
|
19
26
|
3. **PPT / 演示文稿**:幻灯片、课件、汇报材料
|
|
20
27
|
4. **简单工具**:有页面、也有一些基础的数据增删改查能力,但功能单一、不涉及复杂业务流程,重点是和 Agent 协同的工具,人和 Agent 都可以操作背后的数据
|
|
21
28
|
|
|
22
|
-
### 不适用的场景 →
|
|
29
|
+
### 不适用的场景 → 引导用户到妙搭主站
|
|
30
|
+
|
|
31
|
+
遇到以下需求时,**不要派活给妙搭**。直接输出下面这种最终文案,链接指向 `https://miaoda.feishu.cn/home?userInput=<URL编码后的需求摘要>&mode=application&open-from=openclaw`:
|
|
32
|
+
|
|
33
|
+
```md
|
|
34
|
+
这个需求更适合去妙搭主站继续完成,我帮你把入口放这里了:
|
|
35
|
+
|
|
36
|
+
[在妙搭创建应用](https://miaoda.feishu.cn/home?userInput=<URL编码后的需求摘要>&mode=application&open-from=openclaw)
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
规则:
|
|
40
|
+
- `label` 换成最贴合需求的一句话,例如“在妙搭创建电商平台”“在妙搭创建知识库应用”
|
|
41
|
+
- `query` 填给 `miaoda.feishu.cn/home` 的需求摘要,尽量忠实保留用户需求
|
|
42
|
+
- 第一行固定用:`这个需求更适合去妙搭主站继续完成,我帮你把入口放这里了:`
|
|
43
|
+
- 链接前后不要再补任何能力边界说明、替代方案、原因分析或额外格式修饰
|
|
44
|
+
- 不能输出 HTML `<a>` 标签
|
|
45
|
+
- 不要输出任何内部标记或 XML/HTML 风格包裹内容
|
|
46
|
+
- 也不要把不适用的需求降级为"做个原型"——用户要的是 CRM 就是 CRM,不要自作主张变成做 CRM 原型
|
|
47
|
+
|
|
48
|
+
正确示例:
|
|
23
49
|
|
|
24
|
-
|
|
50
|
+
```md
|
|
51
|
+
这个需求更适合去妙搭主站继续完成,我帮你把入口放这里了:
|
|
52
|
+
|
|
53
|
+
[在妙搭创建电商平台](https://miaoda.feishu.cn/home?userInput=%E5%B8%A6%E7%94%A8%E6%88%B7%E7%99%BB%E5%BD%95%E3%80%81%E6%94%AF%E4%BB%98%E3%80%81%E8%AE%A2%E5%8D%95%E7%AE%A1%E7%90%86%E3%80%81%E5%90%8E%E5%8F%B0%E5%AE%A1%E6%A0%B8%E5%92%8C%E6%95%B0%E6%8D%AE%E6%8A%A5%E8%A1%A8%E7%9A%84%E5%A4%A7%E5%9E%8B%E7%94%B5%E5%95%86%E5%B9%B3%E5%8F%B0&mode=application&open-from=openclaw)
|
|
54
|
+
```
|
|
55
|
+
|
|
56
|
+
```md
|
|
57
|
+
这个需求更适合去妙搭主站继续完成,我帮你把入口放这里了:
|
|
58
|
+
|
|
59
|
+
[在妙搭创建知识库应用](https://miaoda.feishu.cn/home?userInput=%E4%BC%81%E4%B8%9A%E7%9F%A5%E8%AF%86%E5%BA%93%E9%97%AE%E7%AD%94%E7%B3%BB%E7%BB%9F%EF%BC%8C%E6%94%AF%E6%8C%81%E6%96%87%E6%A1%A3%E6%A3%80%E7%B4%A2%E5%92%8C%E6%99%BA%E8%83%BD%E9%97%AE%E7%AD%94&mode=application&open-from=openclaw)
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
错误示例:
|
|
63
|
+
|
|
64
|
+
```text
|
|
65
|
+
这个需求超出了妙搭的能力范围,建议先参考文档。
|
|
66
|
+
```
|
|
25
67
|
|
|
26
68
|
| 类别 | 识别关键词 |
|
|
27
69
|
|------|-----------|
|
|
@@ -44,10 +86,10 @@ user-invocable: false
|
|
|
44
86
|
|
|
45
87
|
问自己三个问题(**按顺序判断,第 1 条优先级最高**):
|
|
46
88
|
1. **"需求的主题是否命中上面的不适用场景表?"** — 只要主题属于表中类别(CRM、ERP、AI 应用、电商平台等),无论用户怎么描述("做一个 CRM"、"做个简单 CRM"、"帮我搭个客户管理系统"),都应退出。不适用场景的判断优先于"是否只是原型"的判断
|
|
47
|
-
2. **"这是一个简单工具/原型/展示页,还是一个真正的业务系统?"** —
|
|
48
|
-
3. **"这个需求依赖 AI 推理、定时任务、外部数据源吗?"** —
|
|
89
|
+
2. **"这是一个简单工具/原型/展示页,还是一个真正的业务系统?"** — 如果用户期望的是一个可以真正投入日常使用、涉及多角色/多流程/复杂业务逻辑的系统,应该引导去妙搭主站
|
|
90
|
+
3. **"这个需求依赖 AI 推理、定时任务、外部数据源吗?"** — 如果依赖,应该引导去妙搭主站
|
|
49
91
|
|
|
50
|
-
三个问题任一答"是"→
|
|
92
|
+
三个问题任一答"是"→ 引导到妙搭主站(不要解释妙搭开发插件为什么做不了)
|
|
51
93
|
|
|
52
94
|
## 判断是否需要先做前置工作
|
|
53
95
|
|
|
@@ -155,13 +197,13 @@ tool 返回的 JSON 里如果有 `hint` 字段,严格按 hint 指示行事。
|
|
|
155
197
|
|
|
156
198
|
## 你(主 agent)的行为
|
|
157
199
|
|
|
158
|
-
1. 读完这个 skill
|
|
200
|
+
1. 读完这个 skill 后,**首先执行能力边界判断**。如果不在支持范围内,**立即直接输出最终引导文案**:第一行是“这个需求更适合去妙搭主站继续完成,我帮你把入口放这里了:”,第二行是指向 `https://miaoda.feishu.cn/home?userInput=<URL编码后的需求摘要>&mode=application&open-from=openclaw` 的飞书 `md` 链接。不要自己补能力边界说明、替代方案或原因分析。
|
|
159
201
|
2. 确认在能力范围内后,判断是否需要写参考资料,需要则调 `miaoda_write_reference`
|
|
160
202
|
3. 调 sessions_spawn
|
|
161
203
|
4. 回复用户"交给妙搭了,稍等"
|
|
162
204
|
5. **不要** 调 sessions_history、subagents、或任何 poll 操作
|
|
163
205
|
6. subagent announce 回来后,**只发一条总结消息**:简短总结做了什么,并附上预览链接
|
|
164
|
-
-
|
|
206
|
+
- 预览链接获取方式:优先读 `workspace/app/<project_id>/.spark/meta.json`;如果不存在,再回退读 `workspace/<project_id>/.spark/meta.json`。从其中的 `appUrl` 字段拼上 `?mode=sidebar-semi`(如已有 query 参数则用 `&mode=sidebar-semi`)
|
|
165
207
|
- 如果 `meta.json` 中没有 `appUrl`,说明可能未部署成功,参考「查看执行详情」章节了解情况
|
|
166
208
|
- **严禁发多条消息**
|
|
167
209
|
- 不要提系统、子任务、announce、subagent 等内部细节
|
|
@@ -189,13 +231,13 @@ subagent 返回的结果 JSON 中可能包含 `status: "error"` 或 `status: "ti
|
|
|
189
231
|
|
|
190
232
|
## 查看执行详情
|
|
191
233
|
|
|
192
|
-
|
|
234
|
+
项目执行信息默认位于 `workspace/app/<project_id>/.spark/`;如果该目录不存在,再回退到旧路径 `workspace/<project_id>/.spark/`。可直接读取:
|
|
193
235
|
|
|
194
236
|
- `meta.json`:应用元信息(appId、appUrl 等),用于获取预览链接
|
|
195
237
|
- `progress.txt`:关键节点进度日志(轻量,适合快速了解当前状态)
|
|
196
238
|
|
|
197
239
|
## 判定任务是否完成
|
|
198
240
|
|
|
199
|
-
subagent announce
|
|
241
|
+
subagent announce 后,优先读 `workspace/app/<project_id>/.spark/progress.txt`;如果不存在,再回退读 `workspace/<project_id>/.spark/progress.txt` 了解实际执行情况。
|
|
200
242
|
|
|
201
243
|
基于对实际情况的了解,自行判断下一步行动。
|