Please enter the commit message for your changes. Lines starting
with '#' will be ignored, and an empty message aborts the commit. On branch main Your branch is up to date with 'origin/main'. Changes to be committed: new file: .claude/skills/algorithmic-art/.openskills.json new file: .claude/skills/algorithmic-art/LICENSE.txt new file: .claude/skills/algorithmic-art/SKILL.md new file: .claude/skills/algorithmic-art/templates/generator_template.js new file: .claude/skills/algorithmic-art/templates/viewer.html new file: .claude/skills/brand-guidelines/.openskills.json new file: .claude/skills/brand-guidelines/LICENSE.txt new file: .claude/skills/brand-guidelines/SKILL.md new file: .claude/skills/canvas-design/.openskills.json new file: .claude/skills/canvas-design/LICENSE.txt new file: .claude/skills/canvas-design/SKILL.md new file: .claude/skills/canvas-design/canvas-fonts/ArsenalSC-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/ArsenalSC-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/BigShoulders-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/BigShoulders-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/BigShoulders-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Boldonse-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Boldonse-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/BricolageGrotesque-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/BricolageGrotesque-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/BricolageGrotesque-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/CrimsonPro-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/CrimsonPro-Italic.ttf new file: .claude/skills/canvas-design/canvas-fonts/CrimsonPro-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/CrimsonPro-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/DMMono-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/DMMono-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/EricaOne-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/EricaOne-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/GeistMono-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/GeistMono-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/GeistMono-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Gloock-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Gloock-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexMono-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexMono-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexMono-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexSerif-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexSerif-BoldItalic.ttf new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexSerif-Italic.ttf new file: .claude/skills/canvas-design/canvas-fonts/IBMPlexSerif-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSans-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSans-BoldItalic.ttf new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSans-Italic.ttf new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSans-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSans-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSerif-Italic.ttf new file: .claude/skills/canvas-design/canvas-fonts/InstrumentSerif-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Italiana-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Italiana-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/JetBrainsMono-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/JetBrainsMono-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/JetBrainsMono-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Jura-Light.ttf new file: .claude/skills/canvas-design/canvas-fonts/Jura-Medium.ttf new file: .claude/skills/canvas-design/canvas-fonts/Jura-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/LibreBaskerville-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/LibreBaskerville-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Lora-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/Lora-BoldItalic.ttf new file: .claude/skills/canvas-design/canvas-fonts/Lora-Italic.ttf new file: .claude/skills/canvas-design/canvas-fonts/Lora-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Lora-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/NationalPark-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/NationalPark-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/NationalPark-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/NothingYouCouldDo-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/NothingYouCouldDo-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Outfit-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/Outfit-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Outfit-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/PixelifySans-Medium.ttf new file: .claude/skills/canvas-design/canvas-fonts/PixelifySans-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/PoiretOne-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/PoiretOne-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/RedHatMono-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/RedHatMono-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/RedHatMono-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/Silkscreen-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Silkscreen-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/SmoochSans-Medium.ttf new file: .claude/skills/canvas-design/canvas-fonts/SmoochSans-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Tektur-Medium.ttf new file: .claude/skills/canvas-design/canvas-fonts/Tektur-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/Tektur-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/WorkSans-Bold.ttf new file: .claude/skills/canvas-design/canvas-fonts/WorkSans-BoldItalic.ttf new file: .claude/skills/canvas-design/canvas-fonts/WorkSans-Italic.ttf new file: .claude/skills/canvas-design/canvas-fonts/WorkSans-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/WorkSans-Regular.ttf new file: .claude/skills/canvas-design/canvas-fonts/YoungSerif-OFL.txt new file: .claude/skills/canvas-design/canvas-fonts/YoungSerif-Regular.ttf new file: .claude/skills/doc-coauthoring/.openskills.json new file: .claude/skills/doc-coauthoring/SKILL.md new file: .claude/skills/docx/.openskills.json new file: .claude/skills/docx/LICENSE.txt new file: .claude/skills/docx/SKILL.md new file: .claude/skills/docx/scripts/__init__.py new file: .claude/skills/docx/scripts/accept_changes.py new file: .claude/skills/docx/scripts/comment.py new file: .claude/skills/docx/scripts/office/helpers/__init__.py new file: .claude/skills/docx/scripts/office/helpers/merge_runs.py new file: .claude/skills/docx/scripts/office/helpers/simplify_redlines.py new file: .claude/skills/docx/scripts/office/pack.py new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd new file: .claude/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd new file: .claude/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd new file: .claude/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd new file: .claude/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd new file: .claude/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd new file: .claude/skills/docx/scripts/office/schemas/mce/mc.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-2010.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-2012.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-2018.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-cex-2018.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-cid-2016.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd new file: .claude/skills/docx/scripts/office/schemas/microsoft/wml-symex-2015.xsd new file: .claude/skills/docx/scripts/office/soffice.py new file: .claude/skills/docx/scripts/office/unpack.py new file: .claude/skills/docx/scripts/office/validate.py new file: .claude/skills/docx/scripts/office/validators/__init__.py new file: .claude/skills/docx/scripts/office/validators/base.py new file: .claude/skills/docx/scripts/office/validators/docx.py new file: .claude/skills/docx/scripts/office/validators/pptx.py new file: .claude/skills/docx/scripts/office/validators/redlining.py new file: .claude/skills/docx/scripts/templates/comments.xml new file: .claude/skills/docx/scripts/templates/commentsExtended.xml new file: .claude/skills/docx/scripts/templates/commentsExtensible.xml new file: .claude/skills/docx/scripts/templates/commentsIds.xml new file: .claude/skills/docx/scripts/templates/people.xml new file: .claude/skills/frontend-design/.openskills.json new file: .claude/skills/frontend-design/LICENSE.txt new file: .claude/skills/frontend-design/SKILL.md new file: .claude/skills/internal-comms/.openskills.json new file: .claude/skills/internal-comms/LICENSE.txt new file: .claude/skills/internal-comms/SKILL.md new file: .claude/skills/internal-comms/examples/3p-updates.md new file: .claude/skills/internal-comms/examples/company-newsletter.md new file: .claude/skills/internal-comms/examples/faq-answers.md new file: .claude/skills/internal-comms/examples/general-comms.md new file: .claude/skills/mcp-builder/.openskills.json new file: .claude/skills/mcp-builder/LICENSE.txt new file: .claude/skills/mcp-builder/SKILL.md new file: .claude/skills/mcp-builder/reference/evaluation.md new file: .claude/skills/mcp-builder/reference/mcp_best_practices.md new file: .claude/skills/mcp-builder/reference/node_mcp_server.md new file: .claude/skills/mcp-builder/reference/python_mcp_server.md new file: .claude/skills/mcp-builder/scripts/connections.py new file: .claude/skills/mcp-builder/scripts/evaluation.py new file: .claude/skills/mcp-builder/scripts/example_evaluation.xml new file: .claude/skills/mcp-builder/scripts/requirements.txt new file: .claude/skills/pdf/.openskills.json new file: .claude/skills/pdf/LICENSE.txt new file: .claude/skills/pdf/SKILL.md new file: .claude/skills/pdf/forms.md new file: .claude/skills/pdf/reference.md new file: .claude/skills/pdf/scripts/check_bounding_boxes.py new file: .claude/skills/pdf/scripts/check_fillable_fields.py new file: .claude/skills/pdf/scripts/convert_pdf_to_images.py new file: .claude/skills/pdf/scripts/create_validation_image.py new file: .claude/skills/pdf/scripts/extract_form_field_info.py new file: .claude/skills/pdf/scripts/extract_form_structure.py new file: .claude/skills/pdf/scripts/fill_fillable_fields.py new file: .claude/skills/pdf/scripts/fill_pdf_form_with_annotations.py new file: .claude/skills/pptx/.openskills.json new file: .claude/skills/pptx/LICENSE.txt new file: .claude/skills/pptx/SKILL.md new file: .claude/skills/pptx/editing.md new file: .claude/skills/pptx/pptxgenjs.md new file: .claude/skills/pptx/scripts/__init__.py new file: .claude/skills/pptx/scripts/add_slide.py new file: .claude/skills/pptx/scripts/clean.py new file: .claude/skills/pptx/scripts/office/helpers/__init__.py new file: .claude/skills/pptx/scripts/office/helpers/merge_runs.py new file: .claude/skills/pptx/scripts/office/helpers/simplify_redlines.py new file: .claude/skills/pptx/scripts/office/pack.py new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd new file: .claude/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd new file: .claude/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd new file: .claude/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd new file: .claude/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd new file: .claude/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd new file: .claude/skills/pptx/scripts/office/schemas/mce/mc.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-2010.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-2012.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-2018.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-cex-2018.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-cid-2016.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd new file: .claude/skills/pptx/scripts/office/schemas/microsoft/wml-symex-2015.xsd new file: .claude/skills/pptx/scripts/office/soffice.py new file: .claude/skills/pptx/scripts/office/unpack.py new file: .claude/skills/pptx/scripts/office/validate.py new file: .claude/skills/pptx/scripts/office/validators/__init__.py new file: .claude/skills/pptx/scripts/office/validators/base.py new file: .claude/skills/pptx/scripts/office/validators/docx.py new file: .claude/skills/pptx/scripts/office/validators/pptx.py new file: .claude/skills/pptx/scripts/office/validators/redlining.py new file: .claude/skills/pptx/scripts/thumbnail.py new file: .claude/skills/skill-creator/.openskills.json new file: .claude/skills/skill-creator/LICENSE.txt new file: .claude/skills/skill-creator/SKILL.md new file: .claude/skills/skill-creator/agents/analyzer.md new file: .claude/skills/skill-creator/agents/comparator.md new file: .claude/skills/skill-creator/agents/grader.md new file: .claude/skills/skill-creator/assets/eval_review.html new file: .claude/skills/skill-creator/eval-viewer/generate_review.py new file: .claude/skills/skill-creator/eval-viewer/viewer.html new file: .claude/skills/skill-creator/references/schemas.md new file: .claude/skills/skill-creator/scripts/__init__.py new file: .claude/skills/skill-creator/scripts/aggregate_benchmark.py new file: .claude/skills/skill-creator/scripts/generate_report.py new file: .claude/skills/skill-creator/scripts/improve_description.py new file: .claude/skills/skill-creator/scripts/package_skill.py new file: .claude/skills/skill-creator/scripts/quick_validate.py new file: .claude/skills/skill-creator/scripts/run_eval.py new file: .claude/skills/skill-creator/scripts/run_loop.py new file: .claude/skills/skill-creator/scripts/utils.py new file: .claude/skills/slack-gif-creator/.openskills.json new file: .claude/skills/slack-gif-creator/LICENSE.txt new file: .claude/skills/slack-gif-creator/SKILL.md new file: .claude/skills/slack-gif-creator/core/easing.py new file: .claude/skills/slack-gif-creator/core/frame_composer.py new file: .claude/skills/slack-gif-creator/core/gif_builder.py new file: .claude/skills/slack-gif-creator/core/validators.py new file: .claude/skills/slack-gif-creator/requirements.txt new file: .claude/skills/template/.openskills.json new file: .claude/skills/template/SKILL.md new file: .claude/skills/theme-factory/.openskills.json new file: .claude/skills/theme-factory/LICENSE.txt new file: .claude/skills/theme-factory/SKILL.md new file: .claude/skills/theme-factory/theme-showcase.pdf new file: .claude/skills/theme-factory/themes/arctic-frost.md new file: .claude/skills/theme-factory/themes/botanical-garden.md new file: .claude/skills/theme-factory/themes/desert-rose.md new file: .claude/skills/theme-factory/themes/forest-canopy.md new file: .claude/skills/theme-factory/themes/golden-hour.md new file: .claude/skills/theme-factory/themes/midnight-galaxy.md new file: .claude/skills/theme-factory/themes/modern-minimalist.md new file: .claude/skills/theme-factory/themes/ocean-depths.md new file: .claude/skills/theme-factory/themes/sunset-boulevard.md new file: .claude/skills/theme-factory/themes/tech-innovation.md new file: .claude/skills/web-artifacts-builder/.openskills.json new file: .claude/skills/web-artifacts-builder/LICENSE.txt new file: .claude/skills/web-artifacts-builder/SKILL.md new file: .claude/skills/web-artifacts-builder/scripts/bundle-artifact.sh new file: .claude/skills/web-artifacts-builder/scripts/init-artifact.sh new file: .claude/skills/web-artifacts-builder/scripts/shadcn-components.tar.gz new file: .claude/skills/webapp-testing/.openskills.json new file: .claude/skills/webapp-testing/LICENSE.txt new file: .claude/skills/webapp-testing/SKILL.md new file: .claude/skills/webapp-testing/examples/console_logging.py new file: .claude/skills/webapp-testing/examples/element_discovery.py new file: .claude/skills/webapp-testing/examples/static_html_automation.py new file: .claude/skills/webapp-testing/scripts/with_server.py new file: .claude/skills/xlsx/.openskills.json new file: .claude/skills/xlsx/LICENSE.txt new file: .claude/skills/xlsx/SKILL.md new file: .claude/skills/xlsx/scripts/office/helpers/__init__.py new file: .claude/skills/xlsx/scripts/office/helpers/merge_runs.py new file: .claude/skills/xlsx/scripts/office/helpers/simplify_redlines.py new file: .claude/skills/xlsx/scripts/office/pack.py new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd new file: .claude/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd new file: .claude/skills/xlsx/scripts/office/schemas/mce/mc.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-2010.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-2012.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-2018.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-cex-2018.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-cid-2016.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd new file: .claude/skills/xlsx/scripts/office/schemas/microsoft/wml-symex-2015.xsd new file: .claude/skills/xlsx/scripts/office/soffice.py new file: .claude/skills/xlsx/scripts/office/unpack.py new file: .claude/skills/xlsx/scripts/office/validate.py new file: .claude/skills/xlsx/scripts/office/validators/__init__.py new file: .claude/skills/xlsx/scripts/office/validators/base.py new file: .claude/skills/xlsx/scripts/office/validators/docx.py new file: .claude/skills/xlsx/scripts/office/validators/pptx.py new file: .claude/skills/xlsx/scripts/office/validators/redlining.py new file: .claude/skills/xlsx/scripts/recalc.py new file: .env.example new file: .gitignore new file: config/mcp.json new file: config/models.json new file: config/personalities.json new file: docs/AGENTS.md new file: docs/AI_IMPLEMENTATION.md new file: docs/AI_INTEGRATION_COMPLETE.md new file: docs/AI_QUICKSTART.md new file: docs/AI_SUMMARY.md new file: docs/CHANGELOG.md new file: docs/CONFIG_GUIDE.md new file: docs/FIXES.md new file: docs/PROJECT_REFACTOR.md new file: docs/README.md new file: docs/README_INDEX.md new file: examples/ai_example.py new file: main.py new file: pytest.ini new file: requirements.txt new file: scripts/migrate_to_vector_db.py new file: skills/cmd_zip_skill/README.md new file: skills/cmd_zip_skill/__init__.py new file: skills/cmd_zip_skill/main.py new file: skills/cmd_zip_skill/skill.json new file: skills/cmd_zip_skill_1772465404375/README.md new file: skills/cmd_zip_skill_1772465404375/__init__.py new file: skills/cmd_zip_skill_1772465404375/main.py new file: skills/cmd_zip_skill_1772465404375/skill.json new file: skills/cmd_zip_skill_1772465434774/README.md new file: skills/cmd_zip_skill_1772465434774/__init__.py new file: skills/cmd_zip_skill_1772465434774/main.py new file: skills/cmd_zip_skill_1772465434774/skill.json new file: skills/cmd_zip_skill_1772465467809/README.md new file: skills/cmd_zip_skill_1772465467809/__init__.py new file: skills/cmd_zip_skill_1772465467809/main.py new file: skills/cmd_zip_skill_1772465467809/skill.json new file: skills/cmd_zip_skill_1772465652075/README.md new file: skills/cmd_zip_skill_1772465652075/__init__.py new file: skills/cmd_zip_skill_1772465652075/main.py new file: skills/cmd_zip_skill_1772465652075/skill.json new file: skills/cmd_zip_skill_1772465685352/README.md new file: skills/cmd_zip_skill_1772465685352/__init__.py new file: skills/cmd_zip_skill_1772465685352/main.py new file: skills/cmd_zip_skill_1772465685352/skill.json new file: skills/cmd_zip_skill_1772465936294/README.md new file: skills/cmd_zip_skill_1772465936294/__init__.py new file: skills/cmd_zip_skill_1772465936294/main.py new file: skills/cmd_zip_skill_1772465936294/skill.json new file: skills/cmd_zip_skill_1772465966322/README.md new file: skills/cmd_zip_skill_1772465966322/__init__.py new file: skills/cmd_zip_skill_1772465966322/main.py new file: skills/cmd_zip_skill_1772465966322/skill.json new file: skills/cmd_zip_skill_1772466071278/README.md new file: skills/cmd_zip_skill_1772466071278/__init__.py new file: skills/cmd_zip_skill_1772466071278/main.py new file: skills/cmd_zip_skill_1772466071278/skill.json new file: skills/skills_creator/README.md new file: skills/skills_creator/__init__.py new file: skills/skills_creator/main.py new file: skills/skills_creator/skill.json new file: src/__init__.py new file: src/ai/__init__.py new file: src/ai/base.py new file: src/ai/client.py new file: src/ai/docs/README.md new file: src/ai/mcp/__init__.py new file: src/ai/mcp/base.py new file: src/ai/mcp/servers/__init__.py new file: src/ai/mcp/servers/filesystem.py new file: src/ai/memory.py new file: src/ai/models/__init__.py new file: src/ai/models/anthropic_model.py new file: src/ai/models/openai_model.py new file: src/ai/personality.py new file: src/ai/skills/__init__.py new file: src/ai/skills/base.py new file: src/ai/task_manager.py new file: src/ai/vector_store/__init__.py new file: src/ai/vector_store/base.py new file: src/ai/vector_store/chroma_store.py new file: src/ai/vector_store/json_store.py new file: src/core/__init__.py new file: src/core/bot.py new file: src/core/config.py new file: src/handlers/__init__.py new file: src/handlers/message_handler.py new file: src/handlers/message_handler_ai.py new file: src/utils/__init__.py new file: src/utils/logger.py new file: start.bat new file: tests/test_ai.py
This commit is contained in:
475
src/ai/client.py
Normal file
475
src/ai/client.py
Normal file
@@ -0,0 +1,475 @@
|
||||
"""
|
||||
AI瀹㈡埛绔?- 鏁村悎鎵€鏈堿I鍔熻兘
|
||||
"""
|
||||
import inspect
|
||||
import json
|
||||
import re
|
||||
from typing import List, Optional, Dict, Any, AsyncIterator, Tuple
|
||||
from pathlib import Path
|
||||
from .base import ModelConfig, ModelProvider, Message, ToolRegistry
|
||||
from .models import OpenAIModel, AnthropicModel
|
||||
from .personality import PersonalitySystem
|
||||
from .memory import MemorySystem
|
||||
from .task_manager import LongTaskManager
|
||||
from src.utils.logger import setup_logger
|
||||
|
||||
logger = setup_logger('AIClient')
|
||||
|
||||
|
||||
class AIClient:
|
||||
"""AI瀹㈡埛绔?- 缁熶竴鎺ュ彛"""
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
model_config: ModelConfig,
|
||||
embed_config: Optional[ModelConfig] = None,
|
||||
data_dir: Path = Path("data/ai"),
|
||||
use_vector_db: bool = True
|
||||
):
|
||||
self.config = model_config
|
||||
self.data_dir = data_dir
|
||||
self.data_dir.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
# 初始化主模型
|
||||
self.model = self._create_model(model_config)
|
||||
|
||||
# 初始化嵌入模型(如果提供)
|
||||
self.embed_model = None
|
||||
if embed_config:
|
||||
self.embed_model = self._create_model(embed_config)
|
||||
logger.info(
|
||||
f"嵌入模型初始化完成: {embed_config.provider.value}/{embed_config.model_name}"
|
||||
)
|
||||
|
||||
# 初始化工具注册表
|
||||
self.tools = ToolRegistry()
|
||||
|
||||
# 初始化人格系统
|
||||
self.personality = PersonalitySystem(
|
||||
config_path=data_dir / "personalities.json"
|
||||
)
|
||||
|
||||
# 初始化记忆系统
|
||||
self.memory = MemorySystem(
|
||||
storage_path=data_dir / "long_term_memory.json",
|
||||
embed_func=self._embed_wrapper,
|
||||
importance_evaluator=self._evaluate_memory_importance,
|
||||
use_vector_db=use_vector_db
|
||||
)
|
||||
|
||||
# 初始化长任务管理器
|
||||
self.task_manager = LongTaskManager(
|
||||
storage_path=data_dir / "tasks.json"
|
||||
)
|
||||
|
||||
logger.info(
|
||||
f"AI 客户端初始化完成: {model_config.provider.value}/{model_config.model_name}"
|
||||
)
|
||||
|
||||
def _create_model(self, config: ModelConfig):
|
||||
"""创建模型实例。"""
|
||||
if config.provider == ModelProvider.OPENAI:
|
||||
return OpenAIModel(config)
|
||||
elif config.provider == ModelProvider.ANTHROPIC:
|
||||
return AnthropicModel(config)
|
||||
elif config.provider in [ModelProvider.DEEPSEEK, ModelProvider.QWEN]:
|
||||
# DeepSeek 和 Qwen 使用 OpenAI 兼容接口
|
||||
return OpenAIModel(config)
|
||||
else:
|
||||
raise ValueError(f"不支持的模型提供商: {config.provider}")
|
||||
|
||||
async def _embed_wrapper(self, text: str) -> List[float]:
|
||||
"""嵌入向量包装器。"""
|
||||
try:
|
||||
# 如果有独立的嵌入模型,优先使用
|
||||
if self.embed_model:
|
||||
return await self.embed_model.embed(text)
|
||||
# 否则尝试使用主模型
|
||||
return await self.model.embed(text)
|
||||
except NotImplementedError:
|
||||
# 如果都不支持嵌入,返回 None(记忆系统会降级)
|
||||
logger.warning("Current model does not support embeddings; vector retrieval disabled")
|
||||
return None
|
||||
except Exception as e:
|
||||
logger.error(f"生成嵌入向量失败: {e}")
|
||||
return None
|
||||
|
||||
@staticmethod
|
||||
def _parse_importance_score(raw: str) -> float:
|
||||
text = (raw or "").strip()
|
||||
if not text:
|
||||
raise ValueError("empty importance response")
|
||||
|
||||
try:
|
||||
parsed = json.loads(text)
|
||||
if isinstance(parsed, (int, float)):
|
||||
return float(parsed)
|
||||
if isinstance(parsed, dict):
|
||||
for key in ["importance", "score", "value"]:
|
||||
if key in parsed:
|
||||
return float(parsed[key])
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
match = re.search(r"-?\d+(?:\.\d+)?", text)
|
||||
if not match:
|
||||
raise ValueError(f"cannot parse importance score: {text}")
|
||||
return float(match.group(0))
|
||||
|
||||
async def _evaluate_memory_importance(
|
||||
self, content: str, metadata: Optional[Dict] = None
|
||||
) -> float:
|
||||
"""
|
||||
调用主模型评估记忆重要性,返回 [0, 1] 分值。
|
||||
"""
|
||||
system_prompt = (
|
||||
"你是记忆重要性评估器。请根据输入内容判断该信息是否值得长期记忆。"
|
||||
"输出一个 0 到 1 的数字,数字越大表示越重要。"
|
||||
"只输出数字,不要输出任何解释、单位或多余文本。"
|
||||
)
|
||||
payload = json.dumps(
|
||||
{"content": content, "metadata": metadata or {}},
|
||||
ensure_ascii=False,
|
||||
)
|
||||
messages = [
|
||||
Message(role="system", content=system_prompt),
|
||||
Message(role="user", content=payload),
|
||||
]
|
||||
|
||||
try:
|
||||
response = await self.model.chat(
|
||||
messages=messages,
|
||||
tools=None,
|
||||
temperature=0.0,
|
||||
max_tokens=16,
|
||||
)
|
||||
score = self._parse_importance_score(response.content)
|
||||
return max(0.0, min(1.0, score))
|
||||
except Exception as e:
|
||||
logger.warning(f"memory importance evaluation failed, fallback to neutral score: {e}")
|
||||
return 0.5
|
||||
|
||||
async def chat(
|
||||
self,
|
||||
user_id: str,
|
||||
user_message: str,
|
||||
system_prompt: Optional[str] = None,
|
||||
use_memory: bool = True,
|
||||
use_tools: bool = True,
|
||||
stream: bool = False,
|
||||
**kwargs
|
||||
) -> str:
|
||||
"""对话接口。"""
|
||||
try:
|
||||
# 构建消息列表
|
||||
messages = []
|
||||
|
||||
# 系统提示词
|
||||
if system_prompt is None:
|
||||
system_prompt = self.personality.get_system_prompt()
|
||||
|
||||
# 注入记忆上下文
|
||||
if use_memory:
|
||||
short_term, long_term = await self.memory.get_context(
|
||||
user_id=user_id,
|
||||
query=user_message
|
||||
)
|
||||
|
||||
if short_term or long_term:
|
||||
memory_context = self.memory.format_context(short_term, long_term)
|
||||
system_prompt += f"\n\n{memory_context}"
|
||||
|
||||
messages.append(Message(role="system", content=system_prompt))
|
||||
|
||||
# 添加用户消息
|
||||
messages.append(Message(role="user", content=user_message))
|
||||
|
||||
# 准备工具
|
||||
tools = None
|
||||
if use_tools and self.tools.list():
|
||||
tools = self.tools.to_openai_format()
|
||||
|
||||
# 调用模型
|
||||
if stream:
|
||||
return self._chat_stream(messages, tools, **kwargs)
|
||||
else:
|
||||
response = await self.model.chat(messages, tools, **kwargs)
|
||||
|
||||
# 处理工具调用
|
||||
if response.tool_calls:
|
||||
response = await self._handle_tool_calls(
|
||||
messages, response, tools, **kwargs
|
||||
)
|
||||
|
||||
# 写入记忆
|
||||
if use_memory:
|
||||
stored_memory = await self.memory.add_qa_pair(
|
||||
user_id=user_id,
|
||||
question=user_message,
|
||||
answer=response.content,
|
||||
metadata={"source": "chat"},
|
||||
)
|
||||
if stored_memory:
|
||||
logger.info(
|
||||
"已写入长期记忆问答对:\n"
|
||||
f"{stored_memory.content}\n"
|
||||
f"memory_id={stored_memory.id}, "
|
||||
f"importance={stored_memory.importance:.2f}"
|
||||
)
|
||||
|
||||
return response.content
|
||||
|
||||
except Exception as e:
|
||||
logger.error(f"对话失败: {e}")
|
||||
raise
|
||||
|
||||
async def _chat_stream(
|
||||
self,
|
||||
messages: List[Message],
|
||||
tools: Optional[List[Dict]],
|
||||
**kwargs
|
||||
) -> AsyncIterator[str]:
|
||||
"""流式对话。"""
|
||||
async for chunk in self.model.chat_stream(messages, tools, **kwargs):
|
||||
yield chunk
|
||||
|
||||
async def _handle_tool_calls(
|
||||
self,
|
||||
messages: List[Message],
|
||||
response: Message,
|
||||
tools: Optional[List[Dict]],
|
||||
**kwargs
|
||||
) -> Message:
|
||||
"""处理工具调用。"""
|
||||
messages.append(response)
|
||||
|
||||
# 鎵ц宸ュ叿璋冪敤
|
||||
for tool_call in response.tool_calls or []:
|
||||
try:
|
||||
tool_name, tool_args, tool_call_id = self._parse_tool_call(tool_call)
|
||||
except Exception as e:
|
||||
logger.warning(f"解析工具调用失败: {e}")
|
||||
fallback_id = tool_call.get('id') if isinstance(tool_call, dict) else getattr(tool_call, 'id', None)
|
||||
if fallback_id:
|
||||
messages.append(Message(
|
||||
role="tool",
|
||||
content=f"工具参数解析失败: {str(e)}",
|
||||
tool_call_id=fallback_id
|
||||
))
|
||||
continue
|
||||
if not tool_name:
|
||||
logger.warning(f"跳过无效工具调用: {tool_call}")
|
||||
continue
|
||||
|
||||
tool_def = self.tools.get(tool_name)
|
||||
if not tool_def:
|
||||
error_msg = f"未找到工具: {tool_name}"
|
||||
logger.warning(error_msg)
|
||||
if tool_call_id:
|
||||
messages.append(Message(
|
||||
role="tool",
|
||||
content=error_msg,
|
||||
tool_call_id=tool_call_id
|
||||
))
|
||||
continue
|
||||
|
||||
try:
|
||||
result = tool_def.function(**tool_args)
|
||||
if inspect.isawaitable(result):
|
||||
result = await result
|
||||
if tool_call_id:
|
||||
messages.append(Message(
|
||||
role="tool",
|
||||
content=str(result),
|
||||
tool_call_id=tool_call_id
|
||||
))
|
||||
except Exception as e:
|
||||
if tool_call_id:
|
||||
messages.append(Message(
|
||||
role="tool",
|
||||
content=f"工具执行失败: {str(e)}",
|
||||
tool_call_id=tool_call_id
|
||||
))
|
||||
|
||||
# 再次调用模型获取最终响应
|
||||
return await self.model.chat(messages, tools, **kwargs)
|
||||
|
||||
def _parse_tool_call(self, tool_call: Any) -> Tuple[Optional[str], Dict[str, Any], Optional[str]]:
|
||||
"""兼容不同 SDK 返回的工具调用结构。"""
|
||||
if isinstance(tool_call, dict):
|
||||
tool_call_id = tool_call.get('id')
|
||||
function = tool_call.get('function') or {}
|
||||
tool_name = function.get('name')
|
||||
raw_args = function.get('arguments')
|
||||
else:
|
||||
tool_call_id = getattr(tool_call, 'id', None)
|
||||
function = getattr(tool_call, 'function', None)
|
||||
tool_name = getattr(function, 'name', None) if function else None
|
||||
raw_args = getattr(function, 'arguments', None) if function else None
|
||||
|
||||
tool_args = self._normalize_tool_args(raw_args)
|
||||
return tool_name, tool_args, tool_call_id
|
||||
|
||||
def _normalize_tool_args(self, raw_args: Any) -> Dict[str, Any]:
|
||||
"""将工具参数统一转换为字典。"""
|
||||
if raw_args is None:
|
||||
return {}
|
||||
|
||||
if isinstance(raw_args, dict):
|
||||
return raw_args
|
||||
|
||||
if isinstance(raw_args, str):
|
||||
raw_args = raw_args.strip()
|
||||
if not raw_args:
|
||||
return {}
|
||||
parsed = json.loads(raw_args)
|
||||
if not isinstance(parsed, dict):
|
||||
raise ValueError(f"工具参数必须是 JSON 对象,实际类型: {type(parsed)}")
|
||||
return parsed
|
||||
|
||||
if hasattr(raw_args, 'model_dump'):
|
||||
parsed = raw_args.model_dump()
|
||||
if isinstance(parsed, dict):
|
||||
return parsed
|
||||
|
||||
raise ValueError(f"不支持的工具参数类型: {type(raw_args)}")
|
||||
|
||||
def set_personality(self, personality_name: str) -> bool:
|
||||
"""设置人格。"""
|
||||
return self.personality.set_personality(personality_name)
|
||||
|
||||
def list_personalities(self) -> List[str]:
|
||||
"""列出所有人格。"""
|
||||
return self.personality.list_personalities()
|
||||
|
||||
def switch_model(self, model_config: ModelConfig) -> bool:
|
||||
"""Runtime switch for primary chat model."""
|
||||
new_model = self._create_model(model_config)
|
||||
self.model = new_model
|
||||
self.config = model_config
|
||||
logger.info(
|
||||
f"已切换主模型: {model_config.provider.value}/{model_config.model_name}"
|
||||
)
|
||||
return True
|
||||
|
||||
async def create_long_task(
|
||||
self,
|
||||
user_id: str,
|
||||
title: str,
|
||||
description: str,
|
||||
steps: List[Dict],
|
||||
metadata: Optional[Dict] = None
|
||||
) -> str:
|
||||
"""创建长任务。"""
|
||||
return self.task_manager.create_task(
|
||||
user_id=user_id,
|
||||
title=title,
|
||||
description=description,
|
||||
steps=steps,
|
||||
metadata=metadata
|
||||
)
|
||||
|
||||
async def start_task(
|
||||
self,
|
||||
task_id: str,
|
||||
progress_callback: Optional[callable] = None
|
||||
):
|
||||
"""启动任务。"""
|
||||
await self.task_manager.start_task(task_id, progress_callback)
|
||||
|
||||
def get_task_status(self, task_id: str) -> Optional[Dict]:
|
||||
"""获取任务状态。"""
|
||||
return self.task_manager.get_task_status(task_id)
|
||||
|
||||
def register_tool(self, name: str, description: str, parameters: Dict, function: callable):
|
||||
"""注册工具。"""
|
||||
from .base import ToolDefinition
|
||||
tool = ToolDefinition(
|
||||
name=name,
|
||||
description=description,
|
||||
parameters=parameters,
|
||||
function=function
|
||||
)
|
||||
self.tools.register(tool)
|
||||
logger.info(f"已注册工具: {name}")
|
||||
|
||||
def unregister_tool(self, name: str) -> bool:
|
||||
"""卸载工具。"""
|
||||
removed = self.tools.unregister(name)
|
||||
if removed:
|
||||
logger.info(f"已卸载工具: {name}")
|
||||
return removed
|
||||
|
||||
def unregister_tools_by_prefix(self, prefix: str) -> int:
|
||||
"""按前缀批量卸载工具。"""
|
||||
removed_count = self.tools.unregister_by_prefix(prefix)
|
||||
if removed_count:
|
||||
logger.info(f"Unregistered tools by prefix {prefix}: {removed_count}")
|
||||
return removed_count
|
||||
|
||||
def clear_memory(self, user_id: str):
|
||||
"""清除用户短期记忆。"""
|
||||
self.memory.clear_short_term(user_id)
|
||||
logger.info(f"Cleared short-term memory for user {user_id}")
|
||||
|
||||
async def clear_long_term_memory(self, user_id: str) -> bool:
|
||||
try:
|
||||
await self.memory.clear_long_term(user_id)
|
||||
logger.info(f"Cleared long-term memory for user {user_id}")
|
||||
return True
|
||||
except Exception as e:
|
||||
logger.warning(f"Failed to clear long-term memory for user {user_id}: {e}")
|
||||
return False
|
||||
|
||||
async def list_long_term_memories(self, user_id: str, limit: int = 20):
|
||||
return await self.memory.list_long_term(user_id, limit=limit)
|
||||
|
||||
async def get_long_term_memory(self, user_id: str, memory_id: str):
|
||||
return await self.memory.get_long_term(user_id, memory_id)
|
||||
|
||||
async def add_long_term_memory(
|
||||
self,
|
||||
user_id: str,
|
||||
content: str,
|
||||
importance: float = 0.8,
|
||||
metadata: Optional[Dict] = None,
|
||||
):
|
||||
return await self.memory.add_long_term(
|
||||
user_id=user_id,
|
||||
content=content,
|
||||
importance=importance,
|
||||
metadata=metadata,
|
||||
)
|
||||
|
||||
async def search_long_term_memories(
|
||||
self, user_id: str, query: str, limit: int = 10
|
||||
):
|
||||
return await self.memory.search_long_term(user_id, query=query, limit=limit)
|
||||
|
||||
async def update_long_term_memory(
|
||||
self,
|
||||
user_id: str,
|
||||
memory_id: str,
|
||||
content: Optional[str] = None,
|
||||
importance: Optional[float] = None,
|
||||
metadata: Optional[Dict] = None,
|
||||
):
|
||||
return await self.memory.update_long_term(
|
||||
user_id=user_id,
|
||||
memory_id=memory_id,
|
||||
content=content,
|
||||
importance=importance,
|
||||
metadata=metadata,
|
||||
)
|
||||
|
||||
async def delete_long_term_memory(self, user_id: str, memory_id: str) -> bool:
|
||||
return await self.memory.delete_long_term(user_id, memory_id)
|
||||
|
||||
async def clear_all_memory(self, user_id: str) -> bool:
|
||||
"""清除用户全部记忆(短期 + 长期)。"""
|
||||
self.clear_memory(user_id)
|
||||
try:
|
||||
return await self.clear_long_term_memory(user_id)
|
||||
except Exception:
|
||||
return False
|
||||
|
||||
Reference in New Issue
Block a user