pi-feishu-cli 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +111 -0
- package/dist/src/bot.d.ts +29 -0
- package/dist/src/bot.js +75 -0
- package/dist/src/cards.d.ts +6 -0
- package/dist/src/cards.js +87 -0
- package/dist/src/config.d.ts +3 -0
- package/dist/src/config.js +28 -0
- package/dist/src/daemon.d.ts +2 -0
- package/dist/src/daemon.js +151 -0
- package/dist/src/extension.d.ts +2 -0
- package/dist/src/extension.js +124 -0
- package/dist/src/poller.d.ts +33 -0
- package/dist/src/poller.js +94 -0
- package/dist/src/renderer.d.ts +8 -0
- package/dist/src/renderer.js +31 -0
- package/dist/src/session-registry.d.ts +15 -0
- package/dist/src/session-registry.js +82 -0
- package/dist/src/types.d.ts +25 -0
- package/dist/src/types.js +1 -0
- package/dist/tests/bot.test.d.ts +1 -0
- package/dist/tests/bot.test.js +89 -0
- package/dist/tests/cards.test.d.ts +1 -0
- package/dist/tests/cards.test.js +39 -0
- package/dist/tests/config.test.d.ts +1 -0
- package/dist/tests/config.test.js +59 -0
- package/dist/tests/renderer.test.d.ts +1 -0
- package/dist/tests/renderer.test.js +61 -0
- package/dist/tests/session-registry.test.d.ts +1 -0
- package/dist/tests/session-registry.test.js +92 -0
- package/dist/tests/types.test.d.ts +1 -0
- package/dist/tests/types.test.js +30 -0
- package/package.json +35 -0
- package/skills/lark-approval/SKILL.md +56 -0
- package/skills/lark-apps/SKILL.md +92 -0
- package/skills/lark-apps/references/lark-apps-access-scope-get.md +104 -0
- package/skills/lark-apps/references/lark-apps-access-scope-set.md +126 -0
- package/skills/lark-apps/references/lark-apps-create.md +112 -0
- package/skills/lark-apps/references/lark-apps-html-publish.md +151 -0
- package/skills/lark-apps/references/lark-apps-list.md +95 -0
- package/skills/lark-apps/references/lark-apps-update.md +86 -0
- package/skills/lark-attendance/SKILL.md +57 -0
- package/skills/lark-base/SKILL.md +359 -0
- package/skills/lark-base/references/dashboard-block-data-config.md +350 -0
- package/skills/lark-base/references/examples.md +140 -0
- package/skills/lark-base/references/formula-field-guide.md +737 -0
- package/skills/lark-base/references/lark-base-advperm-disable.md +83 -0
- package/skills/lark-base/references/lark-base-advperm-enable.md +80 -0
- package/skills/lark-base/references/lark-base-base-copy.md +74 -0
- package/skills/lark-base/references/lark-base-base-create.md +68 -0
- package/skills/lark-base/references/lark-base-base-get.md +39 -0
- package/skills/lark-base/references/lark-base-cell-value.md +151 -0
- package/skills/lark-base/references/lark-base-dashboard-arrange.md +83 -0
- package/skills/lark-base/references/lark-base-dashboard-block-create.md +108 -0
- package/skills/lark-base/references/lark-base-dashboard-block-delete.md +46 -0
- package/skills/lark-base/references/lark-base-dashboard-block-get.md +57 -0
- package/skills/lark-base/references/lark-base-dashboard-block-list.md +53 -0
- package/skills/lark-base/references/lark-base-dashboard-block-update.md +84 -0
- package/skills/lark-base/references/lark-base-dashboard-create.md +73 -0
- package/skills/lark-base/references/lark-base-dashboard-delete.md +44 -0
- package/skills/lark-base/references/lark-base-dashboard-get.md +59 -0
- package/skills/lark-base/references/lark-base-dashboard-list.md +52 -0
- package/skills/lark-base/references/lark-base-dashboard-update.md +69 -0
- package/skills/lark-base/references/lark-base-dashboard.md +240 -0
- package/skills/lark-base/references/lark-base-data-analysis-sop.md +88 -0
- package/skills/lark-base/references/lark-base-data-query.md +375 -0
- package/skills/lark-base/references/lark-base-field-create.md +104 -0
- package/skills/lark-base/references/lark-base-field-delete.md +51 -0
- package/skills/lark-base/references/lark-base-field-get.md +42 -0
- package/skills/lark-base/references/lark-base-field-list.md +44 -0
- package/skills/lark-base/references/lark-base-field-search-options.md +48 -0
- package/skills/lark-base/references/lark-base-field-update.md +97 -0
- package/skills/lark-base/references/lark-base-field.md +22 -0
- package/skills/lark-base/references/lark-base-form-create.md +87 -0
- package/skills/lark-base/references/lark-base-form-delete.md +64 -0
- package/skills/lark-base/references/lark-base-form-detail.md +198 -0
- package/skills/lark-base/references/lark-base-form-get.md +68 -0
- package/skills/lark-base/references/lark-base-form-list.md +73 -0
- package/skills/lark-base/references/lark-base-form-questions-create.md +118 -0
- package/skills/lark-base/references/lark-base-form-questions-delete.md +68 -0
- package/skills/lark-base/references/lark-base-form-questions-list.md +84 -0
- package/skills/lark-base/references/lark-base-form-questions-update.md +92 -0
- package/skills/lark-base/references/lark-base-form-questions.md +23 -0
- package/skills/lark-base/references/lark-base-form-submit.md +171 -0
- package/skills/lark-base/references/lark-base-form-update.md +82 -0
- package/skills/lark-base/references/lark-base-form.md +25 -0
- package/skills/lark-base/references/lark-base-history.md +16 -0
- package/skills/lark-base/references/lark-base-record-batch-create.md +58 -0
- package/skills/lark-base/references/lark-base-record-batch-update.md +53 -0
- package/skills/lark-base/references/lark-base-record-delete.md +62 -0
- package/skills/lark-base/references/lark-base-record-history-list.md +86 -0
- package/skills/lark-base/references/lark-base-record-share-link-create.md +72 -0
- package/skills/lark-base/references/lark-base-record-upsert.md +64 -0
- package/skills/lark-base/references/lark-base-record.md +31 -0
- package/skills/lark-base/references/lark-base-role-create.md +89 -0
- package/skills/lark-base/references/lark-base-role-delete.md +83 -0
- package/skills/lark-base/references/lark-base-role-get.md +87 -0
- package/skills/lark-base/references/lark-base-role-list.md +81 -0
- package/skills/lark-base/references/lark-base-role-update.md +94 -0
- package/skills/lark-base/references/lark-base-shortcut-field-properties.md +481 -0
- package/skills/lark-base/references/lark-base-table-create.md +62 -0
- package/skills/lark-base/references/lark-base-table-delete.md +51 -0
- package/skills/lark-base/references/lark-base-table-get.md +46 -0
- package/skills/lark-base/references/lark-base-table-list.md +43 -0
- package/skills/lark-base/references/lark-base-table-update.md +49 -0
- package/skills/lark-base/references/lark-base-table.md +20 -0
- package/skills/lark-base/references/lark-base-view-create.md +50 -0
- package/skills/lark-base/references/lark-base-view-delete.md +48 -0
- package/skills/lark-base/references/lark-base-view-get-card.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-filter.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-group.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-sort.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-timebar.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-visible-fields.md +28 -0
- package/skills/lark-base/references/lark-base-view-get.md +38 -0
- package/skills/lark-base/references/lark-base-view-list.md +44 -0
- package/skills/lark-base/references/lark-base-view-rename.md +44 -0
- package/skills/lark-base/references/lark-base-view-set-card.md +55 -0
- package/skills/lark-base/references/lark-base-view-set-filter.md +181 -0
- package/skills/lark-base/references/lark-base-view-set-group.md +65 -0
- package/skills/lark-base/references/lark-base-view-set-sort.md +63 -0
- package/skills/lark-base/references/lark-base-view-set-timebar.md +51 -0
- package/skills/lark-base/references/lark-base-view-set-visible-fields.md +46 -0
- package/skills/lark-base/references/lark-base-view.md +44 -0
- package/skills/lark-base/references/lark-base-workflow-create.md +180 -0
- package/skills/lark-base/references/lark-base-workflow-disable.md +94 -0
- package/skills/lark-base/references/lark-base-workflow-enable.md +94 -0
- package/skills/lark-base/references/lark-base-workflow-get.md +147 -0
- package/skills/lark-base/references/lark-base-workflow-guide.md +718 -0
- package/skills/lark-base/references/lark-base-workflow-list.md +124 -0
- package/skills/lark-base/references/lark-base-workflow-schema.md +935 -0
- package/skills/lark-base/references/lark-base-workflow-update.md +167 -0
- package/skills/lark-base/references/lark-base-workflow.md +23 -0
- package/skills/lark-base/references/lark-base-workspace.md +18 -0
- package/skills/lark-base/references/lookup-field-guide.md +512 -0
- package/skills/lark-base/references/role-config.md +539 -0
- package/skills/lark-calendar/SKILL.md +154 -0
- package/skills/lark-calendar/references/lark-calendar-agenda.md +78 -0
- package/skills/lark-calendar/references/lark-calendar-create.md +109 -0
- package/skills/lark-calendar/references/lark-calendar-freebusy.md +124 -0
- package/skills/lark-calendar/references/lark-calendar-room-find.md +113 -0
- package/skills/lark-calendar/references/lark-calendar-rsvp.md +42 -0
- package/skills/lark-calendar/references/lark-calendar-schedule-meeting.md +265 -0
- package/skills/lark-calendar/references/lark-calendar-suggestion.md +125 -0
- package/skills/lark-calendar/references/lark-calendar-update.md +105 -0
- package/skills/lark-contact/SKILL.md +45 -0
- package/skills/lark-contact/references/lark-contact-get-user.md +19 -0
- package/skills/lark-contact/references/lark-contact-search-user.md +124 -0
- package/skills/lark-doc/SKILL.md +65 -0
- package/skills/lark-doc/references/lark-doc-create.md +89 -0
- package/skills/lark-doc/references/lark-doc-fetch.md +141 -0
- package/skills/lark-doc/references/lark-doc-md.md +71 -0
- package/skills/lark-doc/references/lark-doc-media-download.md +50 -0
- package/skills/lark-doc/references/lark-doc-media-insert.md +114 -0
- package/skills/lark-doc/references/lark-doc-media-preview.md +41 -0
- package/skills/lark-doc/references/lark-doc-update.md +252 -0
- package/skills/lark-doc/references/lark-doc-whiteboard.md +100 -0
- package/skills/lark-doc/references/lark-doc-xml.md +169 -0
- package/skills/lark-doc/references/style/lark-doc-create-workflow.md +56 -0
- package/skills/lark-doc/references/style/lark-doc-style.md +106 -0
- package/skills/lark-doc/references/style/lark-doc-update-workflow.md +54 -0
- package/skills/lark-drive/SKILL.md +369 -0
- package/skills/lark-drive/references/lark-drive-add-comment.md +182 -0
- package/skills/lark-drive/references/lark-drive-apply-permission.md +77 -0
- package/skills/lark-drive/references/lark-drive-create-folder.md +73 -0
- package/skills/lark-drive/references/lark-drive-create-shortcut.md +103 -0
- package/skills/lark-drive/references/lark-drive-delete.md +79 -0
- package/skills/lark-drive/references/lark-drive-download.md +31 -0
- package/skills/lark-drive/references/lark-drive-export-download.md +50 -0
- package/skills/lark-drive/references/lark-drive-export.md +119 -0
- package/skills/lark-drive/references/lark-drive-import.md +159 -0
- package/skills/lark-drive/references/lark-drive-inspect.md +50 -0
- package/skills/lark-drive/references/lark-drive-move.md +120 -0
- package/skills/lark-drive/references/lark-drive-pull.md +137 -0
- package/skills/lark-drive/references/lark-drive-push.md +162 -0
- package/skills/lark-drive/references/lark-drive-reactions.md +113 -0
- package/skills/lark-drive/references/lark-drive-search.md +266 -0
- package/skills/lark-drive/references/lark-drive-status.md +198 -0
- package/skills/lark-drive/references/lark-drive-task-result.md +302 -0
- package/skills/lark-drive/references/lark-drive-upload.md +67 -0
- package/skills/lark-drive/references/lark-drive-version-delete.md +38 -0
- package/skills/lark-drive/references/lark-drive-version-get.md +71 -0
- package/skills/lark-drive/references/lark-drive-version-history.md +73 -0
- package/skills/lark-drive/references/lark-drive-version-revert.md +198 -0
- package/skills/lark-event/SKILL.md +145 -0
- package/skills/lark-event/references/lark-event-im.md +86 -0
- package/skills/lark-im/SKILL.md +162 -0
- package/skills/lark-im/references/lark-im-chat-create.md +162 -0
- package/skills/lark-im/references/lark-im-chat-identity.md +55 -0
- package/skills/lark-im/references/lark-im-chat-list.md +198 -0
- package/skills/lark-im/references/lark-im-chat-messages-list.md +148 -0
- package/skills/lark-im/references/lark-im-chat-search.md +136 -0
- package/skills/lark-im/references/lark-im-chat-update.md +84 -0
- package/skills/lark-im/references/lark-im-flag-cancel.md +198 -0
- package/skills/lark-im/references/lark-im-flag-create.md +67 -0
- package/skills/lark-im/references/lark-im-flag-list.md +100 -0
- package/skills/lark-im/references/lark-im-messages-mget.md +95 -0
- package/skills/lark-im/references/lark-im-messages-reply.md +228 -0
- package/skills/lark-im/references/lark-im-messages-resources-download.md +94 -0
- package/skills/lark-im/references/lark-im-messages-search.md +232 -0
- package/skills/lark-im/references/lark-im-messages-send.md +229 -0
- package/skills/lark-im/references/lark-im-reactions.md +297 -0
- package/skills/lark-im/references/lark-im-threads-messages-list.md +111 -0
- package/skills/lark-mail/SKILL.md +648 -0
- package/skills/lark-mail/references/lark-mail-decline-receipt.md +115 -0
- package/skills/lark-mail/references/lark-mail-draft-create.md +123 -0
- package/skills/lark-mail/references/lark-mail-draft-edit.md +400 -0
- package/skills/lark-mail/references/lark-mail-forward.md +173 -0
- package/skills/lark-mail/references/lark-mail-message.md +230 -0
- package/skills/lark-mail/references/lark-mail-messages.md +108 -0
- package/skills/lark-mail/references/lark-mail-reply-all.md +206 -0
- package/skills/lark-mail/references/lark-mail-reply.md +242 -0
- package/skills/lark-mail/references/lark-mail-send-receipt.md +198 -0
- package/skills/lark-mail/references/lark-mail-send.md +216 -0
- package/skills/lark-mail/references/lark-mail-share-to-chat.md +198 -0
- package/skills/lark-mail/references/lark-mail-signature.md +98 -0
- package/skills/lark-mail/references/lark-mail-template-create.md +129 -0
- package/skills/lark-mail/references/lark-mail-template-update.md +198 -0
- package/skills/lark-mail/references/lark-mail-thread.md +111 -0
- package/skills/lark-mail/references/lark-mail-triage.md +122 -0
- package/skills/lark-mail/references/lark-mail-watch.md +94 -0
- package/skills/lark-minutes/SKILL.md +139 -0
- package/skills/lark-minutes/references/lark-minutes-download.md +137 -0
- package/skills/lark-minutes/references/lark-minutes-search.md +206 -0
- package/skills/lark-minutes/references/lark-minutes-upload.md +104 -0
- package/skills/lark-okr/SKILL.md +133 -0
- package/skills/lark-okr/references/lark-okr-contentblock.md +359 -0
- package/skills/lark-okr/references/lark-okr-cycle-detail.md +84 -0
- package/skills/lark-okr/references/lark-okr-cycle-list.md +90 -0
- package/skills/lark-okr/references/lark-okr-entities.md +329 -0
- package/skills/lark-okr/references/lark-okr-image-upload.md +116 -0
- package/skills/lark-okr/references/lark-okr-progress-create.md +81 -0
- package/skills/lark-okr/references/lark-okr-progress-delete.md +47 -0
- package/skills/lark-okr/references/lark-okr-progress-get.md +62 -0
- package/skills/lark-okr/references/lark-okr-progress-list.md +80 -0
- package/skills/lark-okr/references/lark-okr-progress-update.md +81 -0
- package/skills/lark-openapi-explorer/SKILL.md +153 -0
- package/skills/lark-shared/SKILL.md +144 -0
- package/skills/lark-sheets/SKILL.md +343 -0
- package/skills/lark-sheets/references/lark-sheets-cell-data.md +197 -0
- package/skills/lark-sheets/references/lark-sheets-cell-images.md +59 -0
- package/skills/lark-sheets/references/lark-sheets-cell-style-and-merge.md +141 -0
- package/skills/lark-sheets/references/lark-sheets-dropdown.md +133 -0
- package/skills/lark-sheets/references/lark-sheets-filter-views.md +193 -0
- package/skills/lark-sheets/references/lark-sheets-float-images.md +125 -0
- package/skills/lark-sheets/references/lark-sheets-formula.md +88 -0
- package/skills/lark-sheets/references/lark-sheets-row-column-management.md +151 -0
- package/skills/lark-sheets/references/lark-sheets-sheet-management.md +164 -0
- package/skills/lark-sheets/references/lark-sheets-spreadsheet-management.md +140 -0
- package/skills/lark-skill-maker/SKILL.md +85 -0
- package/skills/lark-slides/SKILL.md +296 -0
- package/skills/lark-slides/assets/templates/administration--all_hands_meeting.xml +1999 -0
- package/skills/lark-slides/assets/templates/administration--annual_gala.xml +1160 -0
- package/skills/lark-slides/assets/templates/administration--company_intro.xml +1376 -0
- package/skills/lark-slides/assets/templates/administration--corporate_culture.xml +1765 -0
- package/skills/lark-slides/assets/templates/hr--employee_training.xml +912 -0
- package/skills/lark-slides/assets/templates/hr--employee_training_workshop.xml +1504 -0
- package/skills/lark-slides/assets/templates/hr--onboarding.xml +933 -0
- package/skills/lark-slides/assets/templates/marketing--brand_communication.xml +1367 -0
- package/skills/lark-slides/assets/templates/marketing--brand_logo_design.xml +1347 -0
- package/skills/lark-slides/assets/templates/marketing--brand_operations_plan.xml +1309 -0
- package/skills/lark-slides/assets/templates/marketing--business_plan.xml +1646 -0
- package/skills/lark-slides/assets/templates/marketing--marketing_plan.xml +198 -0
- package/skills/lark-slides/assets/templates/marketing--marketing_strategy.xml +1484 -0
- package/skills/lark-slides/assets/templates/marketing--product_whitepaper.xml +198 -0
- package/skills/lark-slides/assets/templates/marketing--roadshow_business_plan.xml +1506 -0
- package/skills/lark-slides/assets/templates/misc--book_sharing.xml +198 -0
- package/skills/lark-slides/assets/templates/misc--club_event_plan.xml +4885 -0
- package/skills/lark-slides/assets/templates/misc--student_career_plan.xml +1854 -0
- package/skills/lark-slides/assets/templates/office--dark_general.xml +3763 -0
- package/skills/lark-slides/assets/templates/office--dept_annual_report.xml +1192 -0
- package/skills/lark-slides/assets/templates/office--light_general.xml +3378 -0
- package/skills/lark-slides/assets/templates/office--project_kickoff.xml +3152 -0
- package/skills/lark-slides/assets/templates/office--quarterly_review.xml +1253 -0
- package/skills/lark-slides/assets/templates/office--work_report.xml +1099 -0
- package/skills/lark-slides/assets/templates/office--work_summary.xml +4420 -0
- package/skills/lark-slides/assets/templates/office--work_summary_report.xml +1523 -0
- package/skills/lark-slides/assets/templates/operations--brand_logo_design.xml +1347 -0
- package/skills/lark-slides/assets/templates/operations--brand_operations_plan.xml +1309 -0
- package/skills/lark-slides/assets/templates/operations--marketing_plan.xml +1469 -0
- package/skills/lark-slides/assets/templates/operations--product_promotion.xml +687 -0
- package/skills/lark-slides/assets/templates/personal--experience_sharing.xml +2242 -0
- package/skills/lark-slides/assets/templates/personal--personal_resume.xml +2047 -0
- package/skills/lark-slides/assets/templates/personal--promotion_defense.xml +1099 -0
- package/skills/lark-slides/assets/templates/personal--promotion_report.xml +1039 -0
- package/skills/lark-slides/assets/templates/personal--self_intro.xml +696 -0
- package/skills/lark-slides/assets/templates/personal--teaching_sharing.xml +3013 -0
- package/skills/lark-slides/assets/templates/product--business_case_analysis.xml +1341 -0
- package/skills/lark-slides/assets/templates/product--market_analysis.xml +898 -0
- package/skills/lark-slides/assets/templates/product--product_analysis.xml +1537 -0
- package/skills/lark-slides/assets/templates/product--product_intro.xml +2838 -0
- package/skills/lark-slides/assets/templates/product--product_promotion.xml +687 -0
- package/skills/lark-slides/assets/templates/product--product_promotion_2.xml +198 -0
- package/skills/lark-slides/references/asset-planning.md +124 -0
- package/skills/lark-slides/references/examples.md +261 -0
- package/skills/lark-slides/references/lark-slides-create.md +137 -0
- package/skills/lark-slides/references/lark-slides-edit-workflows.md +142 -0
- package/skills/lark-slides/references/lark-slides-media-upload.md +128 -0
- package/skills/lark-slides/references/lark-slides-replace-slide.md +239 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-create.md +220 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-delete.md +123 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-get.md +110 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-replace.md +186 -0
- package/skills/lark-slides/references/lark-slides-xml-presentations-get.md +98 -0
- package/skills/lark-slides/references/planning-layer.md +219 -0
- package/skills/lark-slides/references/slide-templates.md +201 -0
- package/skills/lark-slides/references/slides_demo.xml +226 -0
- package/skills/lark-slides/references/slides_xml_schema_definition.xml +3004 -0
- package/skills/lark-slides/references/template-catalog.md +463 -0
- package/skills/lark-slides/references/template-index.json +198 -0
- package/skills/lark-slides/references/troubleshooting.md +198 -0
- package/skills/lark-slides/references/validation-checklist.md +102 -0
- package/skills/lark-slides/references/visual-planning.md +250 -0
- package/skills/lark-slides/references/xml-format-guide.md +369 -0
- package/skills/lark-slides/references/xml-schema-quick-ref.md +215 -0
- package/skills/lark-slides/scripts/template_tool.py +970 -0
- package/skills/lark-slides/scripts/template_tool_test.py +177 -0
- package/skills/lark-slides/scripts/xml_text_overlap_lint.py +367 -0
- package/skills/lark-slides/scripts/xml_text_overlap_lint_test.py +263 -0
- package/skills/lark-task/SKILL.md +165 -0
- package/skills/lark-task/references/lark-task-assign.md +38 -0
- package/skills/lark-task/references/lark-task-comment.md +28 -0
- package/skills/lark-task/references/lark-task-complete.md +27 -0
- package/skills/lark-task/references/lark-task-create.md +57 -0
- package/skills/lark-task/references/lark-task-followers.md +35 -0
- package/skills/lark-task/references/lark-task-get-my-tasks.md +55 -0
- package/skills/lark-task/references/lark-task-get-related-tasks.md +53 -0
- package/skills/lark-task/references/lark-task-reminder.md +36 -0
- package/skills/lark-task/references/lark-task-reopen.md +27 -0
- package/skills/lark-task/references/lark-task-search.md +41 -0
- package/skills/lark-task/references/lark-task-set-ancestor.md +32 -0
- package/skills/lark-task/references/lark-task-subscribe-event.md +86 -0
- package/skills/lark-task/references/lark-task-tasklist-create.md +35 -0
- package/skills/lark-task/references/lark-task-tasklist-members.md +36 -0
- package/skills/lark-task/references/lark-task-tasklist-search.md +38 -0
- package/skills/lark-task/references/lark-task-tasklist-task-add.md +38 -0
- package/skills/lark-task/references/lark-task-update.md +37 -0
- package/skills/lark-task/references/lark-task-upload-attachment.md +59 -0
- package/skills/lark-vc/SKILL.md +168 -0
- package/skills/lark-vc/references/lark-vc-notes.md +126 -0
- package/skills/lark-vc/references/lark-vc-recording.md +153 -0
- package/skills/lark-vc/references/lark-vc-search.md +193 -0
- package/skills/lark-vc-agent/SKILL.md +121 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-events.md +247 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-join.md +133 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-leave.md +111 -0
- package/skills/lark-whiteboard/SKILL.md +144 -0
- package/skills/lark-whiteboard/references/connectors.md +102 -0
- package/skills/lark-whiteboard/references/content.md +40 -0
- package/skills/lark-whiteboard/references/image.md +80 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-query.md +49 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-update.md +100 -0
- package/skills/lark-whiteboard/references/layout.md +374 -0
- package/skills/lark-whiteboard/references/schema.md +357 -0
- package/skills/lark-whiteboard/references/style.md +318 -0
- package/skills/lark-whiteboard/references/typography.md +73 -0
- package/skills/lark-whiteboard/routes/dsl.md +107 -0
- package/skills/lark-whiteboard/routes/mermaid.md +27 -0
- package/skills/lark-whiteboard/routes/svg.md +54 -0
- package/skills/lark-whiteboard/scenes/architecture.md +433 -0
- package/skills/lark-whiteboard/scenes/bar-chart.md +187 -0
- package/skills/lark-whiteboard/scenes/comparison.md +135 -0
- package/skills/lark-whiteboard/scenes/fishbone.md +238 -0
- package/skills/lark-whiteboard/scenes/flowchart.md +198 -0
- package/skills/lark-whiteboard/scenes/flywheel.md +195 -0
- package/skills/lark-whiteboard/scenes/funnel.md +198 -0
- package/skills/lark-whiteboard/scenes/line-chart.md +214 -0
- package/skills/lark-whiteboard/scenes/mermaid.md +130 -0
- package/skills/lark-whiteboard/scenes/milestone.md +139 -0
- package/skills/lark-whiteboard/scenes/organization.md +173 -0
- package/skills/lark-whiteboard/scenes/photo-showcase.md +198 -0
- package/skills/lark-whiteboard/scenes/pyramid.md +99 -0
- package/skills/lark-whiteboard/scenes/swimlane.md +371 -0
- package/skills/lark-whiteboard/scenes/treemap.md +198 -0
- package/skills/lark-wiki/SKILL.md +119 -0
- package/skills/lark-wiki/references/lark-wiki-delete-space.md +205 -0
- package/skills/lark-wiki/references/lark-wiki-member-add.md +66 -0
- package/skills/lark-wiki/references/lark-wiki-member-list.md +76 -0
- package/skills/lark-wiki/references/lark-wiki-member-remove.md +61 -0
- package/skills/lark-wiki/references/lark-wiki-move.md +183 -0
- package/skills/lark-wiki/references/lark-wiki-node-copy.md +72 -0
- package/skills/lark-wiki/references/lark-wiki-node-create.md +127 -0
- package/skills/lark-wiki/references/lark-wiki-node-delete.md +62 -0
- package/skills/lark-wiki/references/lark-wiki-node-get.md +56 -0
- package/skills/lark-wiki/references/lark-wiki-node-list.md +198 -0
- package/skills/lark-wiki/references/lark-wiki-space-create.md +46 -0
- package/skills/lark-wiki/references/lark-wiki-space-list.md +198 -0
- package/skills/lark-workflow-meeting-summary/SKILL.md +104 -0
- package/skills/lark-workflow-standup-report/SKILL.md +120 -0
- package/skills/mmx-cli/SKILL.md +440 -0
- package/src/bot.ts +109 -0
- package/src/cards.ts +105 -0
- package/src/config.ts +33 -0
- package/src/daemon.ts +217 -0
- package/src/extension.ts +132 -0
- package/src/poller.ts +135 -0
- package/src/renderer.ts +47 -0
- package/src/session-registry.ts +90 -0
- package/src/types.ts +29 -0
- package/tests/bot.test.ts +104 -0
- package/tests/cards.test.ts +48 -0
- package/tests/config.test.ts +59 -0
- package/tests/renderer.test.ts +74 -0
- package/tests/session-registry.test.ts +94 -0
- package/tests/types.test.ts +35 -0
- package/tsconfig.json +16 -0
- package/vitest.config.ts +7 -0
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { execFile } from "node:child_process";
|
|
2
|
+
import { promisify } from "node:util";
|
|
3
|
+
const execFileAsync = promisify(execFile);
|
|
4
|
+
export async function pollEvents() {
|
|
5
|
+
try {
|
|
6
|
+
const { stdout } = await execFileAsync("lark-cli", [
|
|
7
|
+
"im",
|
|
8
|
+
"+events-poll",
|
|
9
|
+
"--as",
|
|
10
|
+
"bot",
|
|
11
|
+
], { timeout: 30_000 });
|
|
12
|
+
const lines = stdout.trim().split("\n").filter(Boolean);
|
|
13
|
+
const events = [];
|
|
14
|
+
for (const line of lines) {
|
|
15
|
+
try {
|
|
16
|
+
const raw = JSON.parse(line);
|
|
17
|
+
events.push({
|
|
18
|
+
type: raw.type ?? "unknown",
|
|
19
|
+
event: raw.event,
|
|
20
|
+
raw,
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
// skip unparseable lines
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
return { events, error: null };
|
|
28
|
+
}
|
|
29
|
+
catch (err) {
|
|
30
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
31
|
+
return { events: [], error: message };
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
export async function larkCliAvailable() {
|
|
35
|
+
try {
|
|
36
|
+
await execFileAsync("lark-cli", ["--help"], { timeout: 5000 });
|
|
37
|
+
return true;
|
|
38
|
+
}
|
|
39
|
+
catch {
|
|
40
|
+
return false;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
export async function larkCliConfigured() {
|
|
44
|
+
try {
|
|
45
|
+
const { stdout } = await execFileAsync("lark-cli", [
|
|
46
|
+
"config",
|
|
47
|
+
"show",
|
|
48
|
+
], { timeout: 5000 });
|
|
49
|
+
const config = JSON.parse(stdout);
|
|
50
|
+
return !!(config.appId && config.appSecret);
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
export async function sendMessage(content, chatId, msgType = "text") {
|
|
57
|
+
try {
|
|
58
|
+
const body = JSON.stringify({
|
|
59
|
+
receive_id: chatId,
|
|
60
|
+
msg_type: msgType,
|
|
61
|
+
content,
|
|
62
|
+
});
|
|
63
|
+
await execFileAsync("lark-cli", [
|
|
64
|
+
"im",
|
|
65
|
+
"messages",
|
|
66
|
+
"create",
|
|
67
|
+
"--data",
|
|
68
|
+
body,
|
|
69
|
+
"--as",
|
|
70
|
+
"bot",
|
|
71
|
+
], { timeout: 10_000 });
|
|
72
|
+
return true;
|
|
73
|
+
}
|
|
74
|
+
catch {
|
|
75
|
+
return false;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
export async function downloadResource(messageId, fileKey, fileType, outputPath) {
|
|
79
|
+
try {
|
|
80
|
+
await execFileAsync("lark-cli", [
|
|
81
|
+
"im",
|
|
82
|
+
"+messages-resources-download",
|
|
83
|
+
"--message-id", messageId,
|
|
84
|
+
"--file-key", fileKey,
|
|
85
|
+
"--file-type", fileType,
|
|
86
|
+
"--output", outputPath,
|
|
87
|
+
"--as", "bot",
|
|
88
|
+
], { timeout: 30_000 });
|
|
89
|
+
return true;
|
|
90
|
+
}
|
|
91
|
+
catch {
|
|
92
|
+
return false;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
export declare const MESSAGE_MAX_LENGTH = 30000;
|
|
2
|
+
export interface FeishuTextMessage {
|
|
3
|
+
type: "text";
|
|
4
|
+
text: string;
|
|
5
|
+
}
|
|
6
|
+
export declare function renderText(text: string): FeishuTextMessage[];
|
|
7
|
+
export declare function renderCodeBlock(code: string, lang?: string): FeishuTextMessage[];
|
|
8
|
+
export declare function splitLongMessage(text: string): string[];
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
export const MESSAGE_MAX_LENGTH = 30_000;
|
|
2
|
+
export function renderText(text) {
|
|
3
|
+
const parts = splitLongMessage(text);
|
|
4
|
+
return parts.map((part) => ({ type: "text", text: part }));
|
|
5
|
+
}
|
|
6
|
+
export function renderCodeBlock(code, lang) {
|
|
7
|
+
const header = lang ? `\`\`\`${lang}\n` : "```\n";
|
|
8
|
+
const text = header + code + "\n```";
|
|
9
|
+
return renderText(text);
|
|
10
|
+
}
|
|
11
|
+
export function splitLongMessage(text) {
|
|
12
|
+
if (text.length <= MESSAGE_MAX_LENGTH) {
|
|
13
|
+
return [text];
|
|
14
|
+
}
|
|
15
|
+
const parts = [];
|
|
16
|
+
let remaining = text;
|
|
17
|
+
while (remaining.length > 0) {
|
|
18
|
+
if (remaining.length <= MESSAGE_MAX_LENGTH) {
|
|
19
|
+
parts.push(remaining);
|
|
20
|
+
break;
|
|
21
|
+
}
|
|
22
|
+
let cutPoint = MESSAGE_MAX_LENGTH;
|
|
23
|
+
const newlineIdx = remaining.lastIndexOf("\n", MESSAGE_MAX_LENGTH);
|
|
24
|
+
if (newlineIdx > MESSAGE_MAX_LENGTH * 0.5) {
|
|
25
|
+
cutPoint = newlineIdx;
|
|
26
|
+
}
|
|
27
|
+
parts.push(remaining.slice(0, cutPoint));
|
|
28
|
+
remaining = remaining.slice(cutPoint);
|
|
29
|
+
}
|
|
30
|
+
return parts;
|
|
31
|
+
}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { ChatSessions, SessionInfo } from "./types.js";
|
|
2
|
+
export declare class SessionRegistry {
|
|
3
|
+
private registry;
|
|
4
|
+
private registryPath;
|
|
5
|
+
constructor(registryDir: string);
|
|
6
|
+
private load;
|
|
7
|
+
flush(): void;
|
|
8
|
+
private getOrCreateChat;
|
|
9
|
+
getChatSessions(chatId: string): ChatSessions | null;
|
|
10
|
+
getActiveSessionId(chatId: string): string | null;
|
|
11
|
+
ensureSession(chatId: string): SessionInfo;
|
|
12
|
+
createSession(chatId: string, name: string): SessionInfo;
|
|
13
|
+
switchSession(chatId: string, sessionId: string): boolean;
|
|
14
|
+
deleteSession(chatId: string, sessionId: string): boolean;
|
|
15
|
+
}
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync } from "node:fs";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import { randomUUID } from "node:crypto";
|
|
4
|
+
const REGISTRY_FILE = "registry.json";
|
|
5
|
+
export class SessionRegistry {
|
|
6
|
+
registry;
|
|
7
|
+
registryPath;
|
|
8
|
+
constructor(registryDir) {
|
|
9
|
+
this.registryPath = join(registryDir, REGISTRY_FILE);
|
|
10
|
+
this.registry = this.load();
|
|
11
|
+
}
|
|
12
|
+
load() {
|
|
13
|
+
if (!existsSync(this.registryPath))
|
|
14
|
+
return {};
|
|
15
|
+
try {
|
|
16
|
+
return JSON.parse(readFileSync(this.registryPath, "utf-8"));
|
|
17
|
+
}
|
|
18
|
+
catch {
|
|
19
|
+
return {};
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
flush() {
|
|
23
|
+
const dir = this.registryPath.replace(/\/[^/]+$/, "");
|
|
24
|
+
if (!existsSync(dir))
|
|
25
|
+
mkdirSync(dir, { recursive: true });
|
|
26
|
+
writeFileSync(this.registryPath, JSON.stringify(this.registry, null, 2));
|
|
27
|
+
}
|
|
28
|
+
getOrCreateChat(chatId) {
|
|
29
|
+
if (!this.registry[chatId]) {
|
|
30
|
+
this.registry[chatId] = { sessions: [], active: null };
|
|
31
|
+
}
|
|
32
|
+
return this.registry[chatId];
|
|
33
|
+
}
|
|
34
|
+
getChatSessions(chatId) {
|
|
35
|
+
return this.registry[chatId] ?? null;
|
|
36
|
+
}
|
|
37
|
+
getActiveSessionId(chatId) {
|
|
38
|
+
return this.registry[chatId]?.active ?? null;
|
|
39
|
+
}
|
|
40
|
+
ensureSession(chatId) {
|
|
41
|
+
const chat = this.getOrCreateChat(chatId);
|
|
42
|
+
if (chat.active && chat.sessions.find((s) => s.id === chat.active)) {
|
|
43
|
+
return chat.sessions.find((s) => s.id === chat.active);
|
|
44
|
+
}
|
|
45
|
+
return this.createSession(chatId, "默认会话");
|
|
46
|
+
}
|
|
47
|
+
createSession(chatId, name) {
|
|
48
|
+
const chat = this.getOrCreateChat(chatId);
|
|
49
|
+
const session = {
|
|
50
|
+
id: randomUUID(),
|
|
51
|
+
name,
|
|
52
|
+
createdAt: Date.now(),
|
|
53
|
+
};
|
|
54
|
+
chat.sessions.push(session);
|
|
55
|
+
chat.active = session.id;
|
|
56
|
+
this.flush();
|
|
57
|
+
return session;
|
|
58
|
+
}
|
|
59
|
+
switchSession(chatId, sessionId) {
|
|
60
|
+
const chat = this.registry[chatId];
|
|
61
|
+
if (!chat || !chat.sessions.find((s) => s.id === sessionId)) {
|
|
62
|
+
return false;
|
|
63
|
+
}
|
|
64
|
+
chat.active = sessionId;
|
|
65
|
+
this.flush();
|
|
66
|
+
return true;
|
|
67
|
+
}
|
|
68
|
+
deleteSession(chatId, sessionId) {
|
|
69
|
+
const chat = this.registry[chatId];
|
|
70
|
+
if (!chat)
|
|
71
|
+
return false;
|
|
72
|
+
const idx = chat.sessions.findIndex((s) => s.id === sessionId);
|
|
73
|
+
if (idx === -1)
|
|
74
|
+
return false;
|
|
75
|
+
chat.sessions.splice(idx, 1);
|
|
76
|
+
if (chat.active === sessionId) {
|
|
77
|
+
chat.active = chat.sessions.length > 0 ? chat.sessions[0].id : null;
|
|
78
|
+
}
|
|
79
|
+
this.flush();
|
|
80
|
+
return true;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export interface FeishuImConfig {
|
|
2
|
+
strategy: "open" | "mention";
|
|
3
|
+
model?: string;
|
|
4
|
+
pollInterval: number;
|
|
5
|
+
autoStart?: boolean;
|
|
6
|
+
}
|
|
7
|
+
export interface SessionInfo {
|
|
8
|
+
id: string;
|
|
9
|
+
name: string;
|
|
10
|
+
createdAt: number;
|
|
11
|
+
}
|
|
12
|
+
export interface ChatSessions {
|
|
13
|
+
sessions: SessionInfo[];
|
|
14
|
+
active: string | null;
|
|
15
|
+
}
|
|
16
|
+
export interface Registry {
|
|
17
|
+
[chatId: string]: ChatSessions;
|
|
18
|
+
}
|
|
19
|
+
export interface DaemonStatus {
|
|
20
|
+
running: boolean;
|
|
21
|
+
pid: number | null;
|
|
22
|
+
uptime: number | null;
|
|
23
|
+
sessionCount: number;
|
|
24
|
+
chatCount: number;
|
|
25
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach } from "vitest";
|
|
2
|
+
import { Bot } from "../src/bot.js";
|
|
3
|
+
import { SessionRegistry } from "../src/session-registry.js";
|
|
4
|
+
import { tmpdir } from "node:os";
|
|
5
|
+
import { join } from "node:path";
|
|
6
|
+
import { existsSync, mkdirSync } from "node:fs";
|
|
7
|
+
function makeMsgEvent(chatId, text, mentions, threadId) {
|
|
8
|
+
return {
|
|
9
|
+
type: "im.message.receive_v1",
|
|
10
|
+
event: {
|
|
11
|
+
message: {
|
|
12
|
+
chat_id: chatId,
|
|
13
|
+
message_id: "om_" + Math.random().toString(36).slice(2),
|
|
14
|
+
parent_id: threadId,
|
|
15
|
+
message_type: "text",
|
|
16
|
+
content: JSON.stringify({ text }),
|
|
17
|
+
mentions,
|
|
18
|
+
},
|
|
19
|
+
sender: {
|
|
20
|
+
sender_id: { open_id: "ou_test" },
|
|
21
|
+
sender_type: "user",
|
|
22
|
+
},
|
|
23
|
+
},
|
|
24
|
+
raw: {},
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
describe("Bot routing", () => {
|
|
28
|
+
let tmpDir;
|
|
29
|
+
let registry;
|
|
30
|
+
let bot;
|
|
31
|
+
beforeEach(() => {
|
|
32
|
+
tmpDir = join(tmpdir(), "pi-feishu-cli-test-bot-" + Date.now());
|
|
33
|
+
if (!existsSync(tmpDir))
|
|
34
|
+
mkdirSync(tmpDir, { recursive: true });
|
|
35
|
+
registry = new SessionRegistry(tmpDir);
|
|
36
|
+
bot = new Bot(registry, "mention");
|
|
37
|
+
});
|
|
38
|
+
it("detects /new command", () => {
|
|
39
|
+
const event = makeMsgEvent("oc_chat1", "/new 我的新会话");
|
|
40
|
+
const result = bot.route(event);
|
|
41
|
+
expect(result.type).toBe("command");
|
|
42
|
+
if (result.type === "command") {
|
|
43
|
+
expect(result.command).toBe("new");
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
it("detects /sessions command", () => {
|
|
47
|
+
const event = makeMsgEvent("oc_chat1", "/sessions");
|
|
48
|
+
const result = bot.route(event);
|
|
49
|
+
expect(result.type).toBe("command");
|
|
50
|
+
if (result.type === "command") {
|
|
51
|
+
expect(result.command).toBe("sessions");
|
|
52
|
+
}
|
|
53
|
+
});
|
|
54
|
+
it("detects /switch command", () => {
|
|
55
|
+
const event = makeMsgEvent("oc_chat1", "/switch sess_123");
|
|
56
|
+
const result = bot.route(event);
|
|
57
|
+
expect(result.type).toBe("command");
|
|
58
|
+
if (result.type === "command") {
|
|
59
|
+
expect(result.command).toBe("switch");
|
|
60
|
+
}
|
|
61
|
+
});
|
|
62
|
+
it("detects /rm command", () => {
|
|
63
|
+
const event = makeMsgEvent("oc_chat1", "/rm sess_123");
|
|
64
|
+
const result = bot.route(event);
|
|
65
|
+
expect(result.type).toBe("command");
|
|
66
|
+
if (result.type === "command") {
|
|
67
|
+
expect(result.command).toBe("rm");
|
|
68
|
+
}
|
|
69
|
+
});
|
|
70
|
+
it("detects /model command", () => {
|
|
71
|
+
const event = makeMsgEvent("oc_chat1", "/model");
|
|
72
|
+
const result = bot.route(event);
|
|
73
|
+
expect(result.type).toBe("command");
|
|
74
|
+
if (result.type === "command") {
|
|
75
|
+
expect(result.command).toBe("model");
|
|
76
|
+
}
|
|
77
|
+
});
|
|
78
|
+
it("routes regular text as message in mention mode", () => {
|
|
79
|
+
const event = makeMsgEvent("oc_chat1", "你好");
|
|
80
|
+
const result = bot.route(event);
|
|
81
|
+
expect(result.type).toBe("message");
|
|
82
|
+
});
|
|
83
|
+
it("routes all messages in open mode", () => {
|
|
84
|
+
const openBot = new Bot(registry, "open");
|
|
85
|
+
const event = makeMsgEvent("oc_chat1", "你好");
|
|
86
|
+
const result = openBot.route(event);
|
|
87
|
+
expect(result.type).toBe("message");
|
|
88
|
+
});
|
|
89
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { buildSessionListCard, buildModelSelectCard, } from "../src/cards.js";
|
|
3
|
+
describe("buildSessionListCard", () => {
|
|
4
|
+
const sessions = [
|
|
5
|
+
{ id: "abc", name: "修 bug", createdAt: 1700000000 },
|
|
6
|
+
{ id: "def", name: "新功能", createdAt: 1700000100 },
|
|
7
|
+
];
|
|
8
|
+
it("builds card with session entries", () => {
|
|
9
|
+
const card = buildSessionListCard("oc_chat1", sessions, "abc");
|
|
10
|
+
const json = JSON.parse(card);
|
|
11
|
+
expect(json.card.header).toBeDefined();
|
|
12
|
+
expect(json.card.elements).toBeDefined();
|
|
13
|
+
expect(JSON.stringify(json).length).toBeGreaterThan(100);
|
|
14
|
+
});
|
|
15
|
+
it("shows empty state when no sessions", () => {
|
|
16
|
+
const card = buildSessionListCard("oc_chat1", [], null);
|
|
17
|
+
const json = JSON.parse(card);
|
|
18
|
+
expect(JSON.stringify(json)).toContain("暂无会话");
|
|
19
|
+
});
|
|
20
|
+
it("marks active session", () => {
|
|
21
|
+
const card = buildSessionListCard("oc_chat1", sessions, "def");
|
|
22
|
+
const json = JSON.parse(card);
|
|
23
|
+
const str = JSON.stringify(json);
|
|
24
|
+
expect(str).toContain("def");
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
describe("buildModelSelectCard", () => {
|
|
28
|
+
it("builds card with model options", () => {
|
|
29
|
+
const models = [
|
|
30
|
+
{ id: "claude-sonnet", name: "Claude Sonnet" },
|
|
31
|
+
{ id: "gpt-4o", name: "GPT-4o" },
|
|
32
|
+
];
|
|
33
|
+
const card = buildModelSelectCard("oc_chat1", models, "claude-sonnet");
|
|
34
|
+
const json = JSON.parse(card);
|
|
35
|
+
expect(json.card.header).toBeDefined();
|
|
36
|
+
expect(JSON.stringify(json)).toContain("Claude Sonnet");
|
|
37
|
+
expect(JSON.stringify(json)).toContain("GPT-4o");
|
|
38
|
+
});
|
|
39
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { writeFileSync, unlinkSync, existsSync, mkdirSync, rmdirSync } from "node:fs";
|
|
3
|
+
import { join } from "node:path";
|
|
4
|
+
import { tmpdir } from "node:os";
|
|
5
|
+
import { loadConfig, DEFAULT_CONFIG } from "../src/config.js";
|
|
6
|
+
describe("loadConfig", () => {
|
|
7
|
+
const tmpDir = join(tmpdir(), "pi-feishu-cli-test-config");
|
|
8
|
+
const configPath = join(tmpDir, "config.json");
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
if (!existsSync(tmpDir))
|
|
11
|
+
mkdirSync(tmpDir, { recursive: true });
|
|
12
|
+
});
|
|
13
|
+
afterEach(() => {
|
|
14
|
+
try {
|
|
15
|
+
unlinkSync(configPath);
|
|
16
|
+
}
|
|
17
|
+
catch { }
|
|
18
|
+
try {
|
|
19
|
+
rmdirSync(tmpDir);
|
|
20
|
+
}
|
|
21
|
+
catch { }
|
|
22
|
+
});
|
|
23
|
+
it("returns defaults when no config file exists", () => {
|
|
24
|
+
const config = loadConfig(tmpDir);
|
|
25
|
+
expect(config.strategy).toBe(DEFAULT_CONFIG.strategy);
|
|
26
|
+
expect(config.pollInterval).toBe(DEFAULT_CONFIG.pollInterval);
|
|
27
|
+
expect(config.model).toBeUndefined();
|
|
28
|
+
expect(config.autoStart).toBe(DEFAULT_CONFIG.autoStart);
|
|
29
|
+
});
|
|
30
|
+
it("loads and merges partial config", () => {
|
|
31
|
+
writeFileSync(configPath, JSON.stringify({ strategy: "open", pollInterval: 10 }));
|
|
32
|
+
const config = loadConfig(tmpDir);
|
|
33
|
+
expect(config.strategy).toBe("open");
|
|
34
|
+
expect(config.pollInterval).toBe(10);
|
|
35
|
+
});
|
|
36
|
+
it("loads full config", () => {
|
|
37
|
+
writeFileSync(configPath, JSON.stringify({
|
|
38
|
+
strategy: "mention",
|
|
39
|
+
model: "anthropic/claude-sonnet",
|
|
40
|
+
pollInterval: 3,
|
|
41
|
+
autoStart: true,
|
|
42
|
+
}));
|
|
43
|
+
const config = loadConfig(tmpDir);
|
|
44
|
+
expect(config.strategy).toBe("mention");
|
|
45
|
+
expect(config.model).toBe("anthropic/claude-sonnet");
|
|
46
|
+
expect(config.pollInterval).toBe(3);
|
|
47
|
+
expect(config.autoStart).toBe(true);
|
|
48
|
+
});
|
|
49
|
+
it("ignores extra unknown fields", () => {
|
|
50
|
+
writeFileSync(configPath, JSON.stringify({
|
|
51
|
+
strategy: "open",
|
|
52
|
+
pollInterval: 5,
|
|
53
|
+
unknownField: "should be ignored",
|
|
54
|
+
}));
|
|
55
|
+
const config = loadConfig(tmpDir);
|
|
56
|
+
expect(config.strategy).toBe("open");
|
|
57
|
+
expect(config.unknownField).toBeUndefined();
|
|
58
|
+
});
|
|
59
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { renderText, renderCodeBlock, splitLongMessage, MESSAGE_MAX_LENGTH, } from "../src/renderer.js";
|
|
3
|
+
describe("renderText", () => {
|
|
4
|
+
it("returns plain text unchanged", () => {
|
|
5
|
+
const result = renderText("hello world");
|
|
6
|
+
expect(result).toEqual([{ type: "text", text: "hello world" }]);
|
|
7
|
+
});
|
|
8
|
+
it("splits on large text", () => {
|
|
9
|
+
const long = "x".repeat(MESSAGE_MAX_LENGTH + 100);
|
|
10
|
+
const result = renderText(long);
|
|
11
|
+
expect(result.length).toBe(2);
|
|
12
|
+
expect(result[0].type).toBe("text");
|
|
13
|
+
expect(result[1].type).toBe("text");
|
|
14
|
+
});
|
|
15
|
+
it("handles empty text", () => {
|
|
16
|
+
const result = renderText("");
|
|
17
|
+
expect(result).toEqual([{ type: "text", text: "" }]);
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
describe("renderCodeBlock", () => {
|
|
21
|
+
it("wraps code in code block markers", () => {
|
|
22
|
+
const result = renderCodeBlock("console.log(1)", "javascript");
|
|
23
|
+
expect(result).toEqual([
|
|
24
|
+
{
|
|
25
|
+
type: "text",
|
|
26
|
+
text: "```javascript\nconsole.log(1)\n```",
|
|
27
|
+
},
|
|
28
|
+
]);
|
|
29
|
+
});
|
|
30
|
+
it("uses no language when lang not provided", () => {
|
|
31
|
+
const result = renderCodeBlock("print(1)");
|
|
32
|
+
expect(result).toEqual([
|
|
33
|
+
{
|
|
34
|
+
type: "text",
|
|
35
|
+
text: "```\nprint(1)\n```",
|
|
36
|
+
},
|
|
37
|
+
]);
|
|
38
|
+
});
|
|
39
|
+
});
|
|
40
|
+
describe("splitLongMessage", () => {
|
|
41
|
+
it("does not split short message", () => {
|
|
42
|
+
const result = splitLongMessage("short text");
|
|
43
|
+
expect(result).toEqual(["short text"]);
|
|
44
|
+
});
|
|
45
|
+
it("splits long message at newlines", () => {
|
|
46
|
+
const part1 = "a".repeat(Math.floor(MESSAGE_MAX_LENGTH * 0.6));
|
|
47
|
+
const part2 = "b".repeat(Math.floor(MESSAGE_MAX_LENGTH * 0.6));
|
|
48
|
+
const text = part1 + "\n" + part2;
|
|
49
|
+
const result = splitLongMessage(text);
|
|
50
|
+
expect(result.length).toBeGreaterThanOrEqual(2);
|
|
51
|
+
const combined = result.join("");
|
|
52
|
+
expect(combined).toBe(text);
|
|
53
|
+
});
|
|
54
|
+
it("splits uniformly when no newlines", () => {
|
|
55
|
+
const long = "x".repeat(MESSAGE_MAX_LENGTH + 500);
|
|
56
|
+
const result = splitLongMessage(long);
|
|
57
|
+
expect(result.length).toBeGreaterThanOrEqual(2);
|
|
58
|
+
const combined = result.join("");
|
|
59
|
+
expect(combined).toBe(long);
|
|
60
|
+
});
|
|
61
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { existsSync, mkdirSync, unlinkSync, rmdirSync } from "node:fs";
|
|
3
|
+
import { join } from "node:path";
|
|
4
|
+
import { tmpdir } from "node:os";
|
|
5
|
+
import { SessionRegistry } from "../src/session-registry.js";
|
|
6
|
+
describe("SessionRegistry", () => {
|
|
7
|
+
const tmpDir = join(tmpdir(), "pi-feishu-cli-test-registry");
|
|
8
|
+
const registryDir = join(tmpDir, "feishu-im");
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
if (!existsSync(registryDir))
|
|
11
|
+
mkdirSync(registryDir, { recursive: true });
|
|
12
|
+
});
|
|
13
|
+
afterEach(() => {
|
|
14
|
+
try {
|
|
15
|
+
unlinkSync(join(registryDir, "registry.json"));
|
|
16
|
+
}
|
|
17
|
+
catch { }
|
|
18
|
+
try {
|
|
19
|
+
rmdirSync(registryDir);
|
|
20
|
+
}
|
|
21
|
+
catch { }
|
|
22
|
+
try {
|
|
23
|
+
rmdirSync(tmpDir);
|
|
24
|
+
}
|
|
25
|
+
catch { }
|
|
26
|
+
});
|
|
27
|
+
it("creates a new session for a new chat", () => {
|
|
28
|
+
const reg = new SessionRegistry(registryDir);
|
|
29
|
+
const session = reg.ensureSession("oc_chat1");
|
|
30
|
+
expect(session.name).toBe("默认会话");
|
|
31
|
+
expect(session.id).toBeDefined();
|
|
32
|
+
const data = reg.getChatSessions("oc_chat1");
|
|
33
|
+
expect(data.sessions).toHaveLength(1);
|
|
34
|
+
expect(data.active).toBe(session.id);
|
|
35
|
+
});
|
|
36
|
+
it("reuses active session on subsequent calls", () => {
|
|
37
|
+
const reg = new SessionRegistry(registryDir);
|
|
38
|
+
const s1 = reg.ensureSession("oc_chat1");
|
|
39
|
+
const s2 = reg.ensureSession("oc_chat1");
|
|
40
|
+
expect(s2.id).toBe(s1.id);
|
|
41
|
+
const data = reg.getChatSessions("oc_chat1");
|
|
42
|
+
expect(data.sessions).toHaveLength(1);
|
|
43
|
+
});
|
|
44
|
+
it("creates a new session via command", () => {
|
|
45
|
+
const reg = new SessionRegistry(registryDir);
|
|
46
|
+
reg.ensureSession("oc_chat1");
|
|
47
|
+
const s2 = reg.createSession("oc_chat1", "新功能开发");
|
|
48
|
+
expect(s2.name).toBe("新功能开发");
|
|
49
|
+
const data = reg.getChatSessions("oc_chat1");
|
|
50
|
+
expect(data.sessions).toHaveLength(2);
|
|
51
|
+
expect(data.active).toBe(s2.id);
|
|
52
|
+
});
|
|
53
|
+
it("switches active session", () => {
|
|
54
|
+
const reg = new SessionRegistry(registryDir);
|
|
55
|
+
const s1 = reg.ensureSession("oc_chat1");
|
|
56
|
+
const s2 = reg.createSession("oc_chat1", "test2");
|
|
57
|
+
expect(reg.getActiveSessionId("oc_chat1")).toBe(s2.id);
|
|
58
|
+
reg.switchSession("oc_chat1", s1.id);
|
|
59
|
+
expect(reg.getActiveSessionId("oc_chat1")).toBe(s1.id);
|
|
60
|
+
});
|
|
61
|
+
it("deletes a session", () => {
|
|
62
|
+
const reg = new SessionRegistry(registryDir);
|
|
63
|
+
const s1 = reg.ensureSession("oc_chat1");
|
|
64
|
+
const s2 = reg.createSession("oc_chat1", "to-delete");
|
|
65
|
+
reg.deleteSession("oc_chat1", s2.id);
|
|
66
|
+
const data = reg.getChatSessions("oc_chat1");
|
|
67
|
+
expect(data.sessions).toHaveLength(1);
|
|
68
|
+
expect(data.sessions[0].id).toBe(s1.id);
|
|
69
|
+
});
|
|
70
|
+
it("deleting active session switches to another", () => {
|
|
71
|
+
const reg = new SessionRegistry(registryDir);
|
|
72
|
+
const s1 = reg.ensureSession("oc_chat1");
|
|
73
|
+
reg.createSession("oc_chat1", "test2");
|
|
74
|
+
reg.deleteSession("oc_chat1", reg.getActiveSessionId("oc_chat1"));
|
|
75
|
+
expect(reg.getActiveSessionId("oc_chat1")).toBe(s1.id);
|
|
76
|
+
});
|
|
77
|
+
it("persists and loads registry", () => {
|
|
78
|
+
const reg1 = new SessionRegistry(registryDir);
|
|
79
|
+
const s1 = reg1.ensureSession("oc_chat1");
|
|
80
|
+
reg1.createSession("oc_chat1", "second");
|
|
81
|
+
reg1.flush();
|
|
82
|
+
const reg2 = new SessionRegistry(registryDir);
|
|
83
|
+
const data = reg2.getChatSessions("oc_chat1");
|
|
84
|
+
expect(data.sessions).toHaveLength(2);
|
|
85
|
+
expect(data.active).toBeDefined();
|
|
86
|
+
});
|
|
87
|
+
it("returns null for unknown chat", () => {
|
|
88
|
+
const reg = new SessionRegistry(registryDir);
|
|
89
|
+
expect(reg.getChatSessions("nonexistent")).toBeNull();
|
|
90
|
+
expect(reg.getActiveSessionId("nonexistent")).toBeNull();
|
|
91
|
+
});
|
|
92
|
+
});
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
describe("type definitions", () => {
|
|
3
|
+
it("Registry shape", () => {
|
|
4
|
+
const registry = {
|
|
5
|
+
"oc_xxx": {
|
|
6
|
+
sessions: [
|
|
7
|
+
{ id: "sess_1", name: "修 bug", createdAt: 1700000000 },
|
|
8
|
+
],
|
|
9
|
+
active: "sess_1",
|
|
10
|
+
},
|
|
11
|
+
};
|
|
12
|
+
expect(registry["oc_xxx"].sessions).toHaveLength(1);
|
|
13
|
+
expect(registry["oc_xxx"].active).toBe("sess_1");
|
|
14
|
+
});
|
|
15
|
+
it("FeishuImConfig defaults", () => {
|
|
16
|
+
const config = { strategy: "mention", pollInterval: 5 };
|
|
17
|
+
expect(config.strategy).toBe("mention");
|
|
18
|
+
expect(config.autoStart).toBeUndefined();
|
|
19
|
+
});
|
|
20
|
+
it("SessionInfo fields", () => {
|
|
21
|
+
const info = { id: "abc", name: "test", createdAt: 123 };
|
|
22
|
+
expect(info.id).toBe("abc");
|
|
23
|
+
expect(info.name).toBe("test");
|
|
24
|
+
expect(info.createdAt).toBe(123);
|
|
25
|
+
});
|
|
26
|
+
it("ChatSessions active can be null", () => {
|
|
27
|
+
const cs = { sessions: [], active: null };
|
|
28
|
+
expect(cs.active).toBeNull();
|
|
29
|
+
});
|
|
30
|
+
});
|