@rethinkingstudio/clawpilot 1.1.15-beta.1 → 1.1.15-beta.2
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/commands/pair.js +1 -1
- package/dist/commands/pair.js.map +1 -1
- package/dist/generated/build-config.d.ts +1 -0
- package/dist/generated/build-config.js +2 -0
- package/dist/generated/build-config.js.map +1 -0
- package/dist/index.js +2 -1
- package/dist/index.js.map +1 -1
- package/dist/media/assistant-attachments.d.ts +35 -0
- package/dist/media/assistant-attachments.js +144 -0
- package/dist/media/assistant-attachments.js.map +1 -0
- package/dist/media/oss-uploader.d.ts +22 -0
- package/dist/media/oss-uploader.js +180 -0
- package/dist/media/oss-uploader.js.map +1 -0
- package/dist/platform/service-manager.js +10 -7
- package/dist/platform/service-manager.js.map +1 -1
- package/dist/relay/relay-manager.js +16 -4
- package/dist/relay/relay-manager.js.map +1 -1
- package/package.json +6 -2
- package/scripts/verify-build-config.mjs +35 -0
- package/scripts/write-build-config.mjs +17 -0
- package/src/commands/pair.ts +1 -2
- package/src/generated/build-config.ts +1 -0
- package/src/index.ts +2 -1
- package/src/media/assistant-attachments.ts +205 -0
- package/src/media/oss-uploader.ts +280 -0
- package/src/platform/service-manager.ts +10 -6
- package/src/relay/relay-manager.ts +24 -5
- package/dist/relay/session-proxy.d.ts +0 -18
- package/dist/relay/session-proxy.js +0 -75
- package/dist/relay/session-proxy.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"relay-manager.js","sourceRoot":"","sources":["../../src/relay/relay-manager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AAC/B,OAAO,EAAE,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AAC5D,OAAO,EAAE,kBAAkB,EAAE,MAAM,+BAA+B,CAAC;AACnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,kCAAkC,CAAC;AACzE,OAAO,EAAE,kBAAkB,EAAE,MAAM,gCAAgC,CAAC;AACpE,OAAO,EAAE,OAAO,EAAE,MAAM,IAAI,CAAC;AAC7B,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AAEpC,8EAA8E;AAC9E,YAAY;AACZ,8EAA8E;AAE9E,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,WAAW,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;AAqCvE,8EAA8E;AAC9E,mBAAmB;AACnB,8EAA8E;AAE9E;;;;;;;GAOG;AACH,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,IAAyB;IAC7D,MAAM,KAAK,GAAG,aAAa,CAAC,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;IAEnF,OAAO,IAAI,OAAO,CAAU,CAAC,OAAO,EAAE,EAAE;QACtC,IAAI,OAAkB,CAAC;QACvB,IAAI,CAAC;YACH,OAAO,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,CAAC,KAAK,CAAC,mCAAmC,EAAE,GAAG,CAAC,CAAC;YACxD,OAAO,CAAC,IAAI,CAAC,CAAC;YACd,OAAO;QACT,CAAC;QAED,IAAI,aAAa,GAAiC,IAAI,CAAC;QAEvD,SAAS,IAAI,CAAC,GAAa;YACzB,IAAI,OAAO,CAAC,UAAU,KAAK,SAAS,CAAC,IAAI,EAAE,CAAC;gBAC1C,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC;YACpC,CAAC;QACH,CAAC;QAED,OAAO,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE;YACtB,OAAO,CAAC,GAAG,CAAC,wCAAwC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC;YACvE,IAAI,CAAC,EAAE,IAAI,EAAE,aAAa,EAAE,QAAQ,EAAE,kBAAkB,EAAE,EAAE,CAAC,CAAC;YAC9D,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;YAErB,qEAAqE;YACrE,mEAAmE;YACnE,aAAa,GAAG,IAAI,qBAAqB,CAAC;gBACxC,GAAG,EAAE,IAAI,CAAC,UAAU;gBACpB,KAAK,EAAE,IAAI,CAAC,YAAY;gBACxB,QAAQ,EAAE,IAAI,CAAC,eAAe;gBAE9B,WAAW,EAAE,GAAG,EAAE;oBAChB,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;oBAClC,IAAI,CAAC,EAAE,IAAI,EAAE,mBAAmB,EAAE,CAAC,CAAC;gBACtC,CAAC;gBAED,cAAc,EAAE,CAAC,MAAM,EAAE,EAAE;oBACzB,OAAO,CAAC,GAAG,CAAC,yBAAyB,MAAM,EAAE,CAAC,CAAC;oBAC/C,IAAI,CAAC,EAAE,IAAI,EAAE,sBAAsB,EAAE,MAAM,EAAE,CAAC,CAAC;gBACjD,CAAC;gBAED,OAAO,EAAE,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;oBAC1B,
|
|
1
|
+
{"version":3,"file":"relay-manager.js","sourceRoot":"","sources":["../../src/relay/relay-manager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AAC/B,OAAO,EAAE,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AAC5D,OAAO,EAAE,kBAAkB,EAAE,MAAM,+BAA+B,CAAC;AACnE,OAAO,EAAE,qBAAqB,EAAE,MAAM,kCAAkC,CAAC;AACzE,OAAO,EAAE,kBAAkB,EAAE,MAAM,gCAAgC,CAAC;AACpE,OAAO,EAAE,0BAA0B,EAAE,MAAM,mCAAmC,CAAC;AAC/E,OAAO,EAAE,OAAO,EAAE,MAAM,IAAI,CAAC;AAC7B,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AAEpC,8EAA8E;AAC9E,YAAY;AACZ,8EAA8E;AAE9E,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,WAAW,EAAE,OAAO,EAAE,UAAU,CAAC,CAAC;AAqCvE,8EAA8E;AAC9E,mBAAmB;AACnB,8EAA8E;AAE9E;;;;;;;GAOG;AACH,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,IAAyB;IAC7D,MAAM,KAAK,GAAG,aAAa,CAAC,IAAI,CAAC,cAAc,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;IAEnF,OAAO,IAAI,OAAO,CAAU,CAAC,OAAO,EAAE,EAAE;QACtC,IAAI,OAAkB,CAAC;QACvB,IAAI,CAAC;YACH,OAAO,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,CAAC,KAAK,CAAC,mCAAmC,EAAE,GAAG,CAAC,CAAC;YACxD,OAAO,CAAC,IAAI,CAAC,CAAC;YACd,OAAO;QACT,CAAC;QAED,IAAI,aAAa,GAAiC,IAAI,CAAC;QAEvD,SAAS,IAAI,CAAC,GAAa;YACzB,IAAI,OAAO,CAAC,UAAU,KAAK,SAAS,CAAC,IAAI,EAAE,CAAC;gBAC1C,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC;YACpC,CAAC;QACH,CAAC;QAED,OAAO,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE;YACtB,OAAO,CAAC,GAAG,CAAC,wCAAwC,IAAI,CAAC,SAAS,GAAG,CAAC,CAAC;YACvE,IAAI,CAAC,EAAE,IAAI,EAAE,aAAa,EAAE,QAAQ,EAAE,kBAAkB,EAAE,EAAE,CAAC,CAAC;YAC9D,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;YAErB,qEAAqE;YACrE,mEAAmE;YACnE,aAAa,GAAG,IAAI,qBAAqB,CAAC;gBACxC,GAAG,EAAE,IAAI,CAAC,UAAU;gBACpB,KAAK,EAAE,IAAI,CAAC,YAAY;gBACxB,QAAQ,EAAE,IAAI,CAAC,eAAe;gBAE9B,WAAW,EAAE,GAAG,EAAE;oBAChB,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;oBAClC,IAAI,CAAC,EAAE,IAAI,EAAE,mBAAmB,EAAE,CAAC,CAAC;gBACtC,CAAC;gBAED,cAAc,EAAE,CAAC,MAAM,EAAE,EAAE;oBACzB,OAAO,CAAC,GAAG,CAAC,yBAAyB,MAAM,EAAE,CAAC,CAAC;oBAC/C,IAAI,CAAC,EAAE,IAAI,EAAE,sBAAsB,EAAE,MAAM,EAAE,CAAC,CAAC;gBACjD,CAAC;gBAED,OAAO,EAAE,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;oBAC1B,kFAAkF;oBAClF,+EAA+E;oBAC/E,IAAI,KAAK,KAAK,MAAM,EAAE,CAAC;wBACrB,MAAM,CAAC,GAAG,OAAqF,CAAC;wBAChG,IAAI,CAAC,EAAE,KAAK,KAAK,OAAO,IAAI,CAAC,EAAE,UAAU,EAAE,CAAC;4BAC1C,MAAM,UAAU,GAAG,CAAC,CAAC,UAAU,CAAC;4BAChC,MAAM,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC;4BAGtB,MAAM,YAAY,GAAG,GAAG,EAAE,CACxB,aAAc,CAAC,OAAO,CAAkB,cAAc,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;4BACrF,MAAM,WAAW,GAAG,CAAC,CAA8B,EAAE,EAAE;gCACrD,MAAM,IAAI,GAAG,CAAC,EAAE,QAAQ,IAAI,EAAE,CAAC;gCAC/B,MAAM,IAAI,GAAG,CAAC,GAAG,IAAI,CAAC,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,WAAW,CAAC,CAAC;gCACrE,OAAO,IAAI,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,MAAM,CAAC,EAAE,IAAI,CAAC;4BAC7D,CAAC,CAAC;4BACF,YAAY,EAAE;iCACX,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;gCACtB,IAAI,IAAI,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;gCAChC,sEAAsE;gCACtE,IAAI,CAAC,IAAI,EAAE,CAAC;oCACV,MAAM,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,CAAC;oCACzD,MAAM,YAAY,GAAG,MAAM,YAAY,EAAE,CAAC;oCAC1C,IAAI,GAAG,WAAW,CAAC,YAAY,CAAC,CAAC;oCACjC,OAAO,GAAG,YAAY,CAAC;gCACzB,CAAC;gCACD,IAAI,IAAI,EAAE,CAAC;oCACR,CAA6B,CAAC,OAAO,GAAG,EAAE,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC;gCACjF,CAAC;gCAED,+CAA+C;gCAC/C,IAAI,CAAC;oCACH,MAAM,WAAW,GAAG,MAAM,0BAA0B,CAClD,OAAO,IAAI,EAAE,EACb,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,WAAW,CACjB,CAAC;oCACF,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;wCAC1B,CAA6B,CAAC,WAAW,GAAG,WAAW,CAAC;wCACzD,OAAO,CAAC,GAAG,CAAC,gCAAgC,WAAW,CAAC,MAAM,wBAAwB,KAAK,EAAE,CAAC,CAAC;oCACjG,CAAC;gCACH,CAAC;gCAAC,OAAO,QAAQ,EAAE,CAAC;oCAClB,OAAO,CAAC,KAAK,CAAC,2CAA2C,QAAQ,EAAE,CAAC,CAAC;gCACvE,CAAC;gCAED,OAAO,CAAC,GAAG,CAAC,+CAA+C,KAAK,eAAe,IAAI,EAAE,MAAM,IAAI,CAAC,EAAE,CAAC,CAAC;gCACpG,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;4BAC1C,CAAC,CAAC;iCACD,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE;gCACb,OAAO,CAAC,KAAK,CAAC,sCAAsC,GAAG,EAAE,CAAC,CAAC;gCAC3D,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;4BAC1C,CAAC,CAAC,CAAC;4BACL,OAAO,CAAC,+CAA+C;wBACzD,CAAC;oBACH,CAAC;oBACD,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;gBAC1C,CAAC;aACF,CAAC,CAAC;YAEH,aAAa,CAAC,KAAK,EAAE,CAAC;QACxB,CAAC,CAAC,CAAC;QAEH,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,KAAK,EAAE,GAAG,EAAE,EAAE;YAClC,IAAI,GAAe,CAAC;YACpB,IAAI,CAAC;gBACH,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAe,CAAC;YACjD,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO;YACT,CAAC;YAED,IAAI,GAAG,CAAC,IAAI,KAAK,KAAK,IAAI,CAAC,GAAG,CAAC,MAAM;gBAAE,OAAO;YAE9C,MAAM,SAAS,GAAG,GAAG,CAAC,EAAE,CAAC;YACzB,OAAO,CAAC,GAAG,CAAC,+BAA+B,GAAG,CAAC,MAAM,OAAO,SAAS,IAAI,SAAS,EAAE,CAAC,CAAC;YAEtF,uDAAuD;YACvD,MAAM,eAAe,GAAG,qBAAqB,CAAC,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;YACtE,IAAI,eAAe,KAAK,IAAI,EAAE,CAAC;gBAC7B,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC;gBACrC,IAAI,SAAS,EAAE,CAAC;oBACd,IAAI,CAAC;wBACH,IAAI,EAAE,KAAK;wBACX,EAAE,EAAE,SAAS;wBACb,EAAE,EAAE,MAAM,CAAC,EAAE;wBACb,GAAG,CAAC,MAAM,CAAC,EAAE;4BACX,CAAC,CAAC,EAAE,OAAO,EAAE,MAAM,CAAC,OAAO,EAAE;4BAC7B,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,OAAO,EAAE,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;qBAC1C,CAAC,CAAC;gBACL,CAAC;gBACD,OAAO;YACT,CAAC;YAED,wEAAwE;YACxE,MAAM,WAAW,GAAG,kBAAkB,CAAC,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;YAC/D,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;gBACzB,IAAI,SAAS,EAAE,CAAC;oBACd,IAAI,WAAW,CAAC,EAAE,EAAE,CAAC;wBACnB,IAAI,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,WAAW,CAAC,OAAO,EAAE,CAAC,CAAC;oBAC/E,CAAC;yBAAM,CAAC;wBACN,IAAI,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,OAAO,EAAE,WAAW,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;oBACzF,CAAC;gBACH,CAAC;gBACD,OAAO;YACT,CAAC;YAED,0EAA0E;YAC1E,IAAI,GAAG,CAAC,MAAM,KAAK,WAAW,EAAE,CAAC;gBAC/B,MAAM,MAAM,GAAG,GAAG,CAAC,MAAa,CAAC;gBACjC,qFAAqF;gBACrF,MAAM,CAAC,OAAO,GAAG,KAAK,CAAC;gBACvB,IAAI,MAAM,CAAC,WAAW,IAAI,MAAM,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACxD,MAAM,cAAc,GAAa,EAAE,CAAC;oBAEpC,mCAAmC;oBACnC,MAAM,KAAK,CAAC,YAAY,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;oBAE/C,yDAAyD;oBACzD,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,WAAW,EAAE,CAAC;wBACrC,IAAI,CAAC;4BACH,0BAA0B;4BAC1B,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;4BAClD,MAAM,GAAG,GAAG,GAAG,CAAC,QAAQ,KAAK,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;4BAC3D,MAAM,cAAc,GAAG,GAAG,UAAU,EAAE,GAAG,GAAG,EAAE,CAAC;4BAC/C,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,EAAE,cAAc,CAAC,CAAC;4BAEtD,gBAAgB;4BAChB,MAAM,SAAS,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;4BACpC,OAAO,CAAC,GAAG,CAAC,gCAAgC,UAAU,EAAE,CAAC,CAAC;4BAE1D,+CAA+C;4BAC/C,cAAc,CAAC,IAAI,CACjB,oBAAoB,UAAU,KAAK,GAAG,CAAC,QAAQ,OAAO,UAAU,GAAG,CACpE,CAAC;wBACJ,CAAC;wBAAC,OAAO,GAAG,EAAE,CAAC;4BACb,OAAO,CAAC,KAAK,CAAC,sCAAsC,GAAG,EAAE,CAAC,CAAC;wBAC7D,CAAC;oBACH,CAAC;oBAED,oCAAoC;oBACpC,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;wBAC9B,MAAM,IAAI,GAAG,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;wBACvC,MAAM,CAAC,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,OAAO,OAAO,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;wBACxE,OAAO,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC;oBAC1D,CAAC;gBACH,CAAC;YACH,CAAC;YAED,aAAa;gBACX,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC;iBAChC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;gBACf,OAAO,CAAC,GAAG,CAAC,yBAAyB,GAAG,CAAC,MAAM,OAAO,SAAS,IAAI,SAAS,EAAE,CAAC,CAAC;gBAChF,IAAI,SAAS,EAAE,CAAC;oBACd,IAAI,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;gBAClE,CAAC;YACH,CAAC,CAAC;iBACD,KAAK,CAAC,CAAC,GAAY,EAAE,EAAE;gBACtB,OAAO,CAAC,KAAK,CAAC,6BAA6B,GAAG,CAAC,MAAM,OAAO,SAAS,IAAI,SAAS,KAAK,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBACtG,IAAI,SAAS,EAAE,CAAC;oBACd,IAAI,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,OAAO,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,CAAC;gBACnF,CAAC;YACH,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE;YACnC,OAAO,CAAC,GAAG,CAAC,4BAA4B,IAAI,IAAI,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;YACrE,IAAI,CAAC,cAAc,EAAE,EAAE,CAAC;YACxB,aAAa,EAAE,IAAI,EAAE,CAAC;YACtB,aAAa,GAAG,IAAI,CAAC;YACrB,uEAAuE;YACvE,sEAAsE;YACtE,OAAO,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC;QACzB,CAAC,CAAC,CAAC;QAEH,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YAC1B,OAAO,CAAC,KAAK,CAAC,wBAAwB,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC;YACrD,0BAA0B;QAC5B,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,aAAa,CAAC,SAAiB,EAAE,SAAiB,EAAE,WAAmB;IAC9E,MAAM,IAAI,GAAG,SAAS,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAClE,OAAO,GAAG,IAAI,UAAU,SAAS,WAAW,kBAAkB,CAAC,WAAW,CAAC,EAAE,CAAC;AAChF,CAAC"}
|
package/package.json
CHANGED
|
@@ -1,13 +1,17 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@rethinkingstudio/clawpilot",
|
|
3
|
-
"version": "1.1.15-beta.
|
|
3
|
+
"version": "1.1.15-beta.2",
|
|
4
4
|
"description": "ClawAI relay client for Mac mini",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"bin": {
|
|
7
7
|
"clawpilot": "dist/index.js"
|
|
8
8
|
},
|
|
9
9
|
"scripts": {
|
|
10
|
-
"build": "
|
|
10
|
+
"build:config": "node ./scripts/write-build-config.mjs",
|
|
11
|
+
"build": "npm run build:config && tsc",
|
|
12
|
+
"build:prod": "npm run build",
|
|
13
|
+
"build:internal": "CLAWPILOT_DEFAULT_RELAY_SERVER=https://clawpilot-pre.codeaddict.cn npm run build",
|
|
14
|
+
"prepublishOnly": "node ./scripts/verify-build-config.mjs",
|
|
11
15
|
"dev": "tsx src/index.ts",
|
|
12
16
|
"start": "node dist/index.js"
|
|
13
17
|
},
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { readFileSync } from "fs";
|
|
2
|
+
import { resolve } from "path";
|
|
3
|
+
|
|
4
|
+
const workspace = process.cwd();
|
|
5
|
+
const packageJsonPath = resolve(workspace, "package.json");
|
|
6
|
+
const buildConfigPath = resolve(workspace, "src/generated/build-config.ts");
|
|
7
|
+
|
|
8
|
+
const PROD_SERVER = "https://clawpilot.codeaddict.cn";
|
|
9
|
+
const PRE_SERVER = "https://clawpilot-pre.codeaddict.cn";
|
|
10
|
+
|
|
11
|
+
const pkg = JSON.parse(readFileSync(packageJsonPath, "utf8"));
|
|
12
|
+
const buildConfig = readFileSync(buildConfigPath, "utf8");
|
|
13
|
+
const match = buildConfig.match(/DEFAULT_RELAY_SERVER\s*=\s*"([^"]+)"/);
|
|
14
|
+
|
|
15
|
+
if (!match) {
|
|
16
|
+
console.error(`[verify-build-config] Could not read DEFAULT_RELAY_SERVER from ${buildConfigPath}`);
|
|
17
|
+
process.exit(1);
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
const actualServer = match[1];
|
|
21
|
+
const publishTag = (process.env.npm_config_tag ?? "").trim();
|
|
22
|
+
const version = String(pkg.version ?? "");
|
|
23
|
+
const isInternalRelease = /\binternal\b/i.test(version) || publishTag === "internal";
|
|
24
|
+
const expectedServer = isInternalRelease ? PRE_SERVER : PROD_SERVER;
|
|
25
|
+
|
|
26
|
+
if (actualServer !== expectedServer) {
|
|
27
|
+
console.error(
|
|
28
|
+
`[verify-build-config] Refusing to publish version=${version} tag=${publishTag || "latest"} with DEFAULT_RELAY_SERVER=${actualServer}. Expected ${expectedServer}.`
|
|
29
|
+
);
|
|
30
|
+
process.exit(1);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
console.log(
|
|
34
|
+
`[verify-build-config] OK version=${version} tag=${publishTag || "latest"} DEFAULT_RELAY_SERVER=${actualServer}`
|
|
35
|
+
);
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { mkdirSync, writeFileSync } from "fs";
|
|
2
|
+
import { dirname, resolve } from "path";
|
|
3
|
+
|
|
4
|
+
const target = resolve(process.cwd(), "src/generated/build-config.ts");
|
|
5
|
+
const relayServer =
|
|
6
|
+
process.env.CLAWPILOT_DEFAULT_RELAY_SERVER?.trim()
|
|
7
|
+
|| "https://clawpilot.codeaddict.cn";
|
|
8
|
+
|
|
9
|
+
mkdirSync(dirname(target), { recursive: true });
|
|
10
|
+
|
|
11
|
+
writeFileSync(
|
|
12
|
+
target,
|
|
13
|
+
`export const DEFAULT_RELAY_SERVER = ${JSON.stringify(relayServer)};\n`,
|
|
14
|
+
"utf8"
|
|
15
|
+
);
|
|
16
|
+
|
|
17
|
+
console.log(`[build-config] DEFAULT_RELAY_SERVER=${relayServer}`);
|
package/src/commands/pair.ts
CHANGED
|
@@ -26,8 +26,7 @@ import { configExists, readConfig, writeConfig } from "../config/config.js";
|
|
|
26
26
|
import { installCommand } from "./install.js";
|
|
27
27
|
import qrcodeTerminal from "qrcode-terminal";
|
|
28
28
|
import { t } from "../i18n/index.js";
|
|
29
|
-
|
|
30
|
-
const DEFAULT_RELAY_SERVER = "https://clawpilot.codeaddict.cn";
|
|
29
|
+
import { DEFAULT_RELAY_SERVER } from "../generated/build-config.js";
|
|
31
30
|
|
|
32
31
|
interface PairOptions {
|
|
33
32
|
server?: string;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export const DEFAULT_RELAY_SERVER = "https://clawpilot.codeaddict.cn";
|
package/src/index.ts
CHANGED
|
@@ -6,6 +6,7 @@ import { runCommand } from "./commands/run.js";
|
|
|
6
6
|
import { installCommand, uninstallCommand, stopCommand, restartCommand, resetCommand } from "./commands/install.js";
|
|
7
7
|
import { statusCommand } from "./commands/status.js";
|
|
8
8
|
import { setTokenCommand } from "./commands/set-token.js";
|
|
9
|
+
import { DEFAULT_RELAY_SERVER } from "./generated/build-config.js";
|
|
9
10
|
|
|
10
11
|
const require = createRequire(import.meta.url);
|
|
11
12
|
const { version } = require("../package.json") as { version: string };
|
|
@@ -20,7 +21,7 @@ program
|
|
|
20
21
|
program
|
|
21
22
|
.command("pair")
|
|
22
23
|
.description("Register with relay server and display QR code for iOS pairing")
|
|
23
|
-
.option("-s, --server <url>", "Relay server URL",
|
|
24
|
+
.option("-s, --server <url>", "Relay server URL", DEFAULT_RELAY_SERVER)
|
|
24
25
|
.option("-n, --name <name>", "Display name for this host")
|
|
25
26
|
.option("--code-only", "Print only the access code and skip QR code output", false)
|
|
26
27
|
.action(async (opts: { server: string; name: string; codeOnly?: boolean }) => {
|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
// ---------------------------------------------------------------------------
|
|
2
|
+
// Extract media (image / video) blocks from an OpenClaw gateway chat history
|
|
3
|
+
// response, and upload them to OSS via STS credentials from the relay server.
|
|
4
|
+
// ---------------------------------------------------------------------------
|
|
5
|
+
|
|
6
|
+
import { uploadMedia, type StsCredentials, type UploadResult } from "./oss-uploader.js";
|
|
7
|
+
import { readFile } from "fs/promises";
|
|
8
|
+
|
|
9
|
+
// Max video size accepted for upload (200 MB)
|
|
10
|
+
const MAX_VIDEO_BYTES = 200 * 1024 * 1024;
|
|
11
|
+
|
|
12
|
+
export interface MediaBlock {
|
|
13
|
+
mimeType: string;
|
|
14
|
+
/** Buffer content (decoded from base64 or read from local path) */
|
|
15
|
+
data: Buffer;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export interface AssistantAttachment extends UploadResult {
|
|
19
|
+
width?: number;
|
|
20
|
+
height?: number;
|
|
21
|
+
durationMs?: number;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// ---------------------------------------------------------------------------
|
|
25
|
+
// Extract media blocks from the last assistant message in a history response
|
|
26
|
+
// ---------------------------------------------------------------------------
|
|
27
|
+
|
|
28
|
+
type ContentBlock = {
|
|
29
|
+
type: string;
|
|
30
|
+
text?: string;
|
|
31
|
+
source?: {
|
|
32
|
+
type?: string; // "base64" | "url" | "file" | null
|
|
33
|
+
media_type?: string; // "image/png" etc.
|
|
34
|
+
data?: string; // base64 payload
|
|
35
|
+
url?: string; // external URL (not handled in Phase 1)
|
|
36
|
+
path?: string; // local file path
|
|
37
|
+
};
|
|
38
|
+
// Some gateways embed media_type / data directly on the block
|
|
39
|
+
media_type?: string;
|
|
40
|
+
data?: string;
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
type HistoryMessage = {
|
|
44
|
+
role: string;
|
|
45
|
+
content?: ContentBlock[] | string;
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
export function extractMediaBlocks(history: { messages?: HistoryMessage[] }): MediaBlock[] {
|
|
49
|
+
const msgs = history.messages ?? [];
|
|
50
|
+
const last = [...msgs].reverse().find((m) => m.role === "assistant");
|
|
51
|
+
if (!last || !last.content) return [];
|
|
52
|
+
|
|
53
|
+
const content: ContentBlock[] =
|
|
54
|
+
typeof last.content === "string" ? [] : last.content;
|
|
55
|
+
|
|
56
|
+
const blocks: MediaBlock[] = [];
|
|
57
|
+
|
|
58
|
+
for (const block of content) {
|
|
59
|
+
if (block.type !== "image" && block.type !== "video") continue;
|
|
60
|
+
|
|
61
|
+
const source = block.source;
|
|
62
|
+
const mimeType =
|
|
63
|
+
source?.media_type ?? block.media_type ?? "image/jpeg";
|
|
64
|
+
|
|
65
|
+
// Only handle base64 and local file paths in Phase 1
|
|
66
|
+
if (source?.type === "base64" || (!source?.type && source?.data)) {
|
|
67
|
+
const raw = source?.data ?? block.data ?? "";
|
|
68
|
+
if (!raw) continue;
|
|
69
|
+
try {
|
|
70
|
+
const buf = Buffer.from(raw.replace(/^data:[^;]+;base64,/, ""), "base64");
|
|
71
|
+
blocks.push({ mimeType, data: buf });
|
|
72
|
+
} catch {
|
|
73
|
+
console.warn("[media] failed to decode base64 block, skipping");
|
|
74
|
+
}
|
|
75
|
+
} else if (source?.type === "file" && source.path) {
|
|
76
|
+
// Local file reference — will be read asynchronously below
|
|
77
|
+
blocks.push({ mimeType, data: Buffer.alloc(0), _localPath: source.path } as MediaBlock & { _localPath: string });
|
|
78
|
+
}
|
|
79
|
+
// URL type is not handled in Phase 1
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
return blocks;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// ---------------------------------------------------------------------------
|
|
86
|
+
// Load local-path blocks and filter oversized videos
|
|
87
|
+
// ---------------------------------------------------------------------------
|
|
88
|
+
|
|
89
|
+
async function resolveLocalPaths(blocks: (MediaBlock & { _localPath?: string })[]): Promise<MediaBlock[]> {
|
|
90
|
+
const resolved: MediaBlock[] = [];
|
|
91
|
+
for (const b of blocks) {
|
|
92
|
+
if ((b as any)._localPath) {
|
|
93
|
+
try {
|
|
94
|
+
const buf = await readFile((b as any)._localPath as string);
|
|
95
|
+
resolved.push({ mimeType: b.mimeType, data: buf });
|
|
96
|
+
} catch (err) {
|
|
97
|
+
console.warn(`[media] failed to read local file ${(b as any)._localPath}: ${err}`);
|
|
98
|
+
}
|
|
99
|
+
} else {
|
|
100
|
+
resolved.push(b);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
return resolved.filter((b) => {
|
|
104
|
+
if (b.mimeType.startsWith("video/") && b.data.length > MAX_VIDEO_BYTES) {
|
|
105
|
+
console.warn(`[media] video too large (${b.data.length} bytes), skipping`);
|
|
106
|
+
return false;
|
|
107
|
+
}
|
|
108
|
+
return b.data.length > 0;
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// ---------------------------------------------------------------------------
|
|
113
|
+
// Fetch STS credentials from the relay server
|
|
114
|
+
// ---------------------------------------------------------------------------
|
|
115
|
+
|
|
116
|
+
async function fetchSts(
|
|
117
|
+
relayServerUrl: string,
|
|
118
|
+
gatewayId: string,
|
|
119
|
+
relaySecret: string,
|
|
120
|
+
mimeTypes: string[],
|
|
121
|
+
hasVideo: boolean
|
|
122
|
+
): Promise<StsCredentials> {
|
|
123
|
+
const url = `${relayServerUrl.replace(/\/$/, "")}/api/media/sts`;
|
|
124
|
+
const res = await fetch(url, {
|
|
125
|
+
method: "POST",
|
|
126
|
+
headers: { "Content-Type": "application/json" },
|
|
127
|
+
body: JSON.stringify({
|
|
128
|
+
gatewayId,
|
|
129
|
+
relaySecret,
|
|
130
|
+
count: mimeTypes.length,
|
|
131
|
+
mimeTypes,
|
|
132
|
+
durationSeconds: hasVideo ? 1800 : 600,
|
|
133
|
+
}),
|
|
134
|
+
});
|
|
135
|
+
if (!res.ok) {
|
|
136
|
+
const text = await res.text().catch(() => "");
|
|
137
|
+
throw new Error(`STS fetch failed: ${res.status} ${text}`);
|
|
138
|
+
}
|
|
139
|
+
return res.json() as Promise<StsCredentials>;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
// ---------------------------------------------------------------------------
|
|
143
|
+
// Main: extract + upload → return attachments[]
|
|
144
|
+
// ---------------------------------------------------------------------------
|
|
145
|
+
|
|
146
|
+
export async function uploadAssistantAttachments(
|
|
147
|
+
history: { messages?: HistoryMessage[] },
|
|
148
|
+
relayServerUrl: string,
|
|
149
|
+
gatewayId: string,
|
|
150
|
+
relaySecret: string
|
|
151
|
+
): Promise<AssistantAttachment[]> {
|
|
152
|
+
const rawBlocks = extractMediaBlocks(history);
|
|
153
|
+
if (rawBlocks.length === 0) return [];
|
|
154
|
+
|
|
155
|
+
const blocks = await resolveLocalPaths(rawBlocks as (MediaBlock & { _localPath?: string })[]);
|
|
156
|
+
if (blocks.length === 0) return [];
|
|
157
|
+
|
|
158
|
+
const mimeTypes = blocks.map((b) => b.mimeType);
|
|
159
|
+
const hasVideo = mimeTypes.some((m) => m.startsWith("video/"));
|
|
160
|
+
|
|
161
|
+
let sts: StsCredentials;
|
|
162
|
+
try {
|
|
163
|
+
sts = await fetchSts(relayServerUrl, gatewayId, relaySecret, mimeTypes, hasVideo);
|
|
164
|
+
} catch (err) {
|
|
165
|
+
console.error("[media] failed to fetch STS:", err);
|
|
166
|
+
return [];
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
const results: AssistantAttachment[] = [];
|
|
170
|
+
|
|
171
|
+
// Upload images concurrently (max 3), videos serially
|
|
172
|
+
const imageBlocks = blocks.filter((b) => !b.mimeType.startsWith("video/"));
|
|
173
|
+
const videoBlocks = blocks.filter((b) => b.mimeType.startsWith("video/"));
|
|
174
|
+
|
|
175
|
+
// Images: up to 3 concurrent
|
|
176
|
+
const imageBatches: MediaBlock[][] = [];
|
|
177
|
+
for (let i = 0; i < imageBlocks.length; i += 3) {
|
|
178
|
+
imageBatches.push(imageBlocks.slice(i, i + 3));
|
|
179
|
+
}
|
|
180
|
+
for (const batch of imageBatches) {
|
|
181
|
+
const settled = await Promise.allSettled(
|
|
182
|
+
batch.map((b) => uploadMedia(sts, b.data, b.mimeType))
|
|
183
|
+
);
|
|
184
|
+
for (const r of settled) {
|
|
185
|
+
if (r.status === "fulfilled") {
|
|
186
|
+
results.push(r.value);
|
|
187
|
+
} else {
|
|
188
|
+
console.warn("[media] image upload failed:", r.reason);
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
// Videos: serial
|
|
194
|
+
for (const b of videoBlocks) {
|
|
195
|
+
try {
|
|
196
|
+
const r = await uploadMedia(sts, b.data, b.mimeType);
|
|
197
|
+
results.push(r);
|
|
198
|
+
} catch (err) {
|
|
199
|
+
console.warn("[media] video upload failed:", err);
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
console.log(`[media] uploaded ${results.length}/${blocks.length} attachments`);
|
|
204
|
+
return results;
|
|
205
|
+
}
|
|
@@ -0,0 +1,280 @@
|
|
|
1
|
+
import { createHmac } from "crypto";
|
|
2
|
+
import { randomUUID } from "crypto";
|
|
3
|
+
|
|
4
|
+
// ---------------------------------------------------------------------------
|
|
5
|
+
// Types
|
|
6
|
+
// ---------------------------------------------------------------------------
|
|
7
|
+
|
|
8
|
+
export interface StsCredentials {
|
|
9
|
+
bucket: string;
|
|
10
|
+
region: string;
|
|
11
|
+
endpoint: string;
|
|
12
|
+
baseUrl: string;
|
|
13
|
+
dirPrefix: string;
|
|
14
|
+
expiresAt: number;
|
|
15
|
+
credentials: {
|
|
16
|
+
accessKeyId: string;
|
|
17
|
+
accessKeySecret: string;
|
|
18
|
+
securityToken: string;
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface UploadResult {
|
|
23
|
+
url: string; // public CDN URL
|
|
24
|
+
thumbnailUrl?: string; // OSS snapshot URL (video only)
|
|
25
|
+
mimeType: string;
|
|
26
|
+
size: number;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// ---------------------------------------------------------------------------
|
|
30
|
+
// OSS HMAC-SHA1 request signing (with STS security token)
|
|
31
|
+
// ---------------------------------------------------------------------------
|
|
32
|
+
|
|
33
|
+
function ossSign(
|
|
34
|
+
method: string,
|
|
35
|
+
contentMd5: string,
|
|
36
|
+
contentType: string,
|
|
37
|
+
date: string,
|
|
38
|
+
canonicalizedHeaders: string,
|
|
39
|
+
canonicalizedResource: string,
|
|
40
|
+
secretKey: string
|
|
41
|
+
): string {
|
|
42
|
+
const stringToSign = [
|
|
43
|
+
method,
|
|
44
|
+
contentMd5,
|
|
45
|
+
contentType,
|
|
46
|
+
date,
|
|
47
|
+
canonicalizedHeaders + canonicalizedResource,
|
|
48
|
+
].join("\n");
|
|
49
|
+
return createHmac("sha1", secretKey).update(stringToSign).digest("base64");
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// ---------------------------------------------------------------------------
|
|
53
|
+
// Simple PutObject upload (images / small files)
|
|
54
|
+
// ---------------------------------------------------------------------------
|
|
55
|
+
|
|
56
|
+
export async function putObject(
|
|
57
|
+
sts: StsCredentials,
|
|
58
|
+
data: Buffer,
|
|
59
|
+
ext: string,
|
|
60
|
+
mimeType: string
|
|
61
|
+
): Promise<UploadResult> {
|
|
62
|
+
const { bucket, endpoint, baseUrl, dirPrefix, credentials } = sts;
|
|
63
|
+
const objectKey = `${dirPrefix}/${randomUUID()}${ext}`;
|
|
64
|
+
const date = new Date().toUTCString();
|
|
65
|
+
const canonicalizedResource = `/${bucket}/${objectKey}`;
|
|
66
|
+
const canonicalizedHeaders = `x-oss-security-token:${credentials.securityToken}\n`;
|
|
67
|
+
|
|
68
|
+
const sig = ossSign(
|
|
69
|
+
"PUT",
|
|
70
|
+
"",
|
|
71
|
+
mimeType,
|
|
72
|
+
date,
|
|
73
|
+
canonicalizedHeaders,
|
|
74
|
+
canonicalizedResource,
|
|
75
|
+
credentials.accessKeySecret
|
|
76
|
+
);
|
|
77
|
+
|
|
78
|
+
const url = `${endpoint.replace(/\/$/, "")}/${objectKey}`;
|
|
79
|
+
const res = await fetch(url, {
|
|
80
|
+
method: "PUT",
|
|
81
|
+
headers: {
|
|
82
|
+
"Content-Type": mimeType,
|
|
83
|
+
"Date": date,
|
|
84
|
+
"x-oss-security-token": credentials.securityToken,
|
|
85
|
+
"Authorization": `OSS ${credentials.accessKeyId}:${sig}`,
|
|
86
|
+
},
|
|
87
|
+
body: data as unknown as BodyInit,
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
if (!res.ok) {
|
|
91
|
+
const text = await res.text().catch(() => "");
|
|
92
|
+
throw new Error(`OSS PutObject failed: ${res.status} ${text}`);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
const publicUrl = `${baseUrl.replace(/\/$/, "")}/${objectKey}`;
|
|
96
|
+
return { url: publicUrl, mimeType, size: data.length };
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
// ---------------------------------------------------------------------------
|
|
100
|
+
// Multipart upload (videos > 10 MB)
|
|
101
|
+
// ---------------------------------------------------------------------------
|
|
102
|
+
|
|
103
|
+
const PART_SIZE = 5 * 1024 * 1024; // 5 MB per part
|
|
104
|
+
|
|
105
|
+
export async function multipartUpload(
|
|
106
|
+
sts: StsCredentials,
|
|
107
|
+
data: Buffer,
|
|
108
|
+
ext: string,
|
|
109
|
+
mimeType: string
|
|
110
|
+
): Promise<UploadResult> {
|
|
111
|
+
const { bucket, endpoint, baseUrl, dirPrefix, credentials } = sts;
|
|
112
|
+
const objectKey = `${dirPrefix}/${randomUUID()}${ext}`;
|
|
113
|
+
const baseOssUrl = `${endpoint.replace(/\/$/, "")}/${objectKey}`;
|
|
114
|
+
|
|
115
|
+
function makeHeaders(
|
|
116
|
+
method: string,
|
|
117
|
+
contentType: string,
|
|
118
|
+
extraHeaders: Record<string, string>,
|
|
119
|
+
resource: string
|
|
120
|
+
): Headers {
|
|
121
|
+
const date = new Date().toUTCString();
|
|
122
|
+
const canonicalizedHeaders = [
|
|
123
|
+
"x-oss-security-token:" + credentials.securityToken,
|
|
124
|
+
...Object.entries(extraHeaders)
|
|
125
|
+
.filter(([k]) => k.startsWith("x-oss-"))
|
|
126
|
+
.sort()
|
|
127
|
+
.map(([k, v]) => `${k}:${v}`),
|
|
128
|
+
]
|
|
129
|
+
.sort()
|
|
130
|
+
.join("\n") + "\n";
|
|
131
|
+
|
|
132
|
+
const sig = ossSign(
|
|
133
|
+
method,
|
|
134
|
+
"",
|
|
135
|
+
contentType,
|
|
136
|
+
date,
|
|
137
|
+
canonicalizedHeaders,
|
|
138
|
+
`/${bucket}/${objectKey}`,
|
|
139
|
+
credentials.accessKeySecret
|
|
140
|
+
);
|
|
141
|
+
|
|
142
|
+
const headers = new Headers({
|
|
143
|
+
"Date": date,
|
|
144
|
+
"x-oss-security-token": credentials.securityToken,
|
|
145
|
+
"Authorization": `OSS ${credentials.accessKeyId}:${sig}`,
|
|
146
|
+
...extraHeaders,
|
|
147
|
+
});
|
|
148
|
+
if (contentType) headers.set("Content-Type", contentType);
|
|
149
|
+
return headers;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// 1. Initiate
|
|
153
|
+
const initRes = await fetch(`${baseOssUrl}?uploads`, {
|
|
154
|
+
method: "POST",
|
|
155
|
+
headers: makeHeaders("POST", mimeType, {}, `/${bucket}/${objectKey}`),
|
|
156
|
+
});
|
|
157
|
+
if (!initRes.ok) throw new Error(`OSS InitiateMultipartUpload failed: ${initRes.status}`);
|
|
158
|
+
const initText = await initRes.text();
|
|
159
|
+
const uploadIdMatch = initText.match(/<UploadId>([^<]+)<\/UploadId>/);
|
|
160
|
+
if (!uploadIdMatch) throw new Error("OSS: could not parse UploadId");
|
|
161
|
+
const uploadId = uploadIdMatch[1];
|
|
162
|
+
|
|
163
|
+
// 2. Upload parts
|
|
164
|
+
const parts: Array<{ partNumber: number; etag: string }> = [];
|
|
165
|
+
let partNumber = 1;
|
|
166
|
+
let offset = 0;
|
|
167
|
+
|
|
168
|
+
try {
|
|
169
|
+
while (offset < data.length) {
|
|
170
|
+
const chunk = data.slice(offset, offset + PART_SIZE);
|
|
171
|
+
const partDate = new Date().toUTCString();
|
|
172
|
+
const canonicalizedHeaders = `x-oss-security-token:${credentials.securityToken}\n`;
|
|
173
|
+
const sig = ossSign(
|
|
174
|
+
"PUT", "", "application/octet-stream", partDate,
|
|
175
|
+
canonicalizedHeaders, `/${bucket}/${objectKey}`,
|
|
176
|
+
credentials.accessKeySecret
|
|
177
|
+
);
|
|
178
|
+
const partRes = await fetch(
|
|
179
|
+
`${baseOssUrl}?partNumber=${partNumber}&uploadId=${uploadId}`,
|
|
180
|
+
{
|
|
181
|
+
method: "PUT",
|
|
182
|
+
headers: {
|
|
183
|
+
"Content-Type": "application/octet-stream",
|
|
184
|
+
"Date": partDate,
|
|
185
|
+
"x-oss-security-token": credentials.securityToken,
|
|
186
|
+
"Authorization": `OSS ${credentials.accessKeyId}:${sig}`,
|
|
187
|
+
},
|
|
188
|
+
body: chunk as unknown as BodyInit,
|
|
189
|
+
}
|
|
190
|
+
);
|
|
191
|
+
if (!partRes.ok) throw new Error(`OSS UploadPart ${partNumber} failed: ${partRes.status}`);
|
|
192
|
+
const etag = partRes.headers.get("etag") ?? "";
|
|
193
|
+
parts.push({ partNumber, etag });
|
|
194
|
+
partNumber++;
|
|
195
|
+
offset += PART_SIZE;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
// 3. Complete
|
|
199
|
+
const completeBody = [
|
|
200
|
+
"<CompleteMultipartUpload>",
|
|
201
|
+
...parts.map(
|
|
202
|
+
(p) => `<Part><PartNumber>${p.partNumber}</PartNumber><ETag>${p.etag}</ETag></Part>`
|
|
203
|
+
),
|
|
204
|
+
"</CompleteMultipartUpload>",
|
|
205
|
+
].join("");
|
|
206
|
+
|
|
207
|
+
const completeDate = new Date().toUTCString();
|
|
208
|
+
const cHeaders = `x-oss-security-token:${credentials.securityToken}\n`;
|
|
209
|
+
const cSig = ossSign(
|
|
210
|
+
"POST", "", "application/xml", completeDate,
|
|
211
|
+
cHeaders, `/${bucket}/${objectKey}`, credentials.accessKeySecret
|
|
212
|
+
);
|
|
213
|
+
const completeRes = await fetch(`${baseOssUrl}?uploadId=${uploadId}`, {
|
|
214
|
+
method: "POST",
|
|
215
|
+
headers: {
|
|
216
|
+
"Content-Type": "application/xml",
|
|
217
|
+
"Date": completeDate,
|
|
218
|
+
"x-oss-security-token": credentials.securityToken,
|
|
219
|
+
"Authorization": `OSS ${credentials.accessKeyId}:${cSig}`,
|
|
220
|
+
},
|
|
221
|
+
body: completeBody,
|
|
222
|
+
});
|
|
223
|
+
if (!completeRes.ok) throw new Error(`OSS CompleteMultipartUpload failed: ${completeRes.status}`);
|
|
224
|
+
|
|
225
|
+
} catch (err) {
|
|
226
|
+
// Best-effort abort to avoid billing for incomplete upload
|
|
227
|
+
try {
|
|
228
|
+
const abortDate = new Date().toUTCString();
|
|
229
|
+
const aHeaders = `x-oss-security-token:${credentials.securityToken}\n`;
|
|
230
|
+
const aSig = ossSign(
|
|
231
|
+
"DELETE", "", "", abortDate,
|
|
232
|
+
aHeaders, `/${bucket}/${objectKey}`, credentials.accessKeySecret
|
|
233
|
+
);
|
|
234
|
+
await fetch(`${baseOssUrl}?uploadId=${uploadId}`, {
|
|
235
|
+
method: "DELETE",
|
|
236
|
+
headers: {
|
|
237
|
+
"Date": abortDate,
|
|
238
|
+
"x-oss-security-token": credentials.securityToken,
|
|
239
|
+
"Authorization": `OSS ${credentials.accessKeyId}:${aSig}`,
|
|
240
|
+
},
|
|
241
|
+
});
|
|
242
|
+
} catch { /* ignore abort errors */ }
|
|
243
|
+
throw err;
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
const publicUrl = `${baseUrl.replace(/\/$/, "")}/${objectKey}`;
|
|
247
|
+
// OSS video snapshot thumbnail URL
|
|
248
|
+
const thumbnailUrl = `${publicUrl}?x-oss-process=video/snapshot,t_0,f_jpg,w_0,h_0,m_fast`;
|
|
249
|
+
return { url: publicUrl, thumbnailUrl, mimeType, size: data.length };
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
// ---------------------------------------------------------------------------
|
|
253
|
+
// Convenience: auto-select PutObject vs multipart based on size
|
|
254
|
+
// ---------------------------------------------------------------------------
|
|
255
|
+
|
|
256
|
+
const MULTIPART_THRESHOLD = 10 * 1024 * 1024; // 10 MB
|
|
257
|
+
|
|
258
|
+
export async function uploadMedia(
|
|
259
|
+
sts: StsCredentials,
|
|
260
|
+
data: Buffer,
|
|
261
|
+
mimeType: string
|
|
262
|
+
): Promise<UploadResult> {
|
|
263
|
+
const ext = mimeTypeToExt(mimeType);
|
|
264
|
+
if (data.length >= MULTIPART_THRESHOLD) {
|
|
265
|
+
return multipartUpload(sts, data, ext, mimeType);
|
|
266
|
+
}
|
|
267
|
+
return putObject(sts, data, ext, mimeType);
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
function mimeTypeToExt(mimeType: string): string {
|
|
271
|
+
const map: Record<string, string> = {
|
|
272
|
+
"image/jpeg": ".jpg",
|
|
273
|
+
"image/png": ".png",
|
|
274
|
+
"image/gif": ".gif",
|
|
275
|
+
"image/webp": ".webp",
|
|
276
|
+
"video/mp4": ".mp4",
|
|
277
|
+
"video/quicktime": ".mov",
|
|
278
|
+
};
|
|
279
|
+
return map[mimeType] ?? ".bin";
|
|
280
|
+
}
|
|
@@ -225,19 +225,23 @@ function installLinuxServiceNohup(): boolean {
|
|
|
225
225
|
}
|
|
226
226
|
|
|
227
227
|
function installLinuxService(): boolean {
|
|
228
|
+
try {
|
|
229
|
+
if (installLinuxServiceNohup()) {
|
|
230
|
+
return true;
|
|
231
|
+
}
|
|
232
|
+
} catch {
|
|
233
|
+
// Fall back to systemd below.
|
|
234
|
+
}
|
|
235
|
+
|
|
228
236
|
if (canUseSystemdUser()) {
|
|
229
237
|
try {
|
|
230
238
|
return installLinuxServiceSystemd();
|
|
231
239
|
} catch {
|
|
232
|
-
//
|
|
240
|
+
// Give up below.
|
|
233
241
|
}
|
|
234
242
|
}
|
|
235
243
|
|
|
236
|
-
|
|
237
|
-
return installLinuxServiceNohup();
|
|
238
|
-
} catch {
|
|
239
|
-
return false;
|
|
240
|
-
}
|
|
244
|
+
return false;
|
|
241
245
|
}
|
|
242
246
|
|
|
243
247
|
function installWindowsService(): boolean {
|