From 1b3a3407cba08ed577dc8878376154311c6fe78d Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Tue, 18 Nov 2025 21:16:15 -0800 Subject: [PATCH 01/17] Add Anthropic Tools Common Format --- .../providers/anthropic/transforms.rb | 82 +++++ .../test_agent_common_format_input_schema.yml | 209 +++++++++++ ...est_agent_common_format_multiple_tools.yml | 206 +++++++++++ .../test_agent_common_format_parameters.yml | 209 +++++++++++ ...t_agent_common_format_tool_choice_auto.yml | 105 ++++++ ...ent_common_format_tool_choice_required.yml | 207 +++++++++++ ...ent_common_format_tool_choice_specific.yml | 207 +++++++++++ .../anthropic/common_format/tools_test.rb | 347 ++++++++++++++++++ 8 files changed, 1572 insertions(+) create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_input_schema.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_multiple_tools.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_parameters.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_required.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml create mode 100644 test/integration/anthropic/common_format/tools_test.rb diff --git a/lib/active_agent/providers/anthropic/transforms.rb b/lib/active_agent/providers/anthropic/transforms.rb index e0bccb13..e1c6a736 100644 --- a/lib/active_agent/providers/anthropic/transforms.rb +++ b/lib/active_agent/providers/anthropic/transforms.rb @@ -28,9 +28,90 @@ def normalize_params(params) params = params.dup params[:messages] = normalize_messages(params[:messages]) if params[:messages] params[:system] = normalize_system(params[:system]) if params[:system] + params[:tools] = normalize_tools(params[:tools]) if params[:tools] + params[:tool_choice] = normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] params end + # Normalizes tools from common format to Anthropic format. + # + # Accepts both `parameters` and `input_schema` keys, converting to Anthropic's `input_schema`. + # + # @param tools [Array] + # @return [Array] + def normalize_tools(tools) + return tools unless tools.is_a?(Array) + + tools.map do |tool| + tool_hash = tool.is_a?(Hash) ? tool.deep_symbolize_keys : tool + + # If already in Anthropic format (has input_schema), return as-is + next tool_hash if tool_hash[:input_schema] + + # Convert common format with 'parameters' to Anthropic format with 'input_schema' + if tool_hash[:parameters] + tool_hash = tool_hash.dup + tool_hash[:input_schema] = tool_hash.delete(:parameters) + end + + tool_hash + end + end + + # Normalizes tool_choice from common format to Anthropic gem model objects. + # + # The Anthropic gem expects tool_choice to be a model object (ToolChoiceAuto, + # ToolChoiceAny, ToolChoiceTool, etc.), not a plain hash. + # + # Maps: + # - "required" → ToolChoiceAny (force tool use) + # - "auto" → ToolChoiceAuto (let model decide) + # - { name: "..." } → ToolChoiceTool with name + # + # @param tool_choice [String, Hash, Object] + # @return [Object] Anthropic gem model object + def normalize_tool_choice(tool_choice) + # If already a gem model object, return as-is + return tool_choice if tool_choice.is_a?(::Anthropic::Models::ToolChoiceAuto) || + tool_choice.is_a?(::Anthropic::Models::ToolChoiceAny) || + tool_choice.is_a?(::Anthropic::Models::ToolChoiceTool) || + tool_choice.is_a?(::Anthropic::Models::ToolChoiceNone) + + case tool_choice + when "required" + # Create ToolChoiceAny model for forcing tool use + ::Anthropic::Models::ToolChoiceAny.new(type: :any) + when "auto" + # Create ToolChoiceAuto model for letting model decide + ::Anthropic::Models::ToolChoiceAuto.new(type: :auto) + when Hash + choice_hash = tool_choice.deep_symbolize_keys + + # If has type field, create appropriate model + if choice_hash[:type] + case choice_hash[:type].to_sym + when :any + ::Anthropic::Models::ToolChoiceAny.new(**choice_hash) + when :auto + ::Anthropic::Models::ToolChoiceAuto.new(**choice_hash) + when :tool + ::Anthropic::Models::ToolChoiceTool.new(**choice_hash) + when :none + ::Anthropic::Models::ToolChoiceNone.new(**choice_hash) + else + choice_hash + end + # Convert { name: "..." } to ToolChoiceTool + elsif choice_hash[:name] + ::Anthropic::Models::ToolChoiceTool.new(type: :tool, name: choice_hash[:name]) + else + choice_hash + end + else + tool_choice + end + end + # Merges consecutive same-role messages into single messages with multiple content blocks. # # Required by Anthropic API - consecutive messages with the same role must be combined. @@ -320,6 +401,7 @@ def cleanup_serialized_request(hash, defaults, gem_object = nil) # Remove provider-internal fields that should not be in API request hash.delete(:mcp_servers) # Provider-level config, not API param hash.delete(:stop_sequences) if hash[:stop_sequences] == [] + hash.delete(:tool_choice) if hash[:tool_choice].nil? # Don't send null tool_choice # Remove default values (except max_tokens which is required by API) defaults.each do |key, value| diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_input_schema.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_input_schema.yml new file mode 100644 index 00000000..3fc48e1e --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_input_schema.yml @@ -0,0 +1,209 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather in Boston?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]},"name":"get_weather","description":"Get + the current weather in a given location"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '370' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:53 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:53Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:53Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:52Z' + Retry-After: + - '7' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:53Z' + Request-Id: + - req_011CVGgCyKtoGwvJ4LZMLEX1 + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1361' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d38849d1c176a-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01AWL8cLBBTtvG3abMDNTavE","type":"message","role":"assistant","content":[{"type":"tool_use","id":"toolu_014zxU3N9j3gJnV6wKahfpXA","name":"get_weather","input":{"location":"Boston, + MA"}}],"stop_reason":"tool_use","stop_sequence":null,"usage":{"input_tokens":584,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":56,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 05:13:53 GMT +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather in Boston?","role":"user"},{"content":[{"id":"toolu_014zxU3N9j3gJnV6wKahfpXA","input":{"location":"Boston, + MA"},"name":"get_weather","type":"tool_use"}],"role":"assistant"},{"content":[{"tool_use_id":"toolu_014zxU3N9j3gJnV6wKahfpXA","type":"tool_result","content":"{\"location\":\"Boston, + MA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","is_error":false}],"role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]},"name":"get_weather","description":"Get + the current weather in a given location"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '724' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:54 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:54Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:54Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:53Z' + Retry-After: + - '6' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:54Z' + Request-Id: + - req_011CVGgD62yK5v6BTMo2zNfb + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1093' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d388e7d9d6896-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + eyJtb2RlbCI6ImNsYXVkZS1oYWlrdS00LTUtMjAyNTEwMDEiLCJpZCI6Im1zZ18wMUtybmQycUdGRTFUTHVMNkdVVGhUOWsiLCJ0eXBlIjoibWVzc2FnZSIsInJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJUaGUgd2VhdGhlciBpbiBCb3N0b24sIE1BIGlzIGN1cnJlbnRseSAqKnN1bm55Kiogd2l0aCBhIHRlbXBlcmF0dXJlIG9mICoqNzLCsEYqKi4gSXQncyBhIG5pY2UgZGF5ISJ9XSwic3RvcF9yZWFzb24iOiJlbmRfdHVybiIsInN0b3Bfc2VxdWVuY2UiOm51bGwsInVzYWdlIjp7ImlucHV0X3Rva2VucyI6NjY5LCJjYWNoZV9jcmVhdGlvbl9pbnB1dF90b2tlbnMiOjAsImNhY2hlX3JlYWRfaW5wdXRfdG9rZW5zIjowLCJjYWNoZV9jcmVhdGlvbiI6eyJlcGhlbWVyYWxfNW1faW5wdXRfdG9rZW5zIjowLCJlcGhlbWVyYWxfMWhfaW5wdXRfdG9rZW5zIjowfSwib3V0cHV0X3Rva2VucyI6MzAsInNlcnZpY2VfdGllciI6InN0YW5kYXJkIn19 + recorded_at: Wed, 19 Nov 2025 05:13:54 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_multiple_tools.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_multiple_tools.yml new file mode 100644 index 00000000..d28549a1 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_multiple_tools.yml @@ -0,0 +1,206 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather in NYC and what''s 5 plus 3?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + the current weather"},{"input_schema":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]},"name":"calculate","description":"Perform + basic arithmetic"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '571' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:56 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:55Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:56Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:55Z' + Retry-After: + - '5' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:55Z' + Request-Id: + - req_011CVGgDBV93Cuj7Hpw2cchR + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1310' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d38967c3cf957-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01DjAW69igMMdeAKZ5mzzbyY","type":"message","role":"assistant","content":[{"type":"tool_use","id":"toolu_016rFuCULXZGHkktxmEjaPpJ","name":"get_weather","input":{"location":"NYC"}},{"type":"tool_use","id":"toolu_01RnqNzPw77hCENNf1w5XtMg","name":"calculate","input":{"operation":"add","a":5,"b":3}}],"stop_reason":"tool_use","stop_sequence":null,"usage":{"input_tokens":662,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":122,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 05:13:56 GMT +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather in NYC and what''s 5 plus 3?","role":"user"},{"content":[{"id":"toolu_016rFuCULXZGHkktxmEjaPpJ","input":{"location":"NYC"},"name":"get_weather","type":"tool_use"},{"id":"toolu_01RnqNzPw77hCENNf1w5XtMg","input":{"operation":"add","a":5,"b":3},"name":"calculate","type":"tool_use"}],"role":"assistant"},{"content":[{"tool_use_id":"toolu_016rFuCULXZGHkktxmEjaPpJ","type":"tool_result","content":"{\"location\":\"NYC\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","is_error":false},{"tool_use_id":"toolu_01RnqNzPw77hCENNf1w5XtMg","type":"tool_result","content":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","is_error":false}],"role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + the current weather"},{"input_schema":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]},"name":"calculate","description":"Perform + basic arithmetic"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1180' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:57 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:57Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:57Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:56Z' + Retry-After: + - '3' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:57Z' + Request-Id: + - req_011CVGgDHgwtCewKNMwnASAR + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1304' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d389f8cc9f95f-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + eyJtb2RlbCI6ImNsYXVkZS1oYWlrdS00LTUtMjAyNTEwMDEiLCJpZCI6Im1zZ18wMU1KNmRvdHlmTW9iSmZwV01XcEp2MVEiLCJ0eXBlIjoibWVzc2FnZSIsInJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJHcmVhdCEgSGVyZSdzIHRoZSBpbmZvcm1hdGlvbiB5b3UgcmVxdWVzdGVkOlxuXG4qKldlYXRoZXIgaW4gTllDOioqIEl0J3MgY3VycmVudGx5IDcywrBGIGFuZCBzdW5ueSEg4piA77iPXG5cbioqNSBwbHVzIDM6KiogVGhlIGFuc3dlciBpcyAqKjgqKiJ9XSwic3RvcF9yZWFzb24iOiJlbmRfdHVybiIsInN0b3Bfc2VxdWVuY2UiOm51bGwsInVzYWdlIjp7ImlucHV0X3Rva2VucyI6ODc2LCJjYWNoZV9jcmVhdGlvbl9pbnB1dF90b2tlbnMiOjAsImNhY2hlX3JlYWRfaW5wdXRfdG9rZW5zIjowLCJjYWNoZV9jcmVhdGlvbiI6eyJlcGhlbWVyYWxfNW1faW5wdXRfdG9rZW5zIjowLCJlcGhlbWVyYWxfMWhfaW5wdXRfdG9rZW5zIjowfSwib3V0cHV0X3Rva2VucyI6NDYsInNlcnZpY2VfdGllciI6InN0YW5kYXJkIn19 + recorded_at: Wed, 19 Nov 2025 05:13:57 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_parameters.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_parameters.yml new file mode 100644 index 00000000..d6e6a103 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_parameters.yml @@ -0,0 +1,209 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather in San Francisco?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]},"name":"get_weather","description":"Get + the current weather in a given location"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '377' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:45 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:45Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:45Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:44Z' + Retry-After: + - '15' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:45Z' + Request-Id: + - req_011CVGgCPE3eGZd6g9EB3PE3 + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1084' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d3852ce872517-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01EuuY4DyAnyry5fLgdVzk1w","type":"message","role":"assistant","content":[{"type":"tool_use","id":"toolu_01UXUH8LCk3paEsKxjqLayG2","name":"get_weather","input":{"location":"San + Francisco, CA"}}],"stop_reason":"tool_use","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":57,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 05:13:45 GMT +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather in San Francisco?","role":"user"},{"content":[{"id":"toolu_01UXUH8LCk3paEsKxjqLayG2","input":{"location":"San + Francisco, CA"},"name":"get_weather","type":"tool_use"}],"role":"assistant"},{"content":[{"tool_use_id":"toolu_01UXUH8LCk3paEsKxjqLayG2","type":"tool_result","content":"{\"location\":\"San + Francisco, CA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","is_error":false}],"role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]},"name":"get_weather","description":"Get + the current weather in a given location"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '745' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:46 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:46Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:46Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:45Z' + Retry-After: + - '14' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:46Z' + Request-Id: + - req_011CVGgCUUohvAoZzNCtpVEA + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '929' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d385a7a78d025-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + eyJtb2RlbCI6ImNsYXVkZS1oYWlrdS00LTUtMjAyNTEwMDEiLCJpZCI6Im1zZ18wMU5HdmlKZmpDZUxwRHVKN1NKMXZZdFkiLCJ0eXBlIjoibWVzc2FnZSIsInJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJUaGUgd2VhdGhlciBpbiBTYW4gRnJhbmNpc2NvLCBDQSBpcyBjdXJyZW50bHkgKipzdW5ueSoqIHdpdGggYSB0ZW1wZXJhdHVyZSBvZiAqKjcywrBGKiouIEl0J3MgYSBiZWF1dGlmdWwgZGF5ISJ9XSwic3RvcF9yZWFzb24iOiJlbmRfdHVybiIsInN0b3Bfc2VxdWVuY2UiOm51bGwsInVzYWdlIjp7ImlucHV0X3Rva2VucyI6NjcyLCJjYWNoZV9jcmVhdGlvbl9pbnB1dF90b2tlbnMiOjAsImNhY2hlX3JlYWRfaW5wdXRfdG9rZW5zIjowLCJjYWNoZV9jcmVhdGlvbiI6eyJlcGhlbWVyYWxfNW1faW5wdXRfdG9rZW5zIjowLCJlcGhlbWVyYWxfMWhfaW5wdXRfdG9rZW5zIjowfSwib3V0cHV0X3Rva2VucyI6MzEsInNlcnZpY2VfdGllciI6InN0YW5kYXJkIn19 + recorded_at: Wed, 19 Nov 2025 05:13:46 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml new file mode 100644 index 00000000..e45b13f2 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml @@ -0,0 +1,105 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + weather"}],"tool_choice":{"type":"auto"}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '299' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:59 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:58Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:59Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:57Z' + Retry-After: + - '2' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:58Z' + Request-Id: + - req_011CVGgDPzxgDVtcZxRcL3VR + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1146' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d38a8bff5f9f5-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_012dzMywWG636J19CAyU6R1z","type":"message","role":"assistant","content":[{"type":"text","text":"I''d + be happy to help you with the weather! However, I need to know your location. + Could you please tell me which city or location you''d like the weather for?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":558,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":39,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 05:13:59 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_required.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_required.yml new file mode 100644 index 00000000..0fb3aebe --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_required.yml @@ -0,0 +1,207 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + weather"}],"tool_choice":{"type":"any"}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '298' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:47 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:47Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:47Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:46Z' + Retry-After: + - '13' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:47Z' + Request-Id: + - req_011CVGgCZL1VuQAp6qtQtH4E + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1053' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d38619d7e5c1f-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01GrmnfB317zDPsKT3smbXAw","type":"message","role":"assistant","content":[{"type":"tool_use","id":"toolu_018gB1gszzTHn8ajnaihUDVH","name":"get_weather","input":{"location":"current + location"}}],"stop_reason":"tool_use","stop_sequence":null,"usage":{"input_tokens":650,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":39,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 05:13:47 GMT +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather?","role":"user"},{"content":[{"id":"toolu_018gB1gszzTHn8ajnaihUDVH","input":{"location":"current + location"},"name":"get_weather","type":"tool_use"}],"role":"assistant"},{"content":[{"tool_use_id":"toolu_018gB1gszzTHn8ajnaihUDVH","type":"tool_result","content":"{\"location\":\"current + location\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","is_error":false}],"role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + weather"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '635' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:48 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:48Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:48Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:47Z' + Retry-After: + - '12' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:48Z' + Request-Id: + - req_011CVGgCeejuG18Mghs594NC + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1120' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d38695f81fac6-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + eyJtb2RlbCI6ImNsYXVkZS1oYWlrdS00LTUtMjAyNTEwMDEiLCJpZCI6Im1zZ18wMVlRaWNTTktxa3FESjQxYlNGelB4U2IiLCJ0eXBlIjoibWVzc2FnZSIsInJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJUaGUgd2VhdGhlciBhdCB5b3VyIGN1cnJlbnQgbG9jYXRpb24gaXM6XG4tICoqVGVtcGVyYXR1cmUqKjogNzLCsEZcbi0gKipDb25kaXRpb25zKio6IFN1bm55XG5cbkl0J3MgYSBuaWNlIHN1bm55IGRheSEgSXMgdGhlcmUgYW55dGhpbmcgZWxzZSB5b3UnZCBsaWtlIHRvIGtub3cgYWJvdXQgdGhlIHdlYXRoZXI/In1dLCJzdG9wX3JlYXNvbiI6ImVuZF90dXJuIiwic3RvcF9zZXF1ZW5jZSI6bnVsbCwidXNhZ2UiOnsiaW5wdXRfdG9rZW5zIjo2NDEsImNhY2hlX2NyZWF0aW9uX2lucHV0X3Rva2VucyI6MCwiY2FjaGVfcmVhZF9pbnB1dF90b2tlbnMiOjAsImNhY2hlX2NyZWF0aW9uIjp7ImVwaGVtZXJhbF81bV9pbnB1dF90b2tlbnMiOjAsImVwaGVtZXJhbF8xaF9pbnB1dF90b2tlbnMiOjB9LCJvdXRwdXRfdG9rZW5zIjo0OSwic2VydmljZV90aWVyIjoic3RhbmRhcmQifX0= + recorded_at: Wed, 19 Nov 2025 05:13:48 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml new file mode 100644 index 00000000..6d73430f --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml @@ -0,0 +1,207 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + weather"}],"tool_choice":{"type":"tool","name":"get_weather"}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '320' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:49 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:49Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:49Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:49Z' + Retry-After: + - '11' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:49Z' + Request-Id: + - req_011CVGgCk7QdbL9Ro4qS1mDn + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '885' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d38715cbaf555-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01NviDBZtqZfGTS5vAD8Vo6z","type":"message","role":"assistant","content":[{"type":"tool_use","id":"toolu_01RnUd1Xv78Dx6XPssao9Eou","name":"get_weather","input":{"location":"current + location"}}],"stop_reason":"tool_use","stop_sequence":null,"usage":{"input_tokens":655,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":34,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 05:13:49 GMT +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather?","role":"user"},{"content":[{"id":"toolu_01RnUd1Xv78Dx6XPssao9Eou","input":{"location":"current + location"},"name":"get_weather","type":"tool_use"}],"role":"assistant"},{"content":[{"tool_use_id":"toolu_01RnUd1Xv78Dx6XPssao9Eou","type":"tool_result","content":"{\"location\":\"current + location\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","is_error":false}],"role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]},"name":"get_weather","description":"Get + weather"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '635' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 05:13:52 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T05:13:51Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T05:13:51Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T05:13:50Z' + Retry-After: + - '9' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T05:13:51Z' + Request-Id: + - req_011CVGgCpc2RFuCa64fftcsD + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '1887' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a0d3877ce7f2832-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + eyJtb2RlbCI6ImNsYXVkZS1oYWlrdS00LTUtMjAyNTEwMDEiLCJpZCI6Im1zZ18wMVdMRHN6cUtoaG1IaFFEREEzbXJIU1QiLCJ0eXBlIjoibWVzc2FnZSIsInJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJUaGUgd2VhdGhlciBhdCB5b3VyIGN1cnJlbnQgbG9jYXRpb24gaXMgKipzdW5ueSoqIHdpdGggYSB0ZW1wZXJhdHVyZSBvZiAqKjcywrBGKiouIExvb2tzIGxpa2UgYSBuaWNlIGRheSEgXG5cbklmIHlvdSdkIGxpa2Ugd2VhdGhlciBpbmZvcm1hdGlvbiBmb3IgYSBzcGVjaWZpYyBsb2NhdGlvbiwganVzdCBsZXQgbWUga25vdyB0aGUgY2l0eSBvciBhcmVhIG5hbWUgYW5kIEkgY2FuIGdldCB0aGF0IGZvciB5b3UuIn1dLCJzdG9wX3JlYXNvbiI6ImVuZF90dXJuIiwic3RvcF9zZXF1ZW5jZSI6bnVsbCwidXNhZ2UiOnsiaW5wdXRfdG9rZW5zIjo2NDEsImNhY2hlX2NyZWF0aW9uX2lucHV0X3Rva2VucyI6MCwiY2FjaGVfcmVhZF9pbnB1dF90b2tlbnMiOjAsImNhY2hlX2NyZWF0aW9uIjp7ImVwaGVtZXJhbF81bV9pbnB1dF90b2tlbnMiOjAsImVwaGVtZXJhbF8xaF9pbnB1dF90b2tlbnMiOjB9LCJvdXRwdXRfdG9rZW5zIjo2MCwic2VydmljZV90aWVyIjoic3RhbmRhcmQifX0= + recorded_at: Wed, 19 Nov 2025 05:13:52 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/anthropic/common_format/tools_test.rb b/test/integration/anthropic/common_format/tools_test.rb new file mode 100644 index 00000000..f6de1da0 --- /dev/null +++ b/test/integration/anthropic/common_format/tools_test.rb @@ -0,0 +1,347 @@ +# frozen_string_literal: true + +require_relative "../../test_helper" + +module Integration + module Anthropic + module CommonFormat + class ToolsTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :anthropic, model: "claude-haiku-4-5", max_tokens: 1024 + + def get_weather(location:) + { location: location, temperature: "72°F", conditions: "sunny" } + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + + # Common format with 'parameters' key (recommended) + COMMON_FORMAT_PARAMETERS = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What's the weather in San Francisco?" + } + ], + max_tokens: 1024, + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + } + def common_format_parameters + prompt( + message: "What's the weather in San Francisco?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Common format with 'input_schema' key (Anthropic native) + COMMON_FORMAT_INPUT_SCHEMA = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What's the weather in Boston?" + } + ], + max_tokens: 1024, + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + } + def common_format_input_schema + prompt( + message: "What's the weather in Boston?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Multiple tools in common format + COMMON_FORMAT_MULTIPLE_TOOLS = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What's the weather in NYC and what's 5 plus 3?" + } + ], + max_tokens: 1024, + tools: [ + { + name: "get_weather", + description: "Get the current weather", + input_schema: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + name: "calculate", + description: "Perform basic arithmetic", + input_schema: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ] + } + def common_format_multiple_tools + prompt( + message: "What's the weather in NYC and what's 5 plus 3?", + tools: [ + { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ] + ) + end + + # Tool choice - string format + COMMON_FORMAT_TOOL_CHOICE_AUTO = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + max_tokens: 1024, + tools: [ + { + name: "get_weather", + description: "Get weather", + input_schema: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { type: "auto" } + } + def common_format_tool_choice_auto + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Tool choice - force tool use with "required" + COMMON_FORMAT_TOOL_CHOICE_REQUIRED = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + max_tokens: 1024, + tools: [ + { + name: "get_weather", + description: "Get weather", + input_schema: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { type: "any" } + } + def common_format_tool_choice_required + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "required" + ) + end + + # Tool choice - specific tool + COMMON_FORMAT_TOOL_CHOICE_SPECIFIC = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + max_tokens: 1024, + tools: [ + { + name: "get_weather", + description: "Get weather", + input_schema: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { type: "tool", name: "get_weather" } + } + def common_format_tool_choice_specific + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { name: "get_weather" } + ) + end + end + + ################################################################################ + # This automatically runs all the tests for the test actions + ################################################################################ + [ + :common_format_parameters, + :common_format_input_schema, + :common_format_multiple_tools, + :common_format_tool_choice_auto, + :common_format_tool_choice_required, + :common_format_tool_choice_specific + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) + end + end + end + end +end From b2d377d69b0995521c43d3fc8e77b3b5107c94b3 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Tue, 18 Nov 2025 22:12:32 -0800 Subject: [PATCH 02/17] Add OpenAI Response Tools Common Format --- .../providers/open_ai/responses/request.rb | 6 +- .../providers/open_ai/responses/transforms.rb | 99 ++++ .../providers/open_ai/responses_provider.rb | 34 ++ .../test_agent_common_format_input_schema.yml | 379 +++++++++++++++ ...est_agent_common_format_multiple_tools.yml | 449 ++++++++++++++++++ .../test_agent_common_format_parameters.yml | 379 +++++++++++++++ ...t_agent_common_format_tool_choice_auto.yml | 191 ++++++++ ...ent_common_format_tool_choice_required.yml | 375 +++++++++++++++ ...ent_common_format_tool_choice_specific.yml | 378 +++++++++++++++ .../responses/common_format/tools_test.rb | 329 +++++++++++++ 10 files changed, 2618 insertions(+), 1 deletion(-) create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_input_schema.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_multiple_tools.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_parameters.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_required.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml create mode 100644 test/integration/open_ai/responses/common_format/tools_test.rb diff --git a/lib/active_agent/providers/open_ai/responses/request.rb b/lib/active_agent/providers/open_ai/responses/request.rb index 93d169c4..f42785b0 100644 --- a/lib/active_agent/providers/open_ai/responses/request.rb +++ b/lib/active_agent/providers/open_ai/responses/request.rb @@ -73,7 +73,11 @@ def initialize(**params) # Step 6: Normalize input content for gem compatibility params[:input] = Responses::Transforms.normalize_input(params[:input]) if params[:input] - # Step 7: Create gem model - delegates to OpenAI gem + # Step 7: Normalize tools and tool_choice from common format + params[:tools] = Responses::Transforms.normalize_tools(params[:tools]) if params[:tools] + params[:tool_choice] = Responses::Transforms.normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] + + # Step 8: Create gem model - delegates to OpenAI gem gem_model = ::OpenAI::Models::Responses::ResponseCreateParams.new(**params) # Step 8: Delegate all method calls to gem model diff --git a/lib/active_agent/providers/open_ai/responses/transforms.rb b/lib/active_agent/providers/open_ai/responses/transforms.rb index bfaebbfc..71a5874b 100644 --- a/lib/active_agent/providers/open_ai/responses/transforms.rb +++ b/lib/active_agent/providers/open_ai/responses/transforms.rb @@ -21,6 +21,105 @@ def gem_to_hash(gem_object) JSON.parse(gem_object.to_json, symbolize_names: true) end + # Normalizes tools from common format to OpenAI Responses API format. + # + # Accepts tools in multiple formats: + # - Common format: `{name: "...", description: "...", parameters: {...}}` + # - Nested format: `{type: "function", function: {name: "...", ...}}` + # - Responses format: `{type: "function", name: "...", parameters: {...}}` + # + # Always outputs flat Responses API format. + # + # @param tools [Array] + # @return [Array] + def normalize_tools(tools) + return tools unless tools.is_a?(Array) + + tools.map do |tool| + tool_hash = tool.is_a?(Hash) ? tool.deep_symbolize_keys : tool + + # If already in Responses format (flat with type, name, parameters), return as-is + if tool_hash[:type] == "function" && tool_hash[:name] + next tool_hash + end + + # If in nested Chat API format, flatten it + if tool_hash[:type] == "function" && tool_hash[:function] + func = tool_hash[:function] + next { + type: "function", + name: func[:name], + description: func[:description], + parameters: func[:parameters] || func[:input_schema] + }.compact + end + + # If in common format (no type field), convert to Responses format + if tool_hash[:name] && !tool_hash[:type] + next { + type: "function", + name: tool_hash[:name], + description: tool_hash[:description], + parameters: tool_hash[:parameters] || tool_hash[:input_schema] + }.compact + end + + # Pass through other formats + tool_hash + end + end + + # Normalizes tool_choice from common format to OpenAI Responses API format. + # + # Responses API uses flat format for specific tool choice, unlike Chat API's nested format. + # Must return gem model objects for proper serialization. + # + # Maps: + # - "required" → :required symbol (force tool use) + # - "auto" → :auto symbol (let model decide) + # - { name: "..." } → ToolChoiceFunction model object + # + # @param tool_choice [String, Hash, Object] + # @return [Symbol, Object] Symbol or gem model object + def normalize_tool_choice(tool_choice) + # If already a gem model object, return as-is + return tool_choice if tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceFunction) || + tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceAllowed) || + tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceTypes) || + tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceMcp) || + tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceCustom) + + case tool_choice + when "required" + :required # Return as symbol + when "auto" + :auto # Return as symbol + when "none" + :none # Return as symbol + when Hash + choice_hash = tool_choice.deep_symbolize_keys + + # If already in proper format with type, try to create gem model + if choice_hash[:type] == "function" && choice_hash[:name] + # Create ToolChoiceFunction gem model object + ::OpenAI::Models::Responses::ToolChoiceFunction.new( + type: :function, + name: choice_hash[:name] + ) + # Convert { name: "..." } to ToolChoiceFunction model + elsif choice_hash[:name] && !choice_hash[:type] + ::OpenAI::Models::Responses::ToolChoiceFunction.new( + type: :function, + name: choice_hash[:name] + ) + else + choice_hash + end + else + tool_choice + end + end + # Simplifies input for cleaner API requests # # Unwraps single-element arrays: diff --git a/lib/active_agent/providers/open_ai/responses_provider.rb b/lib/active_agent/providers/open_ai/responses_provider.rb index b1b498e6..ad164a22 100644 --- a/lib/active_agent/providers/open_ai/responses_provider.rb +++ b/lib/active_agent/providers/open_ai/responses_provider.rb @@ -25,6 +25,40 @@ def self.prompt_request_type protected + # @see BaseProvider#prepare_prompt_request + # @return [Request] + def prepare_prompt_request + prepare_prompt_request_tools + + super + end + + # @api private + def prepare_prompt_request_tools + return unless request.tool_choice + + # Get list of function calls that have been made + # In Responses API, message_stack items are flat - each item has a type field + functions_used = message_stack + .select { |item| item[:type] == "function_call" } + .map { |item| item[:name] } + .compact + + # Check if tool_choice is a gem model object or symbol + if request.tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceFunction) + # Specific tool choice - clear if that tool was used + tool_choice_name = request.tool_choice.name + if tool_choice_name && functions_used.include?(tool_choice_name) + request.tool_choice = nil + end + elsif request.tool_choice == :required + # Required tool choice - clear if any tool was used + if functions_used.any? + request.tool_choice = nil + end + end + end + # @return [Object] OpenAI client's responses endpoint def api_prompt_executer client.responses diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_input_schema.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_input_schema.yml new file mode 100644 index 00000000..fc4398ae --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_input_schema.yml @@ -0,0 +1,379 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What''s the weather in Boston?","tools":[{"type":"function","name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '349' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:07 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999723' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_a7ec86589c0e452aae830f8e1aab0b9e + Openai-Processing-Ms: + - '698' + X-Envoy-Upstream-Service-Time: + - '700' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=onoCz1cVaAO3fnxszHT1oDXSaXnTwqJ4vIaaMiK0TNc-1763532667-1.0.1.1-Ab5x7CUkOe4srMhphaKJChvsDvVV5KMkMy6dE143HRbeOTZq_s4V84fSUeeqk_1uiXjbzs4aOTJZ51rqEu_QFm4M2ktvpu0bEexQpCA8JVA; + path=/; expires=Wed, 19-Nov-25 06:41:07 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=2FKxYBOeWG2LIOAqiGHlrDuK2m2YEfALiNHEm_tHMzU-1763532667941-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c618b89cee9-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_018e356c588e624800691d5f7b36d481968c8ecb0314784e2b", + "object": "response", + "created_at": 1763532667, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "fc_018e356c588e624800691d5f7baa188196bcc8c4954a75ecc2", + "type": "function_call", + "status": "completed", + "arguments": "{\"location\":\"Boston, MA\"}", + "call_id": "call_FxpdpMpXNm8kcvvVkUwU779y", + "name": "get_weather" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get the current weather in a given location", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string", + "description": "The city and state, e.g. San Francisco, CA" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 60, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 17, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 77 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:07 GMT +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":[{"role":"user","content":"What''s the weather + in Boston?"},{"arguments":"{\"location\":\"Boston, MA\"}","call_id":"call_FxpdpMpXNm8kcvvVkUwU779y","name":"get_weather","type":"function_call","id":"fc_018e356c588e624800691d5f7baa188196bcc8c4954a75ecc2","status":"completed"},{"call_id":"call_FxpdpMpXNm8kcvvVkUwU779y","output":"{\"location\":\"Boston, + MA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","type":"function_call_output"}],"tools":[{"type":"function","name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '757' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:08 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999681' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_6383c911b15a47e99ce15f9d87da3e54 + Openai-Processing-Ms: + - '869' + X-Envoy-Upstream-Service-Time: + - '872' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=Dmkk7Dmj3.ATH2iKiVR_5oZDAp9sswe0WCH.LgUW4mw-1763532668-1.0.1.1-fnyEjDID8fVSA3o3Ehh1azinNzYA3QvNaCkZrATSPlesFoYcR6RjOVHWnGBaCqfak1i2_pJumumz8K1HuhGSwlLL3zY4P4kOVXeKO9rI0vE; + path=/; expires=Wed, 19-Nov-25 06:41:08 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=Si4zaXfWhtfuqV6w3ccccjMgU0wS.rabg3lg5ny5eC0-1763532668887-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c66fb7ba666-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_018e356c588e624800691d5f7c064481968781661d40264bad", + "object": "response", + "created_at": 1763532668, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "msg_018e356c588e624800691d5f7c7b688196843f1b9f209240f9", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The weather in Boston is currently 72\u00b0F and sunny." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get the current weather in a given location", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string", + "description": "The city and state, e.g. San Francisco, CA" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 102, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 14, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 116 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:08 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_multiple_tools.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_multiple_tools.yml new file mode 100644 index 00000000..e87dbed4 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_multiple_tools.yml @@ -0,0 +1,449 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What''s the weather in NYC and what''s + 5 plus 3?","tools":[{"type":"function","name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}},{"type":"function","name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '566' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:10 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999696' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_6918799381d74405b9a62accc69378c0 + Openai-Processing-Ms: + - '1228' + X-Envoy-Upstream-Service-Time: + - '1232' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=Mnm6UBy1Ew.ZHVCsaJiQLrAQE3OEq5uL42PupnzidB4-1763532670-1.0.1.1-yIcJJBC9OH746vItV.a1guk5oC31jlVRwVuogkaoMUauvg_KWaOk9AxEcxfXxfedwxbrIIFZdNS3cLUSTMu7hIFP6wQRMfpaGr6.gggw15g; + path=/; expires=Wed, 19-Nov-25 06:41:10 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=hG.QaWB4Jd06L3DQqmxyF4YG11P88FguXAu94vxWxIw-1763532670210-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c6cf97eeb22-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_04e95d07f1de59b200691d5f7cff84819a869de18fad9a448a", + "object": "response", + "created_at": 1763532669, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "fc_04e95d07f1de59b200691d5f7dd024819a8d138035f4846a34", + "type": "function_call", + "status": "completed", + "arguments": "{\"location\":\"NYC\"}", + "call_id": "call_Zi12SRBwXTIyjZaBdmxfZvLt", + "name": "get_weather" + }, + { + "id": "fc_04e95d07f1de59b200691d5f7dfba4819a996acd9bbf98e8b5", + "type": "function_call", + "status": "completed", + "arguments": "{\"operation\":\"add\",\"a\":5,\"b\":3}", + "call_id": "call_0vSIbaSaYkamroz4tIupYYe0", + "name": "calculate" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get the current weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + }, + { + "type": "function", + "description": "Perform basic arithmetic", + "name": "calculate", + "parameters": { + "type": "object", + "properties": { + "operation": { + "type": "string", + "enum": [ + "add", + "subtract", + "multiply", + "divide" + ] + }, + "a": { + "type": "number" + }, + "b": { + "type": "number" + } + }, + "required": [ + "operation", + "a", + "b" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 43, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 52, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 95 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:10 GMT +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":[{"role":"user","content":"What''s the weather + in NYC and what''s 5 plus 3?"},{"arguments":"{\"location\":\"NYC\"}","call_id":"call_Zi12SRBwXTIyjZaBdmxfZvLt","name":"get_weather","type":"function_call","id":"fc_04e95d07f1de59b200691d5f7dd024819a8d138035f4846a34","status":"completed"},{"arguments":"{\"operation\":\"add\",\"a\":5,\"b\":3}","call_id":"call_0vSIbaSaYkamroz4tIupYYe0","name":"calculate","type":"function_call","id":"fc_04e95d07f1de59b200691d5f7dfba4819a996acd9bbf98e8b5","status":"completed"},{"call_id":"call_Zi12SRBwXTIyjZaBdmxfZvLt","output":"{\"location\":\"NYC\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","type":"function_call_output"},{"call_id":"call_0vSIbaSaYkamroz4tIupYYe0","output":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","type":"function_call_output"}],"tools":[{"type":"function","name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}},{"type":"function","name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1320' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:11 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999616' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_4bbd5f42683a46dbbb9fa068156c45e6 + Openai-Processing-Ms: + - '955' + X-Envoy-Upstream-Service-Time: + - '957' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=NGa0hm5RXfB_5LMKYtTFpI9_GfXn60SBMH6Kki9ol3o-1763532671-1.0.1.1-NkjOxHaqZceIE.bG_7kVJ2xvCS7x7c3voCWY1gwUWLgdkwFjwAsJLQ1lmaeLoN7lAR22JXCQR6SuekF.KWVh4vITD5_jXWrvrqFHMZj7H.8; + path=/; expires=Wed, 19-Nov-25 06:41:11 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=djqhwD6mZ7kgb9Ey7SMGATG74aowOLANeEIKcX.Luu0-1763532671334-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c751e066896-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_04e95d07f1de59b200691d5f7e5bc4819abd93a0c80931ff2b", + "object": "response", + "created_at": 1763532670, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "msg_04e95d07f1de59b200691d5f7eb608819aa23cefc267f6fb32", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The weather in NYC is currently 72\u00b0F and sunny. Also, 5 plus 3 equals 8." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get the current weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + }, + { + "type": "function", + "description": "Perform basic arithmetic", + "name": "calculate", + "parameters": { + "type": "object", + "properties": { + "operation": { + "type": "string", + "enum": [ + "add", + "subtract", + "multiply", + "divide" + ] + }, + "a": { + "type": "number" + }, + "b": { + "type": "number" + } + }, + "required": [ + "operation", + "a", + "b" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 167, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 25, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 192 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:11 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_parameters.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_parameters.yml new file mode 100644 index 00000000..97cddd30 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_parameters.yml @@ -0,0 +1,379 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What''s the weather in San Francisco?","tools":[{"type":"function","name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '356' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:13 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999722' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_e98adf069fbb426ea7a05f4d066c8323 + Openai-Processing-Ms: + - '673' + X-Envoy-Upstream-Service-Time: + - '675' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=G4mTIIHTZVf3x_nD0fNFbP1TYYAKLc0tyi5MqyWKXus-1763532673-1.0.1.1-0lNJ4.ywfxbHgQviNhfK76eX3bRObUADyXH84E6SB1mwDRai7MuAyuJWdRCBxsQ6GsdoUGZXtbemqQrYV5iPN7nf.hPsFGy6kw8dBoSQ5fY; + path=/; expires=Wed, 19-Nov-25 06:41:13 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=OxrjL31xuRt.LmXIwAh4msKrf779HRAFIPUalLcXVXg-1763532673412-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c84591015b4-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_05b9e4590cdecec900691d5f80be748199838550444bf1fad6", + "object": "response", + "created_at": 1763532672, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "fc_05b9e4590cdecec900691d5f81343481999661e86ddef850c1", + "type": "function_call", + "status": "completed", + "arguments": "{\"location\":\"San Francisco, CA\"}", + "call_id": "call_lRBYXQbTLjY8GvUsBHWt9yqR", + "name": "get_weather" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get the current weather in a given location", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string", + "description": "The city and state, e.g. San Francisco, CA" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 61, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 18, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 79 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:13 GMT +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":[{"role":"user","content":"What''s the weather + in San Francisco?"},{"arguments":"{\"location\":\"San Francisco, CA\"}","call_id":"call_lRBYXQbTLjY8GvUsBHWt9yqR","name":"get_weather","type":"function_call","id":"fc_05b9e4590cdecec900691d5f81343481999661e86ddef850c1","status":"completed"},{"call_id":"call_lRBYXQbTLjY8GvUsBHWt9yqR","output":"{\"location\":\"San + Francisco, CA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","type":"function_call_output"}],"tools":[{"type":"function","name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '778' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:14 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999678' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_1574a48a1716446cbbc82375a4691667 + Openai-Processing-Ms: + - '705' + X-Envoy-Upstream-Service-Time: + - '708' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=7mNinPBkS7_N3DT0HA.ZNHLXSwniIU15U0pnIVNn_Gs-1763532674-1.0.1.1-v4HotQ8JHGZoxBJa66yMEVDWE7eZsbtQkk7ntalWLwkAvWmAFqYW3rnK6FlaD.nUFHZbXZ_qv1nqwfumx3OBmnOWzWF28qUe_CicxJufKPQ; + path=/; expires=Wed, 19-Nov-25 06:41:14 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=OLWAssiS01iK.fjQDmE7ByKsj_n6aW0I2iPEVxA4ajQ-1763532674244-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c8929c5ed3b-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_05b9e4590cdecec900691d5f81874c819993de1c6ea20575b6", + "object": "response", + "created_at": 1763532673, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "msg_05b9e4590cdecec900691d5f81e1c88199b5089bf73ae19a62", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The weather in San Francisco is currently sunny with a temperature of 72\u00b0F." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get the current weather in a given location", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string", + "description": "The city and state, e.g. San Francisco, CA" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 105, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 18, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 123 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:14 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml new file mode 100644 index 00000000..d3bccf47 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml @@ -0,0 +1,191 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What''s the weather?","tools":[{"type":"function","name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '248' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:12 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999745' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_abac2de70ba941569fd44a3be39d40b7 + Openai-Processing-Ms: + - '1149' + X-Envoy-Upstream-Service-Time: + - '1155' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=vve89jfXkilgqsMV3.99d5pTaZhhxsID_aWVFRY2xQ8-1763532672-1.0.1.1-l5f6mxlE3CwatB7GQBleHgldqOAiNNm1OAq0s.KRkrT4iK0NeB_C5Pzn.Jk7MCFzapLyZ0gYmFcFhQHLQsmGtgDj41qfCDdr_8mCueJD53o; + path=/; expires=Wed, 19-Nov-25 06:41:12 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=t54Np9cWGfao5kv.0EHIeg5CcBZRYitdVG.Mi1o8Tb0-1763532672615-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c7c69d0cf1f-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_01938aae9821358a00691d5f7f7ba4819bbc7200a6be080920", + "object": "response", + "created_at": 1763532671, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "msg_01938aae9821358a00691d5f800e1c819ba7d7722efebb9056", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Could you please specify the location for which you want to know the weather?" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 38, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 17, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 55 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:12 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_required.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_required.yml new file mode 100644 index 00000000..66ba38de --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_required.yml @@ -0,0 +1,375 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What''s the weather?","tools":[{"type":"function","name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}],"tool_choice":"required"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '252' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:05 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999736' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_b39af137302747a2809a4261a8abea20 + Openai-Processing-Ms: + - '818' + X-Envoy-Upstream-Service-Time: + - '824' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=1iNRNtfbceMffE6..cNAVlJobes1ggv.O1iuv1HvrzE-1763532665-1.0.1.1-t2_aLgoSgZH7s8a8fze8vYej5xUhB2IEi887ZW3qoGUj0tYyNblOB7sLg5mPO_4fs44ODXAOx2VKNMRSgUYHhFCDKzr0Oc7YWK8V0Hw97jk; + path=/; expires=Wed, 19-Nov-25 06:41:05 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=FwFeiyea_FDkSdlfNlNE4mr1WNurgkaOM3MP.SBv.YM-1763532665969-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c54aed31f2f-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_05939eef746481bd00691d5f7921848198990d68390590d2c6", + "object": "response", + "created_at": 1763532665, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "fc_05939eef746481bd00691d5f799ee88198bd7e244498e2a1b8", + "type": "function_call", + "status": "completed", + "arguments": "{\"location\":\"current location\"}", + "call_id": "call_kios4Y7iAP7MSxok1Y0WfaBi", + "name": "get_weather" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "required", + "tools": [ + { + "type": "function", + "description": "Get weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 47, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 7, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 54 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:05 GMT +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":[{"role":"user","content":"What''s the weather?"},{"arguments":"{\"location\":\"current + location\"}","call_id":"call_kios4Y7iAP7MSxok1Y0WfaBi","name":"get_weather","type":"function_call","id":"fc_05939eef746481bd00691d5f799ee88198bd7e244498e2a1b8","status":"completed"},{"call_id":"call_kios4Y7iAP7MSxok1Y0WfaBi","output":"{\"location\":\"current + location\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","type":"function_call_output"}],"tools":[{"type":"function","name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}],"tool_choice":null}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '666' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:07 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999708' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_51068511f5574ea3b3043e8a3948b814 + Openai-Processing-Ms: + - '943' + X-Envoy-Upstream-Service-Time: + - '946' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=KXDTmj15Qrw0tabWDXcG42522POTu0p46gOGV9yd32c-1763532667-1.0.1.1-WuY909ShUoXvjLXo2XBeotf6CjpnfflNqsZaorFI1dO3H.YSV9aLja4JH1ZXlCEFXE5Atw86oBj.L46McgkC_sYrphBqxcqqqAvjmlr_ksM; + path=/; expires=Wed, 19-Nov-25 06:41:07 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=38f7IUJwhJPbUHcyMlmaZGjuKrKgoXx1xhcd4uuvHlw-1763532667019-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c5aba03cfd9-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_05939eef746481bd00691d5f7a161c819885d59a0f70306e70", + "object": "response", + "created_at": 1763532666, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "msg_05939eef746481bd00691d5f7a68f881989a296625c7190d01", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The current weather is sunny with a temperature of 72\u00b0F. If you\u2019d like the weather for a specific location, just let me know!" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 75, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 31, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 106 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:07 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml new file mode 100644 index 00000000..2b248247 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml @@ -0,0 +1,378 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What''s the weather?","tools":[{"type":"function","name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}],"tool_choice":{"type":"function","name":"get_weather"}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '282' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:15 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999736' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_50f0d2a57ee14fb29d47e4dd4b2d860b + Openai-Processing-Ms: + - '888' + X-Envoy-Upstream-Service-Time: + - '891' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=iUVByUMfMM5exlVVTJuX3PQw4CFFTWebEZ4qeq_Y2kA-1763532675-1.0.1.1-J8Ea5e8aiRoSGnVvMqTJP8OFR8BLZOMNOb0kqdri_PRKcLil7PtvL_cBzDyqYkVra_keHlOrSQTPpwJBAg6HDwsW65AMrYc50R.b4h1sHHI; + path=/; expires=Wed, 19-Nov-25 06:41:15 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=5GimaK8QHedy.31v3NOh55X3nFosmRNrcmUmAFpN07s-1763532675303-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c8e7d08ed3f-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_051add7d88bbd5e800691d5f8267a08197ae1be3d521f0f38f", + "object": "response", + "created_at": 1763532674, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "fc_051add7d88bbd5e800691d5f8305b081979c0c60afb4568e8c", + "type": "function_call", + "status": "completed", + "arguments": "{\"location\":\"your area\"}", + "call_id": "call_qEEg5lgJOgv5HvsQXA3MS7pS", + "name": "get_weather" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": { + "type": "function", + "name": "get_weather" + }, + "tools": [ + { + "type": "function", + "description": "Get weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 47, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 7, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 54 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:15 GMT +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":[{"role":"user","content":"What''s the weather?"},{"arguments":"{\"location\":\"your + area\"}","call_id":"call_qEEg5lgJOgv5HvsQXA3MS7pS","name":"get_weather","type":"function_call","id":"fc_051add7d88bbd5e800691d5f8305b081979c0c60afb4568e8c","status":"completed"},{"call_id":"call_qEEg5lgJOgv5HvsQXA3MS7pS","output":"{\"location\":\"your + area\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","type":"function_call_output"}],"tools":[{"type":"function","name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}],"tool_choice":null}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '652' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:11:16 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999708' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_3f4ebd0f3b534662a6c7cad9ce4f8f60 + Openai-Processing-Ms: + - '1151' + X-Envoy-Upstream-Service-Time: + - '1157' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=7eBDu6y6m59GwWvE60dlNrgX2wqpPtfmU.goQ2KL2uQ-1763532676-1.0.1.1-v6N0ml6yaWTQ2NEzdn4ExXHluLgpVq0myBJCP9Rg0rkBvFIyY2kzibn4AALrz_S.37X6L4.0ckntL2Ns9CYqGxvTMF.RKS9nogXTJSLYSpY; + path=/; expires=Wed, 19-Nov-25 06:41:16 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=tYWGME1kTuEicMzJN8qkZ2y_PRtObijZW5NHwNVB6TQ-1763532676582-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d8c94ec45fa86-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_051add7d88bbd5e800691d5f836b6c8197b066b5185bd18d12", + "object": "response", + "created_at": 1763532675, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "msg_051add7d88bbd5e800691d5f8412e88197a00f1726a0b07ae8", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The current weather in your area is sunny with a temperature of 72\u00b0F. If you need details for a specific location, let me know!" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get weather", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string" + } + }, + "required": [ + "location" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 75, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 31, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 106 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 06:11:16 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/open_ai/responses/common_format/tools_test.rb b/test/integration/open_ai/responses/common_format/tools_test.rb new file mode 100644 index 00000000..a9509f48 --- /dev/null +++ b/test/integration/open_ai/responses/common_format/tools_test.rb @@ -0,0 +1,329 @@ +# frozen_string_literal: true + +require_relative "../../../test_helper" + +module Integration + module OpenAI + module Responses + module CommonFormat + class ToolsTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4.1" + + def get_weather(location:) + { location: location, temperature: "72°F", conditions: "sunny" } + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + + # Common format with 'parameters' key (recommended) + COMMON_FORMAT_PARAMETERS = { + model: "gpt-4.1", + input: "What's the weather in San Francisco?", + tools: [ + { + type: "function", + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + } + def common_format_parameters + prompt( + input: "What's the weather in San Francisco?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Common format with 'input_schema' key (Anthropic-style) + COMMON_FORMAT_INPUT_SCHEMA = { + model: "gpt-4.1", + input: "What's the weather in Boston?", + tools: [ + { + type: "function", + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + } + def common_format_input_schema + prompt( + input: "What's the weather in Boston?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Multiple tools in common format + COMMON_FORMAT_MULTIPLE_TOOLS = { + model: "gpt-4.1", + input: "What's the weather in NYC and what's 5 plus 3?", + tools: [ + { + type: "function", + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + type: "function", + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ], + tool_choice: "auto" + } + def common_format_multiple_tools + prompt( + input: "What's the weather in NYC and what's 5 plus 3?", + tools: [ + { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Tool choice - string format + COMMON_FORMAT_TOOL_CHOICE_AUTO = { + model: "gpt-4.1", + input: "What's the weather?", + tools: [ + { + type: "function", + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + } + def common_format_tool_choice_auto + prompt( + input: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Tool choice - force tool use with "required" + COMMON_FORMAT_TOOL_CHOICE_REQUIRED = { + model: "gpt-4.1", + input: "What's the weather?", + tools: [ + { + type: "function", + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "required" + } + def common_format_tool_choice_required + prompt( + input: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "required" + ) + end + + # Tool choice - specific tool + COMMON_FORMAT_TOOL_CHOICE_SPECIFIC = { + model: "gpt-4.1", + input: "What's the weather?", + tools: [ + { + type: "function", + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { + type: "function", + name: "get_weather" + } + } + def common_format_tool_choice_specific + prompt( + input: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { name: "get_weather" } + ) + end + end + + ################################################################################ + # This automatically runs all the tests for the test actions + ################################################################################ + [ + :common_format_parameters, + :common_format_input_schema, + :common_format_multiple_tools, + :common_format_tool_choice_auto, + :common_format_tool_choice_required, + :common_format_tool_choice_specific + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) + end + end + end + end + end +end From 9a5a1c740dfa0f0a2f7f0cf829ad3457d1097d12 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Tue, 18 Nov 2025 22:20:23 -0800 Subject: [PATCH 03/17] Add OpenAI Chat Tools Common Format --- .../providers/open_ai/chat/transforms.rb | 86 +++- .../providers/open_ai/chat_provider.rb | 34 ++ .../test_agent_common_format_input_schema.yml | 266 ++++++++++++ ...est_agent_common_format_multiple_tools.yml | 275 +++++++++++++ .../test_agent_common_format_parameters.yml | 266 ++++++++++++ ...t_agent_common_format_tool_choice_auto.yml | 145 +++++++ ...ent_common_format_tool_choice_required.yml | 264 ++++++++++++ ...ent_common_format_tool_choice_specific.yml | 264 ++++++++++++ .../open_ai/chat/common_format/tools_test.rb | 384 ++++++++++++++++++ 9 files changed, 1981 insertions(+), 3 deletions(-) create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_input_schema.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_parameters.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml create mode 100644 test/integration/open_ai/chat/common_format/tools_test.rb diff --git a/lib/active_agent/providers/open_ai/chat/transforms.rb b/lib/active_agent/providers/open_ai/chat/transforms.rb index 42e1d71d..bb24b699 100644 --- a/lib/active_agent/providers/open_ai/chat/transforms.rb +++ b/lib/active_agent/providers/open_ai/chat/transforms.rb @@ -24,8 +24,8 @@ def gem_to_hash(gem_object) # Normalizes all request parameters for OpenAI Chat API # # Handles instructions mapping to developer messages, message normalization, - # and response_format conversion. This is the main entry point for parameter - # transformation. + # tools normalization, and response_format conversion. This is the main entry point + # for parameter transformation. # # @param params [Hash] # @return [Hash] normalized parameters @@ -41,6 +41,12 @@ def normalize_params(params) # Normalize messages for gem compatibility params[:messages] = normalize_messages(params[:messages]) if params[:messages] + # Normalize tools from common format to Chat API format + params[:tools] = normalize_tools(params[:tools]) if params[:tools] + + # Normalize tool_choice from common format + params[:tool_choice] = normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] + # Normalize response_format if present params[:response_format] = normalize_response_format(params[:response_format]) if params[:response_format] @@ -68,7 +74,8 @@ def normalize_messages(messages) messages.each do |msg| normalized = normalize_message(msg) - if grouped.empty? || grouped.last.role != normalized.role + # Don't merge tool messages - each needs its own tool_call_id + if grouped.empty? || grouped.last.role != normalized.role || normalized.role.to_s == "tool" grouped << normalized else # Merge consecutive same-role messages @@ -307,6 +314,79 @@ def normalize_response_format(format) end end + # Normalizes tools from common format to OpenAI Chat API format. + # + # Accepts tools in multiple formats: + # - Common format: `{name: "...", description: "...", parameters: {...}}` + # - Common format alt: `{name: "...", description: "...", input_schema: {...}}` + # - Nested format: `{type: "function", function: {name: "...", parameters: {...}}}` + # + # Always outputs nested Chat API format: `{type: "function", function: {...}}` + # + # @param tools [Array] + # @return [Array] + def normalize_tools(tools) + return tools unless tools.is_a?(Array) + + tools.map do |tool| + tool_hash = tool.is_a?(Hash) ? tool.deep_symbolize_keys : tool + + # Already in nested format - return as is + if tool_hash[:type] == "function" && tool_hash[:function] + tool_hash + # Common format - convert to nested format + elsif tool_hash[:name] + { + type: "function", + function: { + name: tool_hash[:name], + description: tool_hash[:description], + parameters: tool_hash[:parameters] || tool_hash[:input_schema] + }.compact + } + else + tool_hash + end + end + end + + # Normalizes tool_choice from common format to OpenAI Chat API format. + # + # Accepts: + # - "auto" (common) → "auto" (passthrough) + # - "required" (common) → "required" (passthrough) + # - `{name: "..."}` (common) → `{type: "function", function: {name: "..."}}` + # - Already nested format → passthrough + # + # @param tool_choice [String, Hash, Symbol] + # @return [String, Hash, Symbol] + def normalize_tool_choice(tool_choice) + case tool_choice + when "auto", :auto, "required", :required + # Passthrough - Chat API accepts these directly + tool_choice.to_s + when Hash + tool_choice_hash = tool_choice.deep_symbolize_keys + + # Already in nested format with type and function keys + if tool_choice_hash[:type] == "function" && tool_choice_hash[:function] + tool_choice_hash + # Common format with just name - convert to nested format + elsif tool_choice_hash[:name] + { + type: "function", + function: { + name: tool_choice_hash[:name] + } + } + else + tool_choice_hash + end + else + tool_choice + end + end + # Normalizes instructions to developer message format # # Converts instructions into developer messages with proper content structure. diff --git a/lib/active_agent/providers/open_ai/chat_provider.rb b/lib/active_agent/providers/open_ai/chat_provider.rb index 75933901..5905ad2e 100644 --- a/lib/active_agent/providers/open_ai/chat_provider.rb +++ b/lib/active_agent/providers/open_ai/chat_provider.rb @@ -30,6 +30,40 @@ def api_prompt_executer client.chat.completions end + # @see BaseProvider#prepare_prompt_request + # @return [Request] + def prepare_prompt_request + prepare_prompt_request_tools + super + end + + # @api private + def prepare_prompt_request_tools + return unless request.tool_choice + + # Get list of function calls that have been made + # In Chat API, tool calls are in the assistant message's tool_calls array + functions_used = message_stack + .select { |msg| msg[:role] == "assistant" && msg[:tool_calls] } + .flat_map { |msg| msg[:tool_calls] } + .map { |tc| tc.dig(:function, :name) } + .compact + + # Check if tool_choice is a hash (specific tool) or string (auto/required) + if request.tool_choice.is_a?(Hash) + # Specific tool choice - clear if that tool was used + tool_choice_name = request.tool_choice.dig(:function, :name) + if tool_choice_name && functions_used.include?(tool_choice_name) + request.tool_choice = nil + end + elsif request.tool_choice == "required" + # Required tool choice - clear if any tool was used + if functions_used.any? + request.tool_choice = nil + end + end + end + # @see BaseProvider#api_response_normalize # @param api_response [OpenAI::Models::ChatCompletion] # @return [Hash] normalized response hash diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_input_schema.yml b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_input_schema.yml new file mode 100644 index 00000000..ccde4796 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_input_schema.yml @@ -0,0 +1,266 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather in Boston?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '376' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:40 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '570' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '583' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999990' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_2e8785ee0f3e48428b408c0743c4cefa + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=i1rrR0yVKiIrh2CIzRIVYKUU6hIt3wDGTOSvmkohsLs-1763533180-1.0.1.1-X6FcBvIlr8K4zU0eQ3hg79iH.sB_ivFXrby2zEvSY8eMM8Ob9RPPdoO6kZ49kDEkB1Q4rSS.qJgAvFEPEbHSHLdRVmFIvQgPRuGq5HpnD7I; + path=/; expires=Wed, 19-Nov-25 06:49:40 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=fJeWpVHtUQBN.20JB.y3oSuc0zZGCHWGFNARyoxSglU-1763533180771-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98e7993a67b2-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: | + { + "id": "chatcmpl-CdVpUCLVDhitsbz3SEYCmlzlAnzYU", + "object": "chat.completion", + "created": 1763533180, + "model": "gpt-4o-mini-2024-07-18", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_5ml0rdF2bbqFWoNL7mYVVXEd", + "type": "function", + "function": { + "name": "get_weather", + "arguments": "{\"location\":\"Boston, MA\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + "logprobs": null, + "finish_reason": "tool_calls" + } + ], + "usage": { + "prompt_tokens": 66, + "completion_tokens": 16, + "total_tokens": 82, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default", + "system_fingerprint": "fp_560af6e559" + } + recorded_at: Wed, 19 Nov 2025 06:19:40 GMT +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather in Boston?"},{"role":"assistant","refusal":null,"annotations":[],"tool_calls":[{"id":"call_5ml0rdF2bbqFWoNL7mYVVXEd","function":{"arguments":"{\"location\":\"Boston, + MA\"}","name":"get_weather"},"type":"function"}]},{"role":"tool","content":"{\"location\":\"Boston, + MA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_5ml0rdF2bbqFWoNL7mYVVXEd"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '735' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:41 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '538' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '551' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999970' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_16ba0436dc5a4325956f80ff1dab8665 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=3vr3.5qq7r7oWpeJ4H9I.Sxe7OJNK7lG3YLM.JurdfI-1763533181-1.0.1.1-VVa0Zj5rKWMs809pecws.aYAxN.gqIygBE4zEkTDs5zudTzNa1nBh0qA4rCDF_Sre0RetJ2YiC0ukVdWZKzlPB5.3ZHwZNQp0YawPLyH5VQ; + path=/; expires=Wed, 19-Nov-25 06:49:41 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=_k_X1Gy7NLBZsFz_Np3TA2sXsdXHYYKctk.79PDBplk-1763533181437-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98ec28132320-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: !binary |- + ewogICJpZCI6ICJjaGF0Y21wbC1DZFZwVWtBNWhFWDRVQzBYa2pvdUVQdTVLWkc3ViIsCiAgIm9iamVjdCI6ICJjaGF0LmNvbXBsZXRpb24iLAogICJjcmVhdGVkIjogMTc2MzUzMzE4MCwKICAibW9kZWwiOiAiZ3B0LTRvLW1pbmktMjAyNC0wNy0xOCIsCiAgImNob2ljZXMiOiBbCiAgICB7CiAgICAgICJpbmRleCI6IDAsCiAgICAgICJtZXNzYWdlIjogewogICAgICAgICJyb2xlIjogImFzc2lzdGFudCIsCiAgICAgICAgImNvbnRlbnQiOiAiVGhlIHdlYXRoZXIgaW4gQm9zdG9uLCBNQSBpcyBjdXJyZW50bHkgNzLCsEYgYW5kIHN1bm55LiIsCiAgICAgICAgInJlZnVzYWwiOiBudWxsLAogICAgICAgICJhbm5vdGF0aW9ucyI6IFtdCiAgICAgIH0sCiAgICAgICJsb2dwcm9icyI6IG51bGwsCiAgICAgICJmaW5pc2hfcmVhc29uIjogInN0b3AiCiAgICB9CiAgXSwKICAidXNhZ2UiOiB7CiAgICAicHJvbXB0X3Rva2VucyI6IDEwNywKICAgICJjb21wbGV0aW9uX3Rva2VucyI6IDE1LAogICAgInRvdGFsX3Rva2VucyI6IDEyMiwKICAgICJwcm9tcHRfdG9rZW5zX2RldGFpbHMiOiB7CiAgICAgICJjYWNoZWRfdG9rZW5zIjogMCwKICAgICAgImF1ZGlvX3Rva2VucyI6IDAKICAgIH0sCiAgICAiY29tcGxldGlvbl90b2tlbnNfZGV0YWlscyI6IHsKICAgICAgInJlYXNvbmluZ190b2tlbnMiOiAwLAogICAgICAiYXVkaW9fdG9rZW5zIjogMCwKICAgICAgImFjY2VwdGVkX3ByZWRpY3Rpb25fdG9rZW5zIjogMCwKICAgICAgInJlamVjdGVkX3ByZWRpY3Rpb25fdG9rZW5zIjogMAogICAgfQogIH0sCiAgInNlcnZpY2VfdGllciI6ICJkZWZhdWx0IiwKICAic3lzdGVtX2ZpbmdlcnByaW50IjogImZwXzU2MGFmNmU1NTkiCn0K + recorded_at: Wed, 19 Nov 2025 06:19:41 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml new file mode 100644 index 00000000..41472fd3 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml @@ -0,0 +1,275 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather in NYC and what''s 5 plus 3?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}},{"type":"function","function":{"name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '606' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:38 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '1499' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '1512' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999987' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_8e3adfd22084496b9afcbd29f61bc837 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=ZDhVnbKJgDJGn800jvM.PZfjfbvE4zCjNXZGqoZql84-1763533178-1.0.1.1-g64kkkW4sGyKlHu5am8LIMdR2x1LndffpxapeOT9iybq_tb6y2oowcUMTZAEPTDd.Ox_KEQvjwhI1ppJgwwQNZfDx1b70PkQYMsQx7nukgQ; + path=/; expires=Wed, 19-Nov-25 06:49:38 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=_VOjYb2ODB2mKFxo3BE.aVL7nLoVqYhgUeSSZgxM5c8-1763533178827-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98d57d1a67d9-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: | + { + "id": "chatcmpl-CdVpRQKSNGceawDsCb1qXM4zU9XSc", + "object": "chat.completion", + "created": 1763533177, + "model": "gpt-4o-mini-2024-07-18", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_i8WxtsPg3J1MGzu9r7ZPUulR", + "type": "function", + "function": { + "name": "get_weather", + "arguments": "{\"location\": \"New York City\"}" + } + }, + { + "id": "call_vZkmLcCQjrNAygM9N5BHRVFH", + "type": "function", + "function": { + "name": "calculate", + "arguments": "{\"operation\": \"add\", \"a\": 5, \"b\": 3}" + } + } + ], + "refusal": null, + "annotations": [] + }, + "logprobs": null, + "finish_reason": "tool_calls" + } + ], + "usage": { + "prompt_tokens": 93, + "completion_tokens": 53, + "total_tokens": 146, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default", + "system_fingerprint": "fp_560af6e559" + } + recorded_at: Wed, 19 Nov 2025 06:19:38 GMT +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather in NYC and what''s 5 plus 3?"},{"role":"assistant","refusal":null,"annotations":[],"tool_calls":[{"id":"call_i8WxtsPg3J1MGzu9r7ZPUulR","function":{"arguments":"{\"location\": + \"New York City\"}","name":"get_weather"},"type":"function"},{"id":"call_vZkmLcCQjrNAygM9N5BHRVFH","function":{"arguments":"{\"operation\": + \"add\", \"a\": 5, \"b\": 3}","name":"calculate"},"type":"function"}]},{"role":"tool","content":"{\"location\":\"New + York City\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_i8WxtsPg3J1MGzu9r7ZPUulR"},{"role":"tool","content":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","tool_call_id":"call_vZkmLcCQjrNAygM9N5BHRVFH"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}},{"type":"function","function":{"name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1248' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:40 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '963' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '975' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999955' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_ad2f5b53733c4bffb7eb64dba54f6ade + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=SuqXPX0K1iEA_4kiC7NSWn2obgTaFu6C3xI4Ug0FFRg-1763533180-1.0.1.1-nrKZPVxQATV5tEDwTqyABFY9gjqS4Ns34vjr5vNuP69GixyjRGgs1qQmjol0CnC6lqTTTkmVCAXeXRmNrKW.5s.mbY7VDwYK0kkVGfIHmOA; + path=/; expires=Wed, 19-Nov-25 06:49:40 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=qDdGdsF64Zlc5NBKVmxMFFknA8SeaLwB5I3mBV8R.Mg-1763533180022-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98e07ed9251d-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: !binary |- + ewogICJpZCI6ICJjaGF0Y21wbC1DZFZwVDFzcldwcDY2T2IwN3VvckpXZm9xMURuciIsCiAgIm9iamVjdCI6ICJjaGF0LmNvbXBsZXRpb24iLAogICJjcmVhdGVkIjogMTc2MzUzMzE3OSwKICAibW9kZWwiOiAiZ3B0LTRvLW1pbmktMjAyNC0wNy0xOCIsCiAgImNob2ljZXMiOiBbCiAgICB7CiAgICAgICJpbmRleCI6IDAsCiAgICAgICJtZXNzYWdlIjogewogICAgICAgICJyb2xlIjogImFzc2lzdGFudCIsCiAgICAgICAgImNvbnRlbnQiOiAiVGhlIHdlYXRoZXIgaW4gTmV3IFlvcmsgQ2l0eSBpcyBzdW5ueSB3aXRoIGEgdGVtcGVyYXR1cmUgb2YgNzLCsEYuIEFkZGl0aW9uYWxseSwgNSBwbHVzIDMgZXF1YWxzIDguIiwKICAgICAgICAicmVmdXNhbCI6IG51bGwsCiAgICAgICAgImFubm90YXRpb25zIjogW10KICAgICAgfSwKICAgICAgImxvZ3Byb2JzIjogbnVsbCwKICAgICAgImZpbmlzaF9yZWFzb24iOiAic3RvcCIKICAgIH0KICBdLAogICJ1c2FnZSI6IHsKICAgICJwcm9tcHRfdG9rZW5zIjogMjA3LAogICAgImNvbXBsZXRpb25fdG9rZW5zIjogMjgsCiAgICAidG90YWxfdG9rZW5zIjogMjM1LAogICAgInByb21wdF90b2tlbnNfZGV0YWlscyI6IHsKICAgICAgImNhY2hlZF90b2tlbnMiOiAwLAogICAgICAiYXVkaW9fdG9rZW5zIjogMAogICAgfSwKICAgICJjb21wbGV0aW9uX3Rva2Vuc19kZXRhaWxzIjogewogICAgICAicmVhc29uaW5nX3Rva2VucyI6IDAsCiAgICAgICJhdWRpb190b2tlbnMiOiAwLAogICAgICAiYWNjZXB0ZWRfcHJlZGljdGlvbl90b2tlbnMiOiAwLAogICAgICAicmVqZWN0ZWRfcHJlZGljdGlvbl90b2tlbnMiOiAwCiAgICB9CiAgfSwKICAic2VydmljZV90aWVyIjogImRlZmF1bHQiLAogICJzeXN0ZW1fZmluZ2VycHJpbnQiOiAiZnBfNTYwYWY2ZTU1OSIKfQo= + recorded_at: Wed, 19 Nov 2025 06:19:40 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_parameters.yml b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_parameters.yml new file mode 100644 index 00000000..d5978684 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_parameters.yml @@ -0,0 +1,266 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather in San Francisco?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '383' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:43 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '719' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '1013' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999987' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_9c76bfcee9e24db6ba2125251b9bbe09 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=zHHfyU7QyWdEXM1r55P0QO.cCLTac4iQqUbRb2pG2gg-1763533183-1.0.1.1-7KKtxBMbDzMV9GIX3HDKiexgHoejU7ulHLiC7yTFQLXO_U70RQXbQ6nTK34kfaqP5qEzXWtvHilgveB9lRtlDf2pQcnFEx0bWQJ3pCnJMsg; + path=/; expires=Wed, 19-Nov-25 06:49:43 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=owN32mUuz6Fwi4TLjUfadE5l4Owuihm8AFUsdhcP86A-1763533183591-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98f0b91315d2-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: | + { + "id": "chatcmpl-CdVpWynp5v1j6c8ipsAX7bga0CL38", + "object": "chat.completion", + "created": 1763533182, + "model": "gpt-4o-mini-2024-07-18", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_aHSN78tOGn5Job0vM895ZA48", + "type": "function", + "function": { + "name": "get_weather", + "arguments": "{\"location\":\"San Francisco, CA\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + "logprobs": null, + "finish_reason": "tool_calls" + } + ], + "usage": { + "prompt_tokens": 67, + "completion_tokens": 17, + "total_tokens": 84, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default", + "system_fingerprint": "fp_560af6e559" + } + recorded_at: Wed, 19 Nov 2025 06:19:43 GMT +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather in San Francisco?"},{"role":"assistant","refusal":null,"annotations":[],"tool_calls":[{"id":"call_aHSN78tOGn5Job0vM895ZA48","function":{"arguments":"{\"location\":\"San + Francisco, CA\"}","name":"get_weather"},"type":"function"}]},{"role":"tool","content":"{\"location\":\"San + Francisco, CA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_aHSN78tOGn5Job0vM895ZA48"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '756' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:44 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '957' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '970' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999967' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_e5e91f6542cb401fbd39d17fd3925b82 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=C614RHk1epWhBNfPqCE_qW3TSyaop_2SSWJj3XKuZbc-1763533184-1.0.1.1-IVd5Tx8PIBhJT7zyPrVK93cXg5SHB2v2QF5ZqaVqazj5n1t3DknF1JFalXKq2egYJF2ei_mNQVeu9zytpKgDQQmQ4sOVzCwA8ZnLnRVNsV0; + path=/; expires=Wed, 19-Nov-25 06:49:44 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=3FbJap_puCS_vLUo2Ae8adAY.ZeEbul47TeohdzhfCc-1763533184733-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98fddb349465-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: !binary |- + ewogICJpZCI6ICJjaGF0Y21wbC1DZFZwWGlkRGlKcUU4S3NWeFFyWlFFQWpVR2dwQyIsCiAgIm9iamVjdCI6ICJjaGF0LmNvbXBsZXRpb24iLAogICJjcmVhdGVkIjogMTc2MzUzMzE4MywKICAibW9kZWwiOiAiZ3B0LTRvLW1pbmktMjAyNC0wNy0xOCIsCiAgImNob2ljZXMiOiBbCiAgICB7CiAgICAgICJpbmRleCI6IDAsCiAgICAgICJtZXNzYWdlIjogewogICAgICAgICJyb2xlIjogImFzc2lzdGFudCIsCiAgICAgICAgImNvbnRlbnQiOiAiVGhlIHdlYXRoZXIgaW4gU2FuIEZyYW5jaXNjbywgQ0EgaXMgY3VycmVudGx5IDcywrBGIGFuZCBzdW5ueS4iLAogICAgICAgICJyZWZ1c2FsIjogbnVsbCwKICAgICAgICAiYW5ub3RhdGlvbnMiOiBbXQogICAgICB9LAogICAgICAibG9ncHJvYnMiOiBudWxsLAogICAgICAiZmluaXNoX3JlYXNvbiI6ICJzdG9wIgogICAgfQogIF0sCiAgInVzYWdlIjogewogICAgInByb21wdF90b2tlbnMiOiAxMTAsCiAgICAiY29tcGxldGlvbl90b2tlbnMiOiAxNiwKICAgICJ0b3RhbF90b2tlbnMiOiAxMjYsCiAgICAicHJvbXB0X3Rva2Vuc19kZXRhaWxzIjogewogICAgICAiY2FjaGVkX3Rva2VucyI6IDAsCiAgICAgICJhdWRpb190b2tlbnMiOiAwCiAgICB9LAogICAgImNvbXBsZXRpb25fdG9rZW5zX2RldGFpbHMiOiB7CiAgICAgICJyZWFzb25pbmdfdG9rZW5zIjogMCwKICAgICAgImF1ZGlvX3Rva2VucyI6IDAsCiAgICAgICJhY2NlcHRlZF9wcmVkaWN0aW9uX3Rva2VucyI6IDAsCiAgICAgICJyZWplY3RlZF9wcmVkaWN0aW9uX3Rva2VucyI6IDAKICAgIH0KICB9LAogICJzZXJ2aWNlX3RpZXIiOiAiZGVmYXVsdCIsCiAgInN5c3RlbV9maW5nZXJwcmludCI6ICJmcF81NjBhZjZlNTU5Igp9Cg== + recorded_at: Wed, 19 Nov 2025 06:19:44 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml new file mode 100644 index 00000000..894ac160 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml @@ -0,0 +1,145 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '296' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:35 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '670' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '685' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999992' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_978be8700a244f8ba1c9bcde3c50c1e8 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=cNPTR03lR9YR_4D2gPDCCkNz3h1jHPyANvDH8ppQAOI-1763533175-1.0.1.1-1uGYIBRpbAxBs_x._dEhYzH4XzKUMtcfhVEYceAn47pSyMCTcRM0n02fupc45esvGNOWXtpk1BYai14mdSxMSzzzR_4GVzlanYDW.v5xweE; + path=/; expires=Wed, 19-Nov-25 06:49:35 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=sDnq4PWVIUj_YHakkCIyHieV6iSN3Lhn8fGZeHfAmBI-1763533175695-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98c738db67cd-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: | + { + "id": "chatcmpl-CdVpPg42Ew60O5JjNzLghk6TyM6EU", + "object": "chat.completion", + "created": 1763533175, + "model": "gpt-4o-mini-2024-07-18", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": "Could you please provide me with the location for which you would like to know the weather?", + "refusal": null, + "annotations": [] + }, + "logprobs": null, + "finish_reason": "stop" + } + ], + "usage": { + "prompt_tokens": 44, + "completion_tokens": 19, + "total_tokens": 63, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default", + "system_fingerprint": "fp_560af6e559" + } + recorded_at: Wed, 19 Nov 2025 06:19:35 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml new file mode 100644 index 00000000..adde4c2f --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml @@ -0,0 +1,264 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":"required"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '300' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:36 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '596' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '613' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999992' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_2c9edccf0e9741b987a7d8cd07144aa6 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=ElmZbT2aJvTXMXwlnGm767QREKqjbYhSivaWflTxDYg-1763533176-1.0.1.1-LQpebpjTT96aDTdmeILrUD3ROY7ZkVlQx5.ta0AlvBNzu4.1bN5Jw95dqjoS0kwgGM2TAKcJ3i9bQi3fGn26Kq73Ga_sOCP7QdJ_kdZty5s; + path=/; expires=Wed, 19-Nov-25 06:49:36 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=Szz5KpMMvuAVbiXch_IFwrlTVWORSkFf0PNZaaKbXXU-1763533176505-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98cccde1ee17-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: | + { + "id": "chatcmpl-CdVpPAZdaDib1KVsiM2evVA0OzEoR", + "object": "chat.completion", + "created": 1763533175, + "model": "gpt-4o-mini-2024-07-18", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_bzruxle33QGVUqAfXPmbniXc", + "type": "function", + "function": { + "name": "get_weather", + "arguments": "{\"location\":\"current location\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + "logprobs": null, + "finish_reason": "tool_calls" + } + ], + "usage": { + "prompt_tokens": 44, + "completion_tokens": 15, + "total_tokens": 59, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default", + "system_fingerprint": "fp_560af6e559" + } + recorded_at: Wed, 19 Nov 2025 06:19:36 GMT +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather?"},{"role":"assistant","refusal":null,"annotations":[],"tool_calls":[{"id":"call_bzruxle33QGVUqAfXPmbniXc","function":{"arguments":"{\"location\":\"current + location\"}","name":"get_weather"},"type":"function"}]},{"role":"tool","content":"{\"location\":\"current + location\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_bzruxle33QGVUqAfXPmbniXc"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":null}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '665' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:37 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '424' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '439' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999972' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_b9a8b734e2e34964b449ace9a97b1f5e + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=1VVGBZBPEXN4yQliDPNlBGb7FyeNdMAjhjjKHhULHOU-1763533177-1.0.1.1-XhdVBiYdx5UWPXnHzAXrIEa9ih7IEhmX23lhz8duzfeSb9_90E4VXkXPMDt8sKbiDR2ObLA1xVKvUOLwUWCog_3bhP7u6hF0vCznVpqYfyk; + path=/; expires=Wed, 19-Nov-25 06:49:37 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=tGUhZAM8i5Le9iUwmdV1_8dil0fvRL2xgYnm9hwN268-1763533177130-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d98d1c974dc0d-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: !binary |- + ewogICJpZCI6ICJjaGF0Y21wbC1DZFZwUVRCTFhNNzNPNkwxMWlLQUhaMDVJVEVxSCIsCiAgIm9iamVjdCI6ICJjaGF0LmNvbXBsZXRpb24iLAogICJjcmVhdGVkIjogMTc2MzUzMzE3NiwKICAibW9kZWwiOiAiZ3B0LTRvLW1pbmktMjAyNC0wNy0xOCIsCiAgImNob2ljZXMiOiBbCiAgICB7CiAgICAgICJpbmRleCI6IDAsCiAgICAgICJtZXNzYWdlIjogewogICAgICAgICJyb2xlIjogImFzc2lzdGFudCIsCiAgICAgICAgImNvbnRlbnQiOiAiVGhlIGN1cnJlbnQgd2VhdGhlciBpcyA3MsKwRiBhbmQgc3VubnkuIiwKICAgICAgICAicmVmdXNhbCI6IG51bGwsCiAgICAgICAgImFubm90YXRpb25zIjogW10KICAgICAgfSwKICAgICAgImxvZ3Byb2JzIjogbnVsbCwKICAgICAgImZpbmlzaF9yZWFzb24iOiAic3RvcCIKICAgIH0KICBdLAogICJ1c2FnZSI6IHsKICAgICJwcm9tcHRfdG9rZW5zIjogODMsCiAgICAiY29tcGxldGlvbl90b2tlbnMiOiAxMSwKICAgICJ0b3RhbF90b2tlbnMiOiA5NCwKICAgICJwcm9tcHRfdG9rZW5zX2RldGFpbHMiOiB7CiAgICAgICJjYWNoZWRfdG9rZW5zIjogMCwKICAgICAgImF1ZGlvX3Rva2VucyI6IDAKICAgIH0sCiAgICAiY29tcGxldGlvbl90b2tlbnNfZGV0YWlscyI6IHsKICAgICAgInJlYXNvbmluZ190b2tlbnMiOiAwLAogICAgICAiYXVkaW9fdG9rZW5zIjogMCwKICAgICAgImFjY2VwdGVkX3ByZWRpY3Rpb25fdG9rZW5zIjogMCwKICAgICAgInJlamVjdGVkX3ByZWRpY3Rpb25fdG9rZW5zIjogMAogICAgfQogIH0sCiAgInNlcnZpY2VfdGllciI6ICJkZWZhdWx0IiwKICAic3lzdGVtX2ZpbmdlcnByaW50IjogImZwXzU2MGFmNmU1NTkiCn0K + recorded_at: Wed, 19 Nov 2025 06:19:37 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml new file mode 100644 index 00000000..bf75bae9 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml @@ -0,0 +1,264 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":{"type":"function","function":{"name":"get_weather"}}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '343' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:45 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '548' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '571' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999992' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_e0ace4243c3941b29e2fb1372df37315 + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=n0dq6T0ebKEtYnIjOWbuy_rezLllJH0SEpv7TmFVqF8-1763533185-1.0.1.1-4TP3jAbGQrO5bEnDMcERuYaPCcevfEJHv6KyuxmBNuuXb094NQHPS8PGchA7swyVhgpcsxPXdRuCgtH3q0gnBSFb_.8oP9X8asbG3TBYzwo; + path=/; expires=Wed, 19-Nov-25 06:49:45 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=6CBvHysJu4rHoKQxg0KRHBd9Dn29._.dC._4oMAq5GQ-1763533185472-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d99050e0f159a-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: | + { + "id": "chatcmpl-CdVpY70udpeZfQDpZ7mIZyjG1FlwH", + "object": "chat.completion", + "created": 1763533184, + "model": "gpt-4o-mini-2024-07-18", + "choices": [ + { + "index": 0, + "message": { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_ygHSYPgcYpF3inglYk2Cem6r", + "type": "function", + "function": { + "name": "get_weather", + "arguments": "{\"location\":\"New York\"}" + } + } + ], + "refusal": null, + "annotations": [] + }, + "logprobs": null, + "finish_reason": "stop" + } + ], + "usage": { + "prompt_tokens": 53, + "completion_tokens": 6, + "total_tokens": 59, + "prompt_tokens_details": { + "cached_tokens": 0, + "audio_tokens": 0 + }, + "completion_tokens_details": { + "reasoning_tokens": 0, + "audio_tokens": 0, + "accepted_prediction_tokens": 0, + "rejected_prediction_tokens": 0 + } + }, + "service_tier": "default", + "system_fingerprint": "fp_560af6e559" + } + recorded_at: Wed, 19 Nov 2025 06:19:45 GMT +- request: + method: post + uri: https://api.openai.com/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"gpt-4o-mini","messages":[{"role":"user","content":"What''s + the weather?"},{"role":"assistant","refusal":null,"annotations":[],"tool_calls":[{"id":"call_ygHSYPgcYpF3inglYk2Cem6r","function":{"arguments":"{\"location\":\"New + York\"}","name":"get_weather"},"type":"function"}]},{"role":"tool","content":"{\"location\":\"New + York\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_ygHSYPgcYpF3inglYk2Cem6r"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":null}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '649' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 06:19:46 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Expose-Headers: + - X-Request-ID + Openai-Organization: + - ORGANIZATION_ID + Openai-Processing-Ms: + - '517' + Openai-Project: + - PROJECT_ID + Openai-Version: + - '2020-10-01' + X-Envoy-Upstream-Service-Time: + - '530' + X-Ratelimit-Limit-Requests: + - '30000' + X-Ratelimit-Limit-Tokens: + - '150000000' + X-Ratelimit-Remaining-Requests: + - '29999' + X-Ratelimit-Remaining-Tokens: + - '149999975' + X-Ratelimit-Reset-Requests: + - 2ms + X-Ratelimit-Reset-Tokens: + - 0s + X-Request-Id: + - req_716f82849f684aad8d6b8ab2fa8cc62a + X-Openai-Proxy-Wasm: + - v0.1 + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=xr7j0ft6BfSCBCwIfw5ZuMgpQvWN0fZe2UDLz8d6QDM-1763533186-1.0.1.1-lgBrXdE45SDPyqcGHN_UUYKO05GxwuGyYQM9nv8uv0b5l9l2icuESpLFpziwcfVDjxW1mOZ680qzGL57COT0nkudiIk8UF9fvSz_8YRWWXk; + path=/; expires=Wed, 19-Nov-25 06:49:46 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=177kKEkHd9lgHO6SeMzv8_YR2aQAILRhC7MIeU8k3Jo-1763533186126-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a0d99099d64efbd-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: !binary |- + ewogICJpZCI6ICJjaGF0Y21wbC1DZFZwWjBiVkZUV3pzZFdKeFdjU25FWkVSWXlFSiIsCiAgIm9iamVjdCI6ICJjaGF0LmNvbXBsZXRpb24iLAogICJjcmVhdGVkIjogMTc2MzUzMzE4NSwKICAibW9kZWwiOiAiZ3B0LTRvLW1pbmktMjAyNC0wNy0xOCIsCiAgImNob2ljZXMiOiBbCiAgICB7CiAgICAgICJpbmRleCI6IDAsCiAgICAgICJtZXNzYWdlIjogewogICAgICAgICJyb2xlIjogImFzc2lzdGFudCIsCiAgICAgICAgImNvbnRlbnQiOiAiVGhlIHdlYXRoZXIgaW4gTmV3IFlvcmsgaXMgY3VycmVudGx5IDcywrBGIGFuZCBzdW5ueS4iLAogICAgICAgICJyZWZ1c2FsIjogbnVsbCwKICAgICAgICAiYW5ub3RhdGlvbnMiOiBbXQogICAgICB9LAogICAgICAibG9ncHJvYnMiOiBudWxsLAogICAgICAiZmluaXNoX3JlYXNvbiI6ICJzdG9wIgogICAgfQogIF0sCiAgInVzYWdlIjogewogICAgInByb21wdF90b2tlbnMiOiA4MywKICAgICJjb21wbGV0aW9uX3Rva2VucyI6IDE0LAogICAgInRvdGFsX3Rva2VucyI6IDk3LAogICAgInByb21wdF90b2tlbnNfZGV0YWlscyI6IHsKICAgICAgImNhY2hlZF90b2tlbnMiOiAwLAogICAgICAiYXVkaW9fdG9rZW5zIjogMAogICAgfSwKICAgICJjb21wbGV0aW9uX3Rva2Vuc19kZXRhaWxzIjogewogICAgICAicmVhc29uaW5nX3Rva2VucyI6IDAsCiAgICAgICJhdWRpb190b2tlbnMiOiAwLAogICAgICAiYWNjZXB0ZWRfcHJlZGljdGlvbl90b2tlbnMiOiAwLAogICAgICAicmVqZWN0ZWRfcHJlZGljdGlvbl90b2tlbnMiOiAwCiAgICB9CiAgfSwKICAic2VydmljZV90aWVyIjogImRlZmF1bHQiLAogICJzeXN0ZW1fZmluZ2VycHJpbnQiOiAiZnBfNTYwYWY2ZTU1OSIKfQo= + recorded_at: Wed, 19 Nov 2025 06:19:46 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/open_ai/chat/common_format/tools_test.rb b/test/integration/open_ai/chat/common_format/tools_test.rb new file mode 100644 index 00000000..7581731e --- /dev/null +++ b/test/integration/open_ai/chat/common_format/tools_test.rb @@ -0,0 +1,384 @@ +# frozen_string_literal: true + +require_relative "../../../test_helper" + +module Integration + module OpenAI + module Chat + module CommonFormat + class ToolsTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4o-mini", api_version: :chat + + def get_weather(location:) + { location: location, temperature: "72°F", conditions: "sunny" } + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + + # Common format with 'parameters' key (recommended) + COMMON_FORMAT_PARAMETERS = { + model: "gpt-4o-mini", + messages: [ + { + role: "user", + content: "What's the weather in San Francisco?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + } + ] + } + def common_format_parameters + prompt( + message: "What's the weather in San Francisco?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + def get_weather(location:) + { location: location, temperature: "72°F", conditions: "sunny" } + end + + # Common format with 'input_schema' key (should normalize to parameters) + COMMON_FORMAT_INPUT_SCHEMA = { + model: "gpt-4o-mini", + messages: [ + { + role: "user", + content: "What's the weather in Boston?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + } + ] + } + + def common_format_input_schema + prompt( + message: "What's the weather in Boston?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Multiple tools in common format + COMMON_FORMAT_MULTIPLE_TOOLS = { + model: "gpt-4o-mini", + messages: [ + { + role: "user", + content: "What's the weather in NYC and what's 5 plus 3?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + }, + { + type: "function", + function: { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + } + ] + } + def common_format_multiple_tools + prompt( + message: "What's the weather in NYC and what's 5 plus 3?", + tools: [ + { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ] + ) + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + + # Tool choice - string format + COMMON_FORMAT_TOOL_CHOICE_AUTO = { + model: "gpt-4o-mini", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: "auto" + } + def common_format_tool_choice_auto + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Tool choice - force tool use with "required" + COMMON_FORMAT_TOOL_CHOICE_REQUIRED = { + model: "gpt-4o-mini", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: "required" + } + def common_format_tool_choice_required + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "required" + ) + end + + # Tool choice - specific tool + COMMON_FORMAT_TOOL_CHOICE_SPECIFIC = { + model: "gpt-4o-mini", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: { + type: "function", + function: { + name: "get_weather" + } + } + } + def common_format_tool_choice_specific + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { name: "get_weather" } + ) + end + end + + ################################################################################ + # This automatically runs all the tests for the test actions + ################################################################################ + [ + :common_format_parameters, + :common_format_input_schema, + :common_format_multiple_tools, + :common_format_tool_choice_auto, + :common_format_tool_choice_required, + :common_format_tool_choice_specific + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) + end + end + end + end + end +end From 57c7e9d1eef98c97b83919840ca86c12ade2a818 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 09:17:36 -0800 Subject: [PATCH 04/17] Add OpenRouter Tools Common Format --- .../providers/open_router/request.rb | 20 + .../providers/open_router/transforms.rb | 30 ++ .../providers/open_router_provider.rb | 34 ++ .../test_agent_common_format_input_schema.yml | 159 ++++++++ ...est_agent_common_format_multiple_tools.yml | 157 ++++++++ .../test_agent_common_format_parameters.yml | 161 ++++++++ ...t_agent_common_format_tool_choice_auto.yml | 79 ++++ ...ent_common_format_tool_choice_required.yml | 79 ++++ ...ent_common_format_tool_choice_specific.yml | 155 ++++++++ .../open_router/common_format/tools_test.rb | 367 ++++++++++++++++++ 10 files changed, 1241 insertions(+) create mode 100644 test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_input_schema.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_multiple_tools.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_parameters.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_required.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml create mode 100644 test/integration/open_router/common_format/tools_test.rb diff --git a/lib/active_agent/providers/open_router/request.rb b/lib/active_agent/providers/open_router/request.rb index 4896abf2..7ba8fb11 100644 --- a/lib/active_agent/providers/open_router/request.rb +++ b/lib/active_agent/providers/open_router/request.rb @@ -148,6 +148,26 @@ def instructions=(*values) self.messages = instructions_messages + current_messages end + # Gets tool_choice bypassing gem validation + # + # OpenRouter supports "any" which isn't valid in OpenAI gem types. + # + # @return [String, Hash, nil] + def tool_choice + __getobj__.instance_variable_get(:@data)[:tool_choice] + end + + # Sets tool_choice bypassing gem validation + # + # OpenRouter supports "any" which isn't valid in OpenAI gem types, + # so we bypass the gem's type validation by setting @data directly. + # + # @param value [String, Hash, nil] + # @return [void] + def tool_choice=(value) + __getobj__.instance_variable_get(:@data)[:tool_choice] = value + end + # Accessor for OpenRouter-specific provider preferences # # @return [Hash, nil] diff --git a/lib/active_agent/providers/open_router/transforms.rb b/lib/active_agent/providers/open_router/transforms.rb index c820364f..f5e4f388 100644 --- a/lib/active_agent/providers/open_router/transforms.rb +++ b/lib/active_agent/providers/open_router/transforms.rb @@ -62,9 +62,39 @@ def normalize_params(params) # Use OpenAI transforms for the base parameters openai_params = OpenAI::Chat::Transforms.normalize_params(params) + # Override tool_choice normalization for OpenRouter's "any" vs "required" difference + if openai_params[:tool_choice] + openai_params[:tool_choice] = normalize_tool_choice(openai_params[:tool_choice]) + end + [ openai_params, openrouter_params ] end + # Normalizes tools using OpenAI transforms + # + # @param tools [Array] + # @return [Array] + def normalize_tools(tools) + OpenAI::Chat::Transforms.normalize_tools(tools) + end + + # Normalizes tool_choice for OpenRouter API differences + # + # OpenRouter uses "any" instead of OpenAI's "required" for forcing tool use. + # Converts common format to OpenRouter-specific format: + # - "required" (common) → "any" (OpenRouter) + # - Everything else delegates to OpenAI transforms + # + # @param tool_choice [String, Hash, Symbol] + # @return [String, Hash, Symbol] + def normalize_tool_choice(tool_choice) + # Convert "required" to OpenRouter's "any" + return "any" if tool_choice.to_s == "required" + + # For everything else, use OpenAI transforms + OpenAI::Chat::Transforms.normalize_tool_choice(tool_choice) + end + # Normalizes messages using OpenAI transforms # # @param messages [Array, String, Hash, nil] diff --git a/lib/active_agent/providers/open_router_provider.rb b/lib/active_agent/providers/open_router_provider.rb index cce5ff92..1204d6bc 100644 --- a/lib/active_agent/providers/open_router_provider.rb +++ b/lib/active_agent/providers/open_router_provider.rb @@ -33,6 +33,40 @@ def self.prompt_request_type protected + # @see BaseProvider#prepare_prompt_request + # @return [Request] + def prepare_prompt_request + prepare_prompt_request_tools + super + end + + # @api private + def prepare_prompt_request_tools + return unless request.tool_choice + + # Get list of function calls that have been made + # In Chat API, tool calls are in the assistant message's tool_calls array + functions_used = message_stack + .select { |msg| msg[:role] == "assistant" && msg[:tool_calls] } + .flat_map { |msg| msg[:tool_calls] } + .map { |tc| tc.dig(:function, :name) } + .compact + + # Check if tool_choice is a hash (specific tool) or string (auto/any) + if request.tool_choice.is_a?(Hash) + # Specific tool choice - clear if that tool was used + tool_choice_name = request.tool_choice.dig(:function, :name) + if tool_choice_name && functions_used.include?(tool_choice_name) + request.tool_choice = nil + end + elsif request.tool_choice == "any" + # OpenRouter uses "any" for required - clear if any tool was used + if functions_used.any? + request.tool_choice = nil + end + end + end + # Merges streaming delta into the message with role cleanup. # # Overrides parent to handle OpenRouter's role copying behavior which duplicates diff --git a/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_input_schema.yml b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_input_schema.yml new file mode 100644 index 00000000..56882366 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_input_schema.yml @@ -0,0 +1,159 @@ +--- +http_interactions: +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather in Boston?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '392' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:08 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1148f44c17ca3e-SJC + body: + encoding: ASCII-8BIT + string: "\n \n{\"id\":\"gen-1763571848-XSghN22Ea3eUi5fgPXm0\",\"provider\":\"Google\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763571848,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"tool_calls\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"\",\"refusal\":null,\"reasoning\":null,\"tool_calls\":[{\"index\":0,\"id\":\"tool_get_weather_yl7m3XDXuvnK3iiaGsjB\",\"type\":\"function\",\"function\":{\"name\":\"get_weather\",\"arguments\":\"{\\\"location\\\":\\\"Boston, + MA\\\"}\"}}]}}],\"usage\":{\"prompt_tokens\":36,\"completion_tokens\":7,\"total_tokens\":43,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 17:04:08 GMT +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather in Boston?"},{"role":"assistant","content":"","refusal":null,"tool_calls":[{"id":"tool_get_weather_yl7m3XDXuvnK3iiaGsjB","function":{"arguments":"{\"location\":\"Boston, + MA\"}","name":"get_weather"},"type":"function","index":0}],"reasoning":null},{"role":"tool","content":"{\"location\":\"Boston, + MA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"tool_get_weather_yl7m3XDXuvnK3iiaGsjB"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '790' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:09 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1148f809191949-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + CiAgICAgICAgIAp7ImlkIjoiZ2VuLTE3NjM1NzE4NDgtajFuaWt4dDBNeUVSMDh0Vmo2dW0iLCJwcm92aWRlciI6Ikdvb2dsZSIsIm1vZGVsIjoiZ29vZ2xlL2dlbWluaS0yLjAtZmxhc2gtMDAxIiwib2JqZWN0IjoiY2hhdC5jb21wbGV0aW9uIiwiY3JlYXRlZCI6MTc2MzU3MTg0OSwiY2hvaWNlcyI6W3sibG9ncHJvYnMiOm51bGwsImZpbmlzaF9yZWFzb24iOiJzdG9wIiwibmF0aXZlX2ZpbmlzaF9yZWFzb24iOiJTVE9QIiwiaW5kZXgiOjAsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOiJUaGUgd2VhdGhlciBpbiBCb3N0b24sIE1BIGlzIHN1bm55IHdpdGggYSB0ZW1wZXJhdHVyZSBvZiA3MsKwRi5cbiIsInJlZnVzYWwiOm51bGwsInJlYXNvbmluZyI6bnVsbH19XSwidXNhZ2UiOnsicHJvbXB0X3Rva2VucyI6NzUsImNvbXBsZXRpb25fdG9rZW5zIjoxOSwidG90YWxfdG9rZW5zIjo5NCwicHJvbXB0X3Rva2Vuc19kZXRhaWxzIjp7ImNhY2hlZF90b2tlbnMiOjB9LCJjb21wbGV0aW9uX3Rva2Vuc19kZXRhaWxzIjp7InJlYXNvbmluZ190b2tlbnMiOjAsImltYWdlX3Rva2VucyI6MH19fQ== + recorded_at: Wed, 19 Nov 2025 17:04:09 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_multiple_tools.yml b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_multiple_tools.yml new file mode 100644 index 00000000..f8a35331 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_multiple_tools.yml @@ -0,0 +1,157 @@ +--- +http_interactions: +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather in NYC and what''s 5 plus 3?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}},{"type":"function","function":{"name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '622' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:10 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1148fc0bea2847-SJC + body: + encoding: ASCII-8BIT + string: "\n \n{\"id\":\"gen-1763571849-fMuPYD0BKShyHqTXgXvN\",\"provider\":\"Google + AI Studio\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763571849,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"tool_calls\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"\",\"refusal\":null,\"reasoning\":null,\"tool_calls\":[{\"index\":0,\"id\":\"tool_get_weather_tIDAj9UZ8o1c2mXwibMI\",\"type\":\"function\",\"function\":{\"name\":\"get_weather\",\"arguments\":\"{\\\"location\\\":\\\"NYC\\\"}\"}},{\"index\":1,\"id\":\"tool_calculate_ltXKJIZfWAss1yWNoDSt\",\"type\":\"function\",\"function\":{\"name\":\"calculate\",\"arguments\":\"{\\\"a\\\":5,\\\"operation\\\":\\\"add\\\",\\\"b\\\":3}\"}}]}}],\"usage\":{\"prompt_tokens\":46,\"completion_tokens\":12,\"total_tokens\":58,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 17:04:10 GMT +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather in NYC and what''s 5 plus 3?"},{"role":"assistant","content":"","refusal":null,"tool_calls":[{"id":"tool_get_weather_tIDAj9UZ8o1c2mXwibMI","function":{"arguments":"{\"location\":\"NYC\"}","name":"get_weather"},"type":"function","index":0},{"id":"tool_calculate_ltXKJIZfWAss1yWNoDSt","function":{"arguments":"{\"a\":5,\"operation\":\"add\",\"b\":3}","name":"calculate"},"type":"function","index":1}],"reasoning":null},{"role":"tool","content":"{\"location\":\"NYC\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"tool_get_weather_tIDAj9UZ8o1c2mXwibMI"},{"role":"tool","content":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","tool_call_id":"tool_calculate_ltXKJIZfWAss1yWNoDSt"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}},{"type":"function","function":{"name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1299' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:10 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1149011f7dcf26-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + CiAgICAgICAgIAp7ImlkIjoiZ2VuLTE3NjM1NzE4NTAtWGUwenNoMmVyQ3htTDE0czJZeFUiLCJwcm92aWRlciI6Ikdvb2dsZSBBSSBTdHVkaW8iLCJtb2RlbCI6Imdvb2dsZS9nZW1pbmktMi4wLWZsYXNoLTAwMSIsIm9iamVjdCI6ImNoYXQuY29tcGxldGlvbiIsImNyZWF0ZWQiOjE3NjM1NzE4NTAsImNob2ljZXMiOlt7ImxvZ3Byb2JzIjpudWxsLCJmaW5pc2hfcmVhc29uIjoic3RvcCIsIm5hdGl2ZV9maW5pc2hfcmVhc29uIjoiU1RPUCIsImluZGV4IjowLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjoiVGhlIHdlYXRoZXIgaW4gTllDIGlzIHN1bm55IHdpdGggYSB0ZW1wZXJhdHVyZSBvZiA3MsKwRi4gNSBwbHVzIDMgaXMgOC5cbiIsInJlZnVzYWwiOm51bGwsInJlYXNvbmluZyI6bnVsbH19XSwidXNhZ2UiOnsicHJvbXB0X3Rva2VucyI6MTEyLCJjb21wbGV0aW9uX3Rva2VucyI6MjYsInRvdGFsX3Rva2VucyI6MTM4LCJwcm9tcHRfdG9rZW5zX2RldGFpbHMiOnsiY2FjaGVkX3Rva2VucyI6MH0sImNvbXBsZXRpb25fdG9rZW5zX2RldGFpbHMiOnsicmVhc29uaW5nX3Rva2VucyI6MCwiaW1hZ2VfdG9rZW5zIjowfX19 + recorded_at: Wed, 19 Nov 2025 17:04:11 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_parameters.yml b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_parameters.yml new file mode 100644 index 00000000..5adb7d17 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_parameters.yml @@ -0,0 +1,161 @@ +--- +http_interactions: +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather in San Francisco?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '399' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:17 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a11490bbbe5efbd-SJC + body: + encoding: ASCII-8BIT + string: "\n \n\n \n\n \n\n \n\n \n\n + \ \n\n \n\n \n\n \n\n \n\n \n\n + \ \n\n \n{\"id\":\"gen-1763571852-oRsToHyDzpZb1RvUO6ow\",\"provider\":\"Google\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763571857,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"tool_calls\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"\",\"refusal\":null,\"reasoning\":null,\"tool_calls\":[{\"index\":0,\"id\":\"tool_get_weather_S9ahEhRffDshS18YeSbL\",\"type\":\"function\",\"function\":{\"name\":\"get_weather\",\"arguments\":\"{\\\"location\\\":\\\"San + Francisco, CA\\\"}\"}}]}}],\"usage\":{\"prompt_tokens\":37,\"completion_tokens\":8,\"total_tokens\":45,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 17:04:17 GMT +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather in San Francisco?"},{"role":"assistant","content":"","refusal":null,"tool_calls":[{"id":"tool_get_weather_S9ahEhRffDshS18YeSbL","function":{"arguments":"{\"location\":\"San + Francisco, CA\"}","name":"get_weather"},"type":"function","index":0}],"reasoning":null},{"role":"tool","content":"{\"location\":\"San + Francisco, CA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"tool_get_weather_S9ahEhRffDshS18YeSbL"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '811' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:18 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1149303ac3cdb3-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + CiAgICAgICAgIAp7ImlkIjoiZ2VuLTE3NjM1NzE4NTctUmpwSGsxVkZncW42eTJxQldnUk0iLCJwcm92aWRlciI6Ikdvb2dsZSBBSSBTdHVkaW8iLCJtb2RlbCI6Imdvb2dsZS9nZW1pbmktMi4wLWZsYXNoLTAwMSIsIm9iamVjdCI6ImNoYXQuY29tcGxldGlvbiIsImNyZWF0ZWQiOjE3NjM1NzE4NTgsImNob2ljZXMiOlt7ImxvZ3Byb2JzIjpudWxsLCJmaW5pc2hfcmVhc29uIjoic3RvcCIsIm5hdGl2ZV9maW5pc2hfcmVhc29uIjoiU1RPUCIsImluZGV4IjowLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjoiVGhlIHdlYXRoZXIgaW4gU2FuIEZyYW5jaXNjbywgQ0EgaXMgc3Vubnkgd2l0aCBhIHRlbXBlcmF0dXJlIG9mIDcywrBGLlxuIiwicmVmdXNhbCI6bnVsbCwicmVhc29uaW5nIjpudWxsfX1dLCJ1c2FnZSI6eyJwcm9tcHRfdG9rZW5zIjo3NywiY29tcGxldGlvbl90b2tlbnMiOjIwLCJ0b3RhbF90b2tlbnMiOjk3LCJwcm9tcHRfdG9rZW5zX2RldGFpbHMiOnsiY2FjaGVkX3Rva2VucyI6MH0sImNvbXBsZXRpb25fdG9rZW5zX2RldGFpbHMiOnsicmVhc29uaW5nX3Rva2VucyI6MCwiaW1hZ2VfdG9rZW5zIjowfX19 + recorded_at: Wed, 19 Nov 2025 17:04:18 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml new file mode 100644 index 00000000..fc9a872a --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml @@ -0,0 +1,79 @@ +--- +http_interactions: +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '312' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:11 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1149061e6f9cd0-SJC + body: + encoding: ASCII-8BIT + string: "\n \n{\"id\":\"gen-1763571851-y4ujPTAg7Fty3VNZ7xfH\",\"provider\":\"Google\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763571851,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"stop\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"What + is the location you want to know the weather for?\\n\",\"refusal\":null,\"reasoning\":null}}],\"usage\":{\"prompt_tokens\":15,\"completion_tokens\":13,\"total_tokens\":28,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 17:04:11 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_required.yml b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_required.yml new file mode 100644 index 00000000..a2380893 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_required.yml @@ -0,0 +1,79 @@ +--- +http_interactions: +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":"any"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '311' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:06 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1148e33af34f08-SJC + body: + encoding: ASCII-8BIT + string: "\n \n{\"id\":\"gen-1763571845-RJNe41mzAIbHwWZykimo\",\"provider\":\"Google\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763571845,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"stop\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"Could + you please tell me where are you located?\\n\",\"refusal\":null,\"reasoning\":null}}],\"usage\":{\"prompt_tokens\":15,\"completion_tokens\":11,\"total_tokens\":26,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 17:04:06 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml new file mode 100644 index 00000000..a39f9c06 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_router/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml @@ -0,0 +1,155 @@ +--- +http_interactions: +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":{"type":"function","function":{"name":"get_weather"}}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '359' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:07 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1148e8ce4eca3e-SJC + body: + encoding: ASCII-8BIT + string: "\n \n{\"id\":\"gen-1763571846-FETieOfNa9AWlT8DFqWa\",\"provider\":\"Google + AI Studio\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763571846,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"tool_calls\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"\",\"refusal\":null,\"reasoning\":null,\"tool_calls\":[{\"index\":0,\"id\":\"tool_get_weather_Q8eSZXDe3h4Z1rQ84pb0\",\"type\":\"function\",\"function\":{\"name\":\"get_weather\",\"arguments\":\"{\\\"location\\\":\\\"unknown\\\"}\"}}]}}],\"usage\":{\"prompt_tokens\":15,\"completion_tokens\":5,\"total_tokens\":20,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 17:04:07 GMT +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather?"},{"role":"assistant","content":"","refusal":null,"tool_calls":[{"id":"tool_get_weather_Q8eSZXDe3h4Z1rQ84pb0","function":{"arguments":"{\"location\":\"unknown\"}","name":"get_weather"},"type":"function","index":0}],"reasoning":null},{"role":"tool","content":"{\"location\":\"unknown\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"tool_get_weather_Q8eSZXDe3h4Z1rQ84pb0"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":null}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '702' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 17:04:08 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1148ef4a17ebed-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + CiAgICAgICAgIAp7ImlkIjoiZ2VuLTE3NjM1NzE4NDctTG01ZzJMV083NWxDdHlRbHlKMEoiLCJwcm92aWRlciI6Ikdvb2dsZSBBSSBTdHVkaW8iLCJtb2RlbCI6Imdvb2dsZS9nZW1pbmktMi4wLWZsYXNoLTAwMSIsIm9iamVjdCI6ImNoYXQuY29tcGxldGlvbiIsImNyZWF0ZWQiOjE3NjM1NzE4NDcsImNob2ljZXMiOlt7ImxvZ3Byb2JzIjpudWxsLCJmaW5pc2hfcmVhc29uIjoic3RvcCIsIm5hdGl2ZV9maW5pc2hfcmVhc29uIjoiU1RPUCIsImluZGV4IjowLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjoiVGhlIHdlYXRoZXIgaXMgc3Vubnkgd2l0aCBhIHRlbXBlcmF0dXJlIG9mIDcywrBGLiBUaGUgbG9jYXRpb24gaXMgdW5rbm93bi5cbiIsInJlZnVzYWwiOm51bGwsInJlYXNvbmluZyI6bnVsbH19XSwidXNhZ2UiOnsicHJvbXB0X3Rva2VucyI6NTIsImNvbXBsZXRpb25fdG9rZW5zIjoyMCwidG90YWxfdG9rZW5zIjo3MiwicHJvbXB0X3Rva2Vuc19kZXRhaWxzIjp7ImNhY2hlZF90b2tlbnMiOjB9LCJjb21wbGV0aW9uX3Rva2Vuc19kZXRhaWxzIjp7InJlYXNvbmluZ190b2tlbnMiOjAsImltYWdlX3Rva2VucyI6MH19fQ== + recorded_at: Wed, 19 Nov 2025 17:04:08 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/open_router/common_format/tools_test.rb b/test/integration/open_router/common_format/tools_test.rb new file mode 100644 index 00000000..a06eb99c --- /dev/null +++ b/test/integration/open_router/common_format/tools_test.rb @@ -0,0 +1,367 @@ +# frozen_string_literal: true + +require_relative "../../test_helper" + +module Integration + module OpenRouter + module CommonFormat + class ToolsTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :openrouter, model: "google/gemini-2.0-flash-001" + + def get_weather(location:) + { location: location, temperature: "72°F", conditions: "sunny" } + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + + # Common format with 'parameters' key (recommended) + COMMON_FORMAT_PARAMETERS = { + model: "google/gemini-2.0-flash-001", + messages: [ + { + role: "user", + content: "What's the weather in San Francisco?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + } + ] + } + def common_format_parameters + prompt( + message: "What's the weather in San Francisco?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Common format with 'input_schema' key (should normalize to parameters) + COMMON_FORMAT_INPUT_SCHEMA = { + model: "google/gemini-2.0-flash-001", + messages: [ + { + role: "user", + content: "What's the weather in Boston?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + } + ] + } + def common_format_input_schema + prompt( + message: "What's the weather in Boston?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Multiple tools in common format + COMMON_FORMAT_MULTIPLE_TOOLS = { + model: "google/gemini-2.0-flash-001", + messages: [ + { + role: "user", + content: "What's the weather in NYC and what's 5 plus 3?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + }, + { + type: "function", + function: { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + } + ] + } + def common_format_multiple_tools + prompt( + message: "What's the weather in NYC and what's 5 plus 3?", + tools: [ + { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ] + ) + end + + # Tool choice - string format + COMMON_FORMAT_TOOL_CHOICE_AUTO = { + model: "google/gemini-2.0-flash-001", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: "auto" + } + def common_format_tool_choice_auto + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Tool choice - force tool use with "required" + COMMON_FORMAT_TOOL_CHOICE_REQUIRED = { + model: "google/gemini-2.0-flash-001", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: "any" + } + def common_format_tool_choice_required + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "required" + ) + end + + # Tool choice - specific tool + COMMON_FORMAT_TOOL_CHOICE_SPECIFIC = { + model: "google/gemini-2.0-flash-001", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: { + type: "function", + function: { + name: "get_weather" + } + } + } + def common_format_tool_choice_specific + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { name: "get_weather" } + ) + end + end + + ################################################################################ + # This automatically runs all the tests for the test actions + ################################################################################ + [ + :common_format_parameters, + :common_format_input_schema, + :common_format_multiple_tools, + :common_format_tool_choice_auto, + :common_format_tool_choice_required, + :common_format_tool_choice_specific + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) + end + end + end + end +end From 55893b1a39a1c70e3c914b2a703b8c4506c9170c Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 09:23:33 -0800 Subject: [PATCH 05/17] Add Ollama Tools Common Format --- .../test_agent_common_format_input_schema.yml | 127 ++++++ ...est_agent_common_format_multiple_tools.yml | 144 +++++++ .../test_agent_common_format_parameters.yml | 133 +++++++ ...t_agent_common_format_tool_choice_auto.yml | 67 ++++ ...ent_common_format_tool_choice_required.yml | 68 ++++ ...ent_common_format_tool_choice_specific.yml | 69 ++++ .../ollama/chat/common_format/tools_test.rb | 369 ++++++++++++++++++ 7 files changed, 977 insertions(+) create mode 100644 test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_input_schema.yml create mode 100644 test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml create mode 100644 test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_parameters.yml create mode 100644 test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml create mode 100644 test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml create mode 100644 test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml create mode 100644 test/integration/ollama/chat/common_format/tools_test.rb diff --git a/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_input_schema.yml b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_input_schema.yml new file mode 100644 index 00000000..3434b2fe --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_input_schema.yml @@ -0,0 +1,127 @@ +--- +http_interactions: +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather in Boston?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '377' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:35 GMT + Content-Length: + - '775' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-883","object":"chat.completion","created":1763572835,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"","reasoning":"Okay, + the user is asking for the weather in Boston. Let me check the tools available. + There''s a get_weather function that requires a location parameter. The user + mentioned \"Boston,\" so I need to specify the location as Boston, MA to match + the format like San Francisco, CA. I''ll call the function with that parameter.\n","tool_calls":[{"id":"call_u167uvno","index":0,"type":"function","function":{"name":"get_weather","arguments":"{\"location\":\"Boston, + MA\"}"}}]},"finish_reason":"tool_calls"}],"usage":{"prompt_tokens":160,"completion_tokens":93,"total_tokens":253}} + + ' + recorded_at: Wed, 19 Nov 2025 17:20:35 GMT +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather in Boston?"},{"role":"assistant","content":"","tool_calls":[{"id":"call_u167uvno","function":{"arguments":"{\"location\":\"Boston, + MA\"}","name":"get_weather"},"type":"function","index":0}],"reasoning":"Okay, + the user is asking for the weather in Boston. Let me check the tools available. + There''s a get_weather function that requires a location parameter. The user + mentioned \"Boston,\" so I need to specify the location as Boston, MA to match + the format like San Francisco, CA. I''ll call the function with that parameter.\n"},{"role":"tool","content":"{\"location\":\"Boston, + MA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_u167uvno"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1031' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:39 GMT + Content-Length: + - '783' + body: + encoding: ASCII-8BIT + string: !binary |- + eyJpZCI6ImNoYXRjbXBsLTU4NSIsIm9iamVjdCI6ImNoYXQuY29tcGxldGlvbiIsImNyZWF0ZWQiOjE3NjM1NzI4MzksIm1vZGVsIjoicXdlbjM6bGF0ZXN0Iiwic3lzdGVtX2ZpbmdlcnByaW50IjoiZnBfb2xsYW1hIiwiY2hvaWNlcyI6W3siaW5kZXgiOjAsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOiJUaGUgY3VycmVudCB3ZWF0aGVyIGluIEJvc3RvbiwgTUEgaXMgNzLCsEYgYW5kIHN1bm55LiBJdCBsb29rcyBsaWtlIGEgcGxlYXNhbnQgZGF5ISIsInJlYXNvbmluZyI6Ik9rYXksIHRoZSB1c2VyIGFza2VkIGZvciB0aGUgd2VhdGhlciBpbiBCb3N0b24uIEkgY2FsbGVkIHRoZSBnZXRfd2VhdGhlciBmdW5jdGlvbiB3aXRoIEJvc3RvbiwgTUEgYXMgdGhlIGxvY2F0aW9uLiBUaGUgcmVzcG9uc2UgY2FtZSBiYWNrIHdpdGggYSB0ZW1wZXJhdHVyZSBvZiA3MsKwRiBhbmQgc3VubnkgY29uZGl0aW9ucy4gTm93IEkgbmVlZCB0byBwcmVzZW50IHRoaXMgaW5mb3JtYXRpb24gY2xlYXJseS4gTGV0IG1lIG1ha2Ugc3VyZSB0byBtZW50aW9uIHRoZSB0ZW1wZXJhdHVyZSwgY29uZGl0aW9ucywgYW5kIG1heWJlIGFkZCBhIGZyaWVuZGx5IG5vdGUgYWJvdXQgdGhlIHdlYXRoZXIgYmVpbmcgcGxlYXNhbnQuIEtlZXAgaXQgY29uY2lzZSBhbmQgc3RyYWlnaHRmb3J3YXJkLlxuIn0sImZpbmlzaF9yZWFzb24iOiJzdG9wIn1dLCJ1c2FnZSI6eyJwcm9tcHRfdG9rZW5zIjoyODEsImNvbXBsZXRpb25fdG9rZW5zIjoxMDUsInRvdGFsX3Rva2VucyI6Mzg2fX0K + recorded_at: Wed, 19 Nov 2025 17:20:39 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml new file mode 100644 index 00000000..fa320216 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_multiple_tools.yml @@ -0,0 +1,144 @@ +--- +http_interactions: +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather in NYC and what''s 5 plus 3?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}},{"type":"function","function":{"name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '607' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:21:09 GMT + Content-Length: + - '1602' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-547","object":"chat.completion","created":1763572869,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"","reasoning":"Okay, + let''s tackle this user query. The user is asking two things: the weather + in NYC and the result of 5 plus 3. \n\nFirst, I need to check which functions + are available. There''s get_weather for the weather info and calculate for + arithmetic operations. \n\nFor the first part, \"What''s the weather in NYC?\", + I should call get_weather with the location parameter set to \"NYC\". That + should retrieve the current weather data.\n\nNext, the second part is \"what''s + 5 plus 3?\" Here, the operation is addition. The calculate function requires + operation, a, and b. So operation is \"add\", a is 5, and b is 3. I''ll need + to make sure the parameters are correctly formatted as numbers.\n\nI need + to make two separate tool calls. First for get_weather with location NYC, + then for calculate with add, 5, and 3. Let me structure each tool call properly + in JSON within the XML tags. Double-checking the parameters to ensure they + match the required types and enums. No mistakes there. Alright, ready to output + the tool calls.\n","tool_calls":[{"id":"call_nwftoaxi","index":0,"type":"function","function":{"name":"get_weather","arguments":"{\"location\":\"NYC\"}"}},{"id":"call_vialglfv","index":1,"type":"function","function":{"name":"calculate","arguments":"{\"a\":5,\"b\":3,\"operation\":\"add\"}"}}]},"finish_reason":"tool_calls"}],"usage":{"prompt_tokens":215,"completion_tokens":285,"total_tokens":500}} + + ' + recorded_at: Wed, 19 Nov 2025 17:21:09 GMT +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather in NYC and what''s 5 plus 3?"},{"role":"assistant","content":"","tool_calls":[{"id":"call_nwftoaxi","function":{"arguments":"{\"location\":\"NYC\"}","name":"get_weather"},"type":"function","index":0},{"id":"call_vialglfv","function":{"arguments":"{\"a\":5,\"b\":3,\"operation\":\"add\"}","name":"calculate"},"type":"function","index":1}],"reasoning":"Okay, + let''s tackle this user query. The user is asking two things: the weather + in NYC and the result of 5 plus 3. \n\nFirst, I need to check which functions + are available. There''s get_weather for the weather info and calculate for + arithmetic operations. \n\nFor the first part, \"What''s the weather in NYC?\", + I should call get_weather with the location parameter set to \"NYC\". That + should retrieve the current weather data.\n\nNext, the second part is \"what''s + 5 plus 3?\" Here, the operation is addition. The calculate function requires + operation, a, and b. So operation is \"add\", a is 5, and b is 3. I''ll need + to make sure the parameters are correctly formatted as numbers.\n\nI need + to make two separate tool calls. First for get_weather with location NYC, + then for calculate with add, 5, and 3. Let me structure each tool call properly + in JSON within the XML tags. Double-checking the parameters to ensure they + match the required types and enums. No mistakes there. Alright, ready to output + the tool calls.\n"},{"role":"tool","content":"{\"location\":\"NYC\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","tool_call_id":"call_nwftoaxi"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}},{"type":"function","function":{"name":"calculate","description":"Perform + basic arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string","enum":["add","subtract","multiply","divide"]},"a":{"type":"number"},"b":{"type":"number"}},"required":["operation","a","b"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '2132' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:21:16 GMT + Content-Length: + - '895' + body: + encoding: ASCII-8BIT + string: !binary |- + eyJpZCI6ImNoYXRjbXBsLTg0NyIsIm9iamVjdCI6ImNoYXQuY29tcGxldGlvbiIsImNyZWF0ZWQiOjE3NjM1NzI4NzYsIm1vZGVsIjoicXdlbjM6bGF0ZXN0Iiwic3lzdGVtX2ZpbmdlcnByaW50IjoiZnBfb2xsYW1hIiwiY2hvaWNlcyI6W3siaW5kZXgiOjAsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOiJUaGUgY3VycmVudCB3ZWF0aGVyIGluIE5ZQyBpcyA3MsKwRiBhbmQgc3VubnkuIFxuXG41IHBsdXMgMyBlcXVhbHMgOC4iLCJyZWFzb25pbmciOiJPa2F5LCBsZXQncyBzZWUuIFRoZSB1c2VyIGFza2VkIGZvciB0d28gdGhpbmdzOiB0aGUgd2VhdGhlciBpbiBOWUMgYW5kIHRoZSBzdW0gb2YgNSBhbmQgMy4gRmlyc3QsIEkgbmVlZCB0byBwcm9jZXNzIHRoZSB3ZWF0aGVyIGRhdGEuIFRoZSB0b29sIHJlc3BvbnNlIHNheXMgaXQncyA3MsKwRiBhbmQgc3VubnkgaW4gTllDLiBHb3QgdGhhdC4gVGhlbiwgdGhlIGNhbGN1bGF0aW9uIHBhcnQ6IDUgcGx1cyAzIGVxdWFscyA4LiBUaGUgdG9vbCByZXNwb25zZSBjb25maXJtcyB0aGUgcmVzdWx0IGlzIDguIE5vdywgSSBzaG91bGQgcHJlc2VudCBib3RoIGFuc3dlcnMgY2xlYXJseS4gTGV0IG1lIHN0YXJ0IHdpdGggdGhlIHdlYXRoZXIsIG1lbnRpb24gdGhlIHRlbXBlcmF0dXJlIGFuZCBjb25kaXRpb25zLCB0aGVuIHN0YXRlIHRoZSBtYXRoIHJlc3VsdC4gTWFrZSBzdXJlIHRoZSByZXNwb25zZSBpcyBmcmllbmRseSBhbmQgc3RyYWlnaHRmb3J3YXJkLiBBbHJpZ2h0LCBwdXR0aW5nIGl0IGFsbCB0b2dldGhlci5cbiJ9LCJmaW5pc2hfcmVhc29uIjoic3RvcCJ9XSwidXNhZ2UiOnsicHJvbXB0X3Rva2VucyI6NTM1LCJjb21wbGV0aW9uX3Rva2VucyI6MTUxLCJ0b3RhbF90b2tlbnMiOjY4Nn19Cg== + recorded_at: Wed, 19 Nov 2025 17:21:16 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_parameters.yml b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_parameters.yml new file mode 100644 index 00000000..aa284f3b --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_parameters.yml @@ -0,0 +1,133 @@ +--- +http_interactions: +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather in San Francisco?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '384' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:44 GMT + Content-Length: + - '993' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-837","object":"chat.completion","created":1763572844,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"","reasoning":"Okay, + the user is asking for the weather in San Francisco. Let me check the tools + provided. There''s a function called get_weather that takes a location parameter. + The required parameter is location, which should be a string like \"San Francisco, + CA\". The user just said \"San Francisco\", so I should format that into the + required format. Let me make sure to include the state abbreviation. The function + doesn''t have any other parameters, so I just need to pass \"San Francisco, + CA\" as the location. Alright, that should do it.\n","tool_calls":[{"id":"call_1mebjeab","index":0,"type":"function","function":{"name":"get_weather","arguments":"{\"location\":\"San + Francisco, CA\"}"}}]},"finish_reason":"tool_calls"}],"usage":{"prompt_tokens":161,"completion_tokens":138,"total_tokens":299}} + + ' + recorded_at: Wed, 19 Nov 2025 17:20:44 GMT +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather in San Francisco?"},{"role":"assistant","content":"","tool_calls":[{"id":"call_1mebjeab","function":{"arguments":"{\"location\":\"San + Francisco, CA\"}","name":"get_weather"},"type":"function","index":0}],"reasoning":"Okay, + the user is asking for the weather in San Francisco. Let me check the tools + provided. There''s a function called get_weather that takes a location parameter. + The required parameter is location, which should be a string like \"San Francisco, + CA\". The user just said \"San Francisco\", so I should format that into the + required format. Let me make sure to include the state abbreviation. The function + doesn''t have any other parameters, so I just need to pass \"San Francisco, + CA\" as the location. Alright, that should do it.\n"},{"role":"tool","content":"{\"location\":\"San + Francisco, CA\",\"temperature\":\"72°F\",\"conditions\":\"sunny\"}","tool_call_id":"call_1mebjeab"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + the current weather in a given location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"The + city and state, e.g. San Francisco, CA"}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1262' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:49 GMT + Content-Length: + - '988' + body: + encoding: ASCII-8BIT + string: !binary |- + eyJpZCI6ImNoYXRjbXBsLTY0MSIsIm9iamVjdCI6ImNoYXQuY29tcGxldGlvbiIsImNyZWF0ZWQiOjE3NjM1NzI4NDksIm1vZGVsIjoicXdlbjM6bGF0ZXN0Iiwic3lzdGVtX2ZpbmdlcnByaW50IjoiZnBfb2xsYW1hIiwiY2hvaWNlcyI6W3siaW5kZXgiOjAsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOiJUaGUgY3VycmVudCB3ZWF0aGVyIGluIFNhbiBGcmFuY2lzY28sIENBIGlzICoqNzLCsEYqKiBhbmQgKipzdW5ueSoqLiBJdCBsb29rcyBsaWtlIGEgcGxlYXNhbnQgZGF5IHdpdGggY2xlYXIgc2tpZXMhIExldCBtZSBrbm93IGlmIHlvdSdkIGxpa2UgYWRkaXRpb25hbCBkZXRhaWxzLiDwn4yeIiwicmVhc29uaW5nIjoiT2theSwgdGhlIHVzZXIgYXNrZWQgZm9yIHRoZSB3ZWF0aGVyIGluIFNhbiBGcmFuY2lzY28uIEkgY2FsbGVkIHRoZSBnZXRfd2VhdGhlciBmdW5jdGlvbiB3aXRoIFwiU2FuIEZyYW5jaXNjbywgQ0FcIiBhcyB0aGUgbG9jYXRpb24uIFRoZSByZXNwb25zZSBjYW1lIGJhY2sgd2l0aCBhIHRlbXBlcmF0dXJlIG9mIDcywrBGIGFuZCBzdW5ueSBjb25kaXRpb25zLiBOb3cgSSBuZWVkIHRvIHByZXNlbnQgdGhpcyBpbmZvcm1hdGlvbiBjbGVhcmx5LiBMZXQgbWUgc3RhcnQgYnkgc3RhdGluZyB0aGUgY3VycmVudCB3ZWF0aGVyLCBtZW50aW9uIHRoZSB0ZW1wZXJhdHVyZSBhbmQgY29uZGl0aW9ucy4gS2VlcCBpdCBzaW1wbGUgYW5kIGZyaWVuZGx5LiBNYXliZSBhZGQgYSBub3RlIGFib3V0IHRoZSB3ZWF0aGVyIGJlaW5nIHBsZWFzYW50LiBNYWtlIHN1cmUgdGhlIHVzZXIga25vd3MgdGhleSBjYW4gYXNrIGZvciBtb3JlIGRldGFpbHMgaWYgbmVlZGVkLiBBbHJpZ2h0LCB0aGF0IHNob3VsZCBjb3ZlciBpdC5cbiJ9LCJmaW5pc2hfcmVhc29uIjoic3RvcCJ9XSwidXNhZ2UiOnsicHJvbXB0X3Rva2VucyI6MzI4LCJjb21wbGV0aW9uX3Rva2VucyI6MTU0LCJ0b3RhbF90b2tlbnMiOjQ4Mn19Cg== + recorded_at: Wed, 19 Nov 2025 17:20:49 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml new file mode 100644 index 00000000..89e45428 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_auto.yml @@ -0,0 +1,67 @@ +--- +http_interactions: +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":"auto"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '297' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:59 GMT + Content-Length: + - '858' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-691","object":"chat.completion","created":1763572859,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"Could + you please specify the location you''d like the weather information for?","reasoning":"Okay, + the user asked, \"What''s the weather?\" but didn''t specify the location. + I need to call the get_weather function, which requires a location parameter. + Since the user didn''t provide one, I should ask them to clarify where they + want the weather information for. Let me check the function details again + to confirm the parameters. The function requires ''location'' as a string. + Alright, so I can''t proceed without that info. Time to prompt the user for + the missing details.\n"},"finish_reason":"stop"}],"usage":{"prompt_tokens":137,"completion_tokens":118,"total_tokens":255}} + + ' + recorded_at: Wed, 19 Nov 2025 17:20:59 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml new file mode 100644 index 00000000..0793738c --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_required.yml @@ -0,0 +1,68 @@ +--- +http_interactions: +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":"required"}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '301' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:31 GMT + Content-Length: + - '988' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-615","object":"chat.completion","created":1763572831,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"I + need to know the location to check the weather. Could you please specify the + city or region you''re interested in?","reasoning":"Okay, the user asked, + \"What''s the weather?\" I need to figure out which function to use. The available + tool is get_weather, which requires a location parameter. But the user didn''t + specify a location. Hmm, maybe I should ask them for the location. Wait, the + function requires it, so I can''t proceed without that info. Let me check + the function details again. Yes, location is a required parameter. Since the + user didn''t mention where, I should prompt them to provide the location so + I can fetch the weather data. Alright, time to respond by asking for the location.\n"},"finish_reason":"stop"}],"usage":{"prompt_tokens":137,"completion_tokens":151,"total_tokens":288}} + + ' + recorded_at: Wed, 19 Nov 2025 17:20:31 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml new file mode 100644 index 00000000..48d9cd7a --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/ollama/chat/common_format/tools_test/test_agent_common_format_tool_choice_specific.yml @@ -0,0 +1,69 @@ +--- +http_interactions: +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + weather","parameters":{"type":"object","properties":{"location":{"type":"string"}},"required":["location"]}}}],"tool_choice":{"type":"function","function":{"name":"get_weather"}}}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '344' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 17:20:55 GMT + Content-Length: + - '1012' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-675","object":"chat.completion","created":1763572855,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"I + need to know the location to check the weather. Could you please specify the + city or region you''re interested in?","reasoning":"Okay, the user asked, + \"What''s the weather?\" I need to figure out which function to use. The available + tool is get_weather, which requires a location parameter. But the user didn''t + specify a location. Hmm, maybe I should ask them for the location. Wait, the + function''s parameters are required to have \"location\", so I can''t call + it without that info. Since the user didn''t provide it, I should prompt them + to specify the location so I can retrieve the weather. Let me check the function + again to confirm. Yep, location is required. Alright, time to ask the user + for the location details.\n"},"finish_reason":"stop"}],"usage":{"prompt_tokens":137,"completion_tokens":157,"total_tokens":294}} + + ' + recorded_at: Wed, 19 Nov 2025 17:20:55 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/ollama/chat/common_format/tools_test.rb b/test/integration/ollama/chat/common_format/tools_test.rb new file mode 100644 index 00000000..f5fa4d46 --- /dev/null +++ b/test/integration/ollama/chat/common_format/tools_test.rb @@ -0,0 +1,369 @@ +# frozen_string_literal: true + +require_relative "../../../test_helper" + +module Integration + module Ollama + module Chat + module CommonFormat + class ToolsTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :ollama, model: "qwen3:latest" + + def get_weather(location:) + { location: location, temperature: "72°F", conditions: "sunny" } + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + + # Common format with 'parameters' key (recommended) + COMMON_FORMAT_PARAMETERS = { + model: "qwen3:latest", + messages: [ + { + role: "user", + content: "What's the weather in San Francisco?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + } + ] + } + def common_format_parameters + prompt( + message: "What's the weather in San Francisco?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Common format with 'input_schema' key (should normalize to parameters) + COMMON_FORMAT_INPUT_SCHEMA = { + model: "qwen3:latest", + messages: [ + { + role: "user", + content: "What's the weather in Boston?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + } + ] + } + def common_format_input_schema + prompt( + message: "What's the weather in Boston?", + tools: [ + { + name: "get_weather", + description: "Get the current weather in a given location", + input_schema: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" + } + }, + required: [ "location" ] + } + } + ] + ) + end + + # Multiple tools in common format + COMMON_FORMAT_MULTIPLE_TOOLS = { + model: "qwen3:latest", + messages: [ + { + role: "user", + content: "What's the weather in NYC and what's 5 plus 3?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + }, + { + type: "function", + function: { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + } + ] + } + def common_format_multiple_tools + prompt( + message: "What's the weather in NYC and what's 5 plus 3?", + tools: [ + { + name: "get_weather", + description: "Get the current weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + }, + { + name: "calculate", + description: "Perform basic arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string", enum: [ "add", "subtract", "multiply", "divide" ] }, + a: { type: "number" }, + b: { type: "number" } + }, + required: [ "operation", "a", "b" ] + } + } + ] + ) + end + + # Tool choice - string format + COMMON_FORMAT_TOOL_CHOICE_AUTO = { + model: "qwen3:latest", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: "auto" + } + def common_format_tool_choice_auto + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "auto" + ) + end + + # Tool choice - force tool use with "required" + COMMON_FORMAT_TOOL_CHOICE_REQUIRED = { + model: "qwen3:latest", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: "required" + } + def common_format_tool_choice_required + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: "required" + ) + end + + # Tool choice - specific tool + COMMON_FORMAT_TOOL_CHOICE_SPECIFIC = { + model: "qwen3:latest", + messages: [ + { + role: "user", + content: "What's the weather?" + } + ], + tools: [ + { + type: "function", + function: { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + } + ], + tool_choice: { + type: "function", + function: { + name: "get_weather" + } + } + } + def common_format_tool_choice_specific + prompt( + message: "What's the weather?", + tools: [ + { + name: "get_weather", + description: "Get weather", + parameters: { + type: "object", + properties: { + location: { type: "string" } + }, + required: [ "location" ] + } + } + ], + tool_choice: { name: "get_weather" } + ) + end + end + + ################################################################################ + # This automatically runs all the tests for the test actions + ################################################################################ + [ + :common_format_parameters, + :common_format_input_schema, + :common_format_multiple_tools, + :common_format_tool_choice_auto, + :common_format_tool_choice_required, + :common_format_tool_choice_specific + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) + end + end + end + end + end +end From 7f4ece3bcf90cff8ad3ab889e63b3863d032fe16 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 09:30:37 -0800 Subject: [PATCH 06/17] Update Tools Previewing --- .../providers/concerns/previewable.rb | 44 ++++++++++++++++--- 1 file changed, 39 insertions(+), 5 deletions(-) diff --git a/lib/active_agent/providers/concerns/previewable.rb b/lib/active_agent/providers/concerns/previewable.rb index dcdf81d7..bef818f8 100644 --- a/lib/active_agent/providers/concerns/previewable.rb +++ b/lib/active_agent/providers/concerns/previewable.rb @@ -86,7 +86,13 @@ def render_single_message(message, index) "### Message #{index} (#{role.capitalize})\n#{content}" end - # Renders available tools with descriptions and parameter schemas. + # Renders tools section for preview. + # + # Handles multiple tool formats: + # - Common format: {name: "...", description: "...", parameters: {...}} + # - Anthropic format: {name: "...", description: "...", input_schema: {...}} + # - Chat API format: {type: "function", function: {name: "...", description: "...", parameters: {...}}} + # - Responses API format: {type: "function", name: "...", description: "...", parameters: {...}} # # @param tools [Array] # @return [String] @@ -96,17 +102,45 @@ def render_tools_section(tools) content = +"## Tools\n\n" tools.each_with_index do |tool, index| - content << "### #{tool[:name] || "Tool #{index + 1}"}\n" - content << "**Description:** #{tool[:description] || 'No description'}\n\n" + # Extract name and description from different formats + tool_name, tool_description, tool_params = extract_tool_details(tool) + + content << "### #{tool_name || "Tool #{index + 1}"}\n" + content << "**Description:** #{tool_description || 'No description'}\n\n" - if tool[:parameters] - content << "**Parameters:**\n```json\n#{JSON.pretty_generate(tool[:parameters])}\n```\n\n" + if tool_params + content << "**Parameters:**\n```json\n#{JSON.pretty_generate(tool_params)}\n```\n\n" end end content.chomp end + # Extracts tool details from different formats. + # + # @param tool [Hash] + # @return [Array] [name, description, parameters] + def extract_tool_details(tool) + tool_hash = tool.is_a?(Hash) ? tool : {} + + # Chat API nested format: {type: "function", function: {...}} + if tool_hash[:type] == "function" && tool_hash[:function] + func = tool_hash[:function] + return [ + func[:name], + func[:description], + func[:parameters] || func[:input_schema] + ] + end + + # Flat formats (common, Anthropic, Responses) + [ + tool_hash[:name], + tool_hash[:description], + tool_hash[:parameters] || tool_hash[:input_schema] + ] + end + # Extracts text content from various message formats. # # Handles string messages, hash messages with :content key, and From 5b55146e4ff318bc1f3f0cce12d1e7e09750810d Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 09:57:12 -0800 Subject: [PATCH 07/17] Refactor Tool Choice Clearing --- lib/active_agent/providers/_base_provider.rb | 2 + .../providers/anthropic_provider.rb | 33 ++++++---- .../concerns/tool_choice_clearing.rb | 62 +++++++++++++++++++ .../providers/open_ai/chat_provider.rb | 40 ++++++------ .../providers/open_ai/responses_provider.rb | 40 ++++++------ .../providers/open_router_provider.rb | 30 ++------- 6 files changed, 134 insertions(+), 73 deletions(-) create mode 100644 lib/active_agent/providers/concerns/tool_choice_clearing.rb diff --git a/lib/active_agent/providers/_base_provider.rb b/lib/active_agent/providers/_base_provider.rb index db254f3a..63931692 100644 --- a/lib/active_agent/providers/_base_provider.rb +++ b/lib/active_agent/providers/_base_provider.rb @@ -4,6 +4,7 @@ require_relative "concerns/exception_handler" require_relative "concerns/instrumentation" require_relative "concerns/previewable" +require_relative "concerns/tool_choice_clearing" # @private GEM_LOADERS = { @@ -45,6 +46,7 @@ class BaseProvider include ExceptionHandler include Instrumentation include Previewable + include ToolChoiceClearing class ProvidersError < StandardError; end diff --git a/lib/active_agent/providers/anthropic_provider.rb b/lib/active_agent/providers/anthropic_provider.rb index f243355d..b6eb03c4 100644 --- a/lib/active_agent/providers/anthropic_provider.rb +++ b/lib/active_agent/providers/anthropic_provider.rb @@ -39,22 +39,31 @@ def prepare_prompt_request super end - # @api private - def prepare_prompt_request_tools - return unless request.tool_choice - return unless request.tool_choice.respond_to?(:type) + # Extracts function names from Anthropic's tool_use content blocks. + # + # @return [Array] + def extract_used_function_names + message_stack.pluck(:content).flatten.select { _1[:type] == "tool_use" }.pluck(:name) + end - functions_used = message_stack.pluck(:content).flatten.select { _1[:type] == "tool_use" }.pluck(:name) + # Returns true if tool_choice forces any tool use (type == :any). + # + # @return [Boolean] + def tool_choice_forces_required? + return false unless request.tool_choice.respond_to?(:type) - # tool_choice is always a gem model object (ToolChoiceAny, ToolChoiceTool, ToolChoiceAuto) - tool_choice_type = request.tool_choice.type - tool_choice_name = request.tool_choice.respond_to?(:name) ? request.tool_choice.name : nil + request.tool_choice.type == :any + end - if (tool_choice_type == :any && functions_used.any?) || - (tool_choice_type == :tool && tool_choice_name && functions_used.include?(tool_choice_name)) + # Returns [true, name] if tool_choice forces a specific tool (type == :tool). + # + # @return [Array] + def tool_choice_forces_specific? + return [ false, nil ] unless request.tool_choice.respond_to?(:type) + return [ false, nil ] unless request.tool_choice.type == :tool - request.tool_choice = nil - end + tool_name = request.tool_choice.respond_to?(:name) ? request.tool_choice.name : nil + [ true, tool_name ] end # @api private diff --git a/lib/active_agent/providers/concerns/tool_choice_clearing.rb b/lib/active_agent/providers/concerns/tool_choice_clearing.rb new file mode 100644 index 00000000..27c369f0 --- /dev/null +++ b/lib/active_agent/providers/concerns/tool_choice_clearing.rb @@ -0,0 +1,62 @@ +# frozen_string_literal: true + +module ActiveAgent + module Providers + # Provides unified logic for clearing tool_choice after tool execution. + # + # When a tool_choice is set to "required" or to a specific tool name, + # it forces the model to use that tool. After the tool is executed, + # we need to clear the tool_choice to prevent infinite loops where + # the model keeps calling the same tool repeatedly. + # + # Each provider implements: + # - `extract_used_function_names`: Returns array of tool names that have been called + # - `tool_choice_forces_required?`: Returns true if tool_choice forces any tool use + # - `tool_choice_forces_specific?`: Returns [true, name] if tool_choice forces specific tool + module ToolChoiceClearing + extend ActiveSupport::Concern + + # @api private + def prepare_prompt_request_tools + return unless request.tool_choice + + functions_used = extract_used_function_names + + # Clear if forcing required and any tool was used + if tool_choice_forces_required? && functions_used.any? + request.tool_choice = nil + return + end + + # Clear if forcing specific tool and that tool was used + forces_specific, tool_name = tool_choice_forces_specific? + if forces_specific && tool_name && functions_used.include?(tool_name) + request.tool_choice = nil + end + end + + private + + # Extracts the list of function names that have been called. + # + # @return [Array] function names + def extract_used_function_names + raise NotImplementedError, "#{self.class} must implement #extract_used_function_names" + end + + # Returns true if tool_choice forces any tool to be used (e.g., "required", "any"). + # + # @return [Boolean] + def tool_choice_forces_required? + raise NotImplementedError, "#{self.class} must implement #tool_choice_forces_required?" + end + + # Returns [true, tool_name] if tool_choice forces a specific tool, [false, nil] otherwise. + # + # @return [Array] + def tool_choice_forces_specific? + raise NotImplementedError, "#{self.class} must implement #tool_choice_forces_specific?" + end + end + end +end diff --git a/lib/active_agent/providers/open_ai/chat_provider.rb b/lib/active_agent/providers/open_ai/chat_provider.rb index 5905ad2e..4c1bd656 100644 --- a/lib/active_agent/providers/open_ai/chat_provider.rb +++ b/lib/active_agent/providers/open_ai/chat_provider.rb @@ -12,6 +12,8 @@ module OpenAI # @see Base # @see https://platform.openai.com/docs/api-reference/chat class ChatProvider < Base + include ToolChoiceClearing + # @return [Class] the options class for this provider def self.options_klass Options @@ -37,30 +39,32 @@ def prepare_prompt_request super end - # @api private - def prepare_prompt_request_tools - return unless request.tool_choice - - # Get list of function calls that have been made - # In Chat API, tool calls are in the assistant message's tool_calls array - functions_used = message_stack + # Extracts function names from Chat API tool_calls in assistant messages. + # + # @return [Array] + def extract_used_function_names + message_stack .select { |msg| msg[:role] == "assistant" && msg[:tool_calls] } .flat_map { |msg| msg[:tool_calls] } .map { |tc| tc.dig(:function, :name) } .compact + end - # Check if tool_choice is a hash (specific tool) or string (auto/required) + # Returns true if tool_choice == "required". + # + # @return [Boolean] + def tool_choice_forces_required? + request.tool_choice == "required" + end + + # Returns [true, name] if tool_choice is a hash with nested function name. + # + # @return [Array] + def tool_choice_forces_specific? if request.tool_choice.is_a?(Hash) - # Specific tool choice - clear if that tool was used - tool_choice_name = request.tool_choice.dig(:function, :name) - if tool_choice_name && functions_used.include?(tool_choice_name) - request.tool_choice = nil - end - elsif request.tool_choice == "required" - # Required tool choice - clear if any tool was used - if functions_used.any? - request.tool_choice = nil - end + [ true, request.tool_choice.dig(:function, :name) ] + else + [ false, nil ] end end diff --git a/lib/active_agent/providers/open_ai/responses_provider.rb b/lib/active_agent/providers/open_ai/responses_provider.rb index ad164a22..2734030e 100644 --- a/lib/active_agent/providers/open_ai/responses_provider.rb +++ b/lib/active_agent/providers/open_ai/responses_provider.rb @@ -13,6 +13,8 @@ module OpenAI # @see Base # @see https://platform.openai.com/docs/api-reference/responses class ResponsesProvider < Base + include ToolChoiceClearing + # @return [Class] def self.options_klass Options @@ -33,29 +35,31 @@ def prepare_prompt_request super end - # @api private - def prepare_prompt_request_tools - return unless request.tool_choice - - # Get list of function calls that have been made - # In Responses API, message_stack items are flat - each item has a type field - functions_used = message_stack + # Extracts function names from Responses API function_call items. + # + # @return [Array] + def extract_used_function_names + message_stack .select { |item| item[:type] == "function_call" } .map { |item| item[:name] } .compact + end - # Check if tool_choice is a gem model object or symbol + # Returns true if tool_choice == :required. + # + # @return [Boolean] + def tool_choice_forces_required? + request.tool_choice == :required + end + + # Returns [true, name] if tool_choice is a ToolChoiceFunction model object. + # + # @return [Array] + def tool_choice_forces_specific? if request.tool_choice.is_a?(::OpenAI::Models::Responses::ToolChoiceFunction) - # Specific tool choice - clear if that tool was used - tool_choice_name = request.tool_choice.name - if tool_choice_name && functions_used.include?(tool_choice_name) - request.tool_choice = nil - end - elsif request.tool_choice == :required - # Required tool choice - clear if any tool was used - if functions_used.any? - request.tool_choice = nil - end + [ true, request.tool_choice.name ] + else + [ false, nil ] end end diff --git a/lib/active_agent/providers/open_router_provider.rb b/lib/active_agent/providers/open_router_provider.rb index 1204d6bc..88a01567 100644 --- a/lib/active_agent/providers/open_router_provider.rb +++ b/lib/active_agent/providers/open_router_provider.rb @@ -40,31 +40,11 @@ def prepare_prompt_request super end - # @api private - def prepare_prompt_request_tools - return unless request.tool_choice - - # Get list of function calls that have been made - # In Chat API, tool calls are in the assistant message's tool_calls array - functions_used = message_stack - .select { |msg| msg[:role] == "assistant" && msg[:tool_calls] } - .flat_map { |msg| msg[:tool_calls] } - .map { |tc| tc.dig(:function, :name) } - .compact - - # Check if tool_choice is a hash (specific tool) or string (auto/any) - if request.tool_choice.is_a?(Hash) - # Specific tool choice - clear if that tool was used - tool_choice_name = request.tool_choice.dig(:function, :name) - if tool_choice_name && functions_used.include?(tool_choice_name) - request.tool_choice = nil - end - elsif request.tool_choice == "any" - # OpenRouter uses "any" for required - clear if any tool was used - if functions_used.any? - request.tool_choice = nil - end - end + # Returns true if tool_choice == "any" (OpenRouter's equivalent of "required"). + # + # @return [Boolean] + def tool_choice_forces_required? + request.tool_choice == "any" end # Merges streaming delta into the message with role cleanup. From 6f1486cabc564c2ca62aec2455ddf9846dec778e Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 10:21:05 -0800 Subject: [PATCH 08/17] Update Tools Docs for Common Format --- docs/actions/tools.md | 77 ++- test/docs/actions/tools_examples_test.rb | 152 ++++-- test/docs/actions_examples_test.rb | 1 - .../actions/tools/cross_provider_usage.yml | 448 ++++++++++++++++++ 4 files changed, 631 insertions(+), 47 deletions(-) create mode 100644 test/fixtures/vcr_cassettes/docs/actions/tools/cross_provider_usage.yml diff --git a/docs/actions/tools.md b/docs/actions/tools.md index e649d666..0a040282 100644 --- a/docs/actions/tools.md +++ b/docs/actions/tools.md @@ -25,13 +25,13 @@ The LLM calls `get_weather` automatically when it needs weather data, and uses t | **Ollama** | 🟩 | ❌ | ❌ | Model-dependent capabilities | | **Mock** | 🟦 | ❌ | ❌ | Accepted but not enforced | -## Functions (Universal Support) +## Functions -Functions are the core tool capability supported by all providers. Define methods in your agent that the LLM can call with appropriate parameters. +Functions are callable methods in your agent that LLMs can trigger with appropriate parameters. All providers support the **common format** described above. ### Basic Function Registration -Register functions by passing tool definitions to the `tools` parameter: +Using the common format, register functions by passing tool definitions to the `tools` parameter: ::: code-group <<< @/../test/docs/actions/tools_examples_test.rb#anthropic_basic_function {ruby:line-numbers} [Anthropic] @@ -42,24 +42,83 @@ Register functions by passing tool definitions to the `tools` parameter: When the LLM decides to call a tool, ActiveAgent routes the call to your agent method and returns the result automatically. +## Common Tools Format (Recommended) + +ActiveAgent supports a **universal common format** for tool definitions that works seamlessly across all providers. This format eliminates the need to learn provider-specific syntax and makes your code portable. + +### Format Specification + +```ruby +{ + name: "function_name", # Required: function name to call + description: "What it does", # Required: clear description for LLM + parameters: { # Required: JSON Schema for parameters + type: "object", + properties: { + param_name: { + type: "string", + description: "Parameter description" + } + }, + required: ["param_name"] + } +} +``` + +### Cross-Provider Example + +The same tool definition works everywhere: + +<<< @/../test/docs/actions/tools_examples_test.rb#cross_provider_module {ruby:line-numbers} + +::: code-group +<<< @/../test/docs/actions/tools_examples_test.rb#cross_provider_anthropic {ruby:line-numbers} [Anthropic] +<<< @/../test/docs/actions/tools_examples_test.rb#cross_provider_ollama{ruby:line-numbers} [Ollama] +<<< @/../test/docs/actions/tools_examples_test.rb#cross_provider_openai{ruby:line-numbers} [OpenAI] +<<< @/../test/docs/actions/tools_examples_test.rb#cross_provider_openrouter {ruby:line-numbers} [OpenRouter] +::: + +### Alternative: `input_schema` Key + +You can also use `input_schema` instead of `parameters` - both work identically: + +```ruby +{ + name: "get_weather", + description: "Get current weather", + input_schema: { # Alternative to 'parameters' + type: "object", + properties: { ... } + } +} +``` + +ActiveAgent automatically converts between common format and each provider's native format behind the scenes. + ### Tool Choice Control -Control which tools the LLM can use: +Control when and which tools the LLM uses with the `tool_choice` parameter: ```ruby -# Let the model decide (default) +# Auto (default) - Let the model decide whether to use tools prompt(message: "...", tools: tools, tool_choice: "auto") -# Force the model to use a tool +# Required - Force the model to use at least one tool prompt(message: "...", tools: tools, tool_choice: "required") -# Prevent tool usage +# None - Prevent tool usage entirely prompt(message: "...", tools: tools, tool_choice: "none") -# Force a specific tool (provider-dependent) -prompt(message: "...", tools: tools, tool_choice: { type: "function", name: "get_weather" }) +# Specific tool - Force a particular tool (common format) +prompt(message: "...", tools: tools, tool_choice: { name: "get_weather" }) ``` +ActiveAgent automatically maps these common values to provider-specific formats: +- **OpenAI**: `"auto"`, `"required"`, `"none"`, or `{type: "function", function: {name: "..."}}` +- **Anthropic**: `{type: :auto}`, `{type: :any}`, `{type: :tool, name: "..."}` +- **OpenRouter**: `"auto"`, `"any"` (equivalent to "required") +- **Ollama**: Model-dependent tool choice support + ## Server-Side Tools (Provider-Specific) Some providers offer built-in tools that run on their servers, providing capabilities like web search and code execution without custom implementation. diff --git a/test/docs/actions/tools_examples_test.rb b/test/docs/actions/tools_examples_test.rb index 9f68149f..396cf27e 100644 --- a/test/docs/actions/tools_examples_test.rb +++ b/test/docs/actions/tools_examples_test.rb @@ -12,7 +12,6 @@ def weather_update prompt( input: "What's the weather in Boston?", tools: [ { - type: "function", name: "get_weather", description: "Get current weather for a location", parameters: { @@ -55,7 +54,6 @@ def weather_update prompt( input: "What's the weather in Boston?", tools: [ { - type: "function", name: "get_current_weather", description: "Get the current weather in a given location", parameters: { @@ -105,7 +103,7 @@ def weather_update tools: [ { name: "get_weather", description: "Get the current weather in a given location", - input_schema: { + parameters: { type: "object", properties: { location: { @@ -145,24 +143,21 @@ def weather_update prompt( message: "What's the weather in Boston?", tools: [ { - type: "function", - function: { - name: "get_current_weather", - description: "Get the current weather in a given location", - parameters: { - type: "object", - properties: { - location: { - type: "string", - description: "The city and state, e.g. San Francisco, CA" - }, - unit: { - type: "string", - enum: [ "celsius", "fahrenheit" ] - } + name: "get_current_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" }, - required: [ "location" ] - } + unit: { + type: "string", + enum: [ "celsius", "fahrenheit" ] + } + }, + required: [ "location" ] } } ] ) @@ -194,24 +189,21 @@ def weather_update prompt( message: "What's the weather in Boston?", tools: [ { - type: "function", - function: { - name: "get_current_weather", - description: "Get the current weather in a given location", - parameters: { - type: "object", - properties: { - location: { - type: "string", - description: "The city and state, e.g. San Francisco, CA" - }, - unit: { - type: "string", - enum: [ "celsius", "fahrenheit" ] - } + name: "get_current_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA" }, - required: [ "location" ] - } + unit: { + type: "string", + enum: [ "celsius", "fahrenheit" ] + } + }, + required: [ "location" ] } } ] ) @@ -233,6 +225,92 @@ def get_current_weather(location:, unit: "fahrenheit") end end end + + class CrossProviderExample < ActiveSupport::TestCase + test "cross provider usage" do + VCR.use_cassette("docs/actions/tools/cross_provider_usage") do + # region cross_provider_module + # Define once, use with any provider + module WeatherTool + extend ActiveSupport::Concern + + WEATHER_TOOL = { + name: "get_weather", + description: "Get current weather for a location", + parameters: { + type: "object", + properties: { + location: { type: "string", description: "City and state" }, + unit: { type: "string", enum: [ "celsius", "fahrenheit" ] } + }, + required: [ "location" ] + } + } + + def get_current_weather(location:, unit: "fahrenheit") + { location: location, unit: unit, temperature: "22" } + end + end + # endregion cross_provider_module + + # region cross_provider_openai + class OpenAIAgent < ApplicationAgent + include WeatherTool + generate_with :openai, model: "gpt-4o" + + def check_weather + prompt(input: "What's the weather?", tools: [ WEATHER_TOOL ]) + end + end + # endregion cross_provider_openai + + # region cross_provider_anthropic + class AnthropicAgent < ApplicationAgent + include WeatherTool + generate_with :anthropic, model: "claude-sonnet-4-20250514" + + def check_weather + prompt(message: "What's the weather?", tools: [ WEATHER_TOOL ]) + end + end + # endregion cross_provider_anthropic + + # region cross_provider_ollama + class OllamaAgent < ApplicationAgent + include WeatherTool + generate_with :ollama, model: "qwen3:latest" + + def check_weather + prompt(message: "What's the weather?", tools: [ WEATHER_TOOL ]) + end + end + # endregion cross_provider_ollama + + # region cross_provider_openrouter + class OpenRouterAgent < ApplicationAgent + include WeatherTool + generate_with :openrouter, model: "google/gemini-2.0-flash-001" + + def check_weather + prompt(message: "What's the weather?", tools: [ WEATHER_TOOL ]) + end + end + # endregion cross_provider_openrouter + + response = OpenAIAgent.check_weather.generate_now + assert response.message.content.present? + + response = AnthropicAgent.check_weather.generate_now + assert response.message.content.present? + + response = OllamaAgent.check_weather.generate_now + assert response.message.content.present? + + response = OpenRouterAgent.check_weather.generate_now + assert response.message.content.present? + end + end + end end end end diff --git a/test/docs/actions_examples_test.rb b/test/docs/actions_examples_test.rb index 45de6a82..f9d0b590 100644 --- a/test/docs/actions_examples_test.rb +++ b/test/docs/actions_examples_test.rb @@ -55,7 +55,6 @@ def weather_update prompt( input: "What's the weather in Boston?", tools: [ { - type: "function", name: "get_current_weather", description: "Get the current weather in a given location", parameters: { diff --git a/test/fixtures/vcr_cassettes/docs/actions/tools/cross_provider_usage.yml b/test/fixtures/vcr_cassettes/docs/actions/tools/cross_provider_usage.yml new file mode 100644 index 00000000..9f9e4250 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/tools/cross_provider_usage.yml @@ -0,0 +1,448 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4o","input":"What''s the weather?","tools":[{"type":"function","name":"get_weather","description":"Get + current weather for a location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"City + and state"},"unit":{"type":"string","enum":["celsius","fahrenheit"]}},"required":["location"]}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '337' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 18:18:33 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999725' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_8513283bef2e4a1faf08f9d91b8debdb + Openai-Processing-Ms: + - '1782' + X-Envoy-Upstream-Service-Time: + - '1786' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=k_pHGl_NaO7cz7cXFUH461e6R3jh2QlExmfoMdyRMmY-1763576313-1.0.1.1-xomS.7WXMfUlaxu8GUfrYZdZg0W8_vsG_PHcWjzvSAKpDctb1ztMqMDCNpoCmhmLYmbFcAYL8Q5S2zkKRSCMF6HYj0I4G_Wyf6uQAHBKrmw; + path=/; expires=Wed, 19-Nov-25 18:48:33 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=AFN0qelmGZnujGNMi3.dkE6SkuXAnw1oMNw8lM_.AAY-1763576313407-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a11b5e6d9ba67fb-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_0bc6fad39e50a44b00691e09f79f44819b8a01809fe1c1e1e1", + "object": "response", + "created_at": 1763576311, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4o-2024-08-06", + "output": [ + { + "id": "msg_0bc6fad39e50a44b00691e09f8d6bc819bb5b189268456efea", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Could you please specify the location for which you'd like to know the weather?" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Get current weather for a location", + "name": "get_weather", + "parameters": { + "type": "object", + "properties": { + "location": { + "type": "string", + "description": "City and state" + }, + "unit": { + "type": "string", + "enum": [ + "celsius", + "fahrenheit" + ] + } + }, + "required": [ + "location", + "unit" + ], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 58, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 17, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 75 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 18:18:33 GMT +- request: + method: post + uri: https://api.anthropic.com/v1/messages + body: + encoding: UTF-8 + string: '{"model":"claude-sonnet-4-20250514","messages":[{"content":"What''s + the weather?","role":"user"}],"tools":[{"input_schema":{"type":"object","properties":{"location":{"type":"string","description":"City + and state"},"unit":{"type":"string","enum":["celsius","fahrenheit"]}},"required":["location"]},"name":"get_weather","description":"Get + current weather for a location"}],"max_tokens":4096}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '388' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 18:18:36 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '2000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '2000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T18:18:35Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '400000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '400000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T18:18:36Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T18:18:33Z' + Retry-After: + - '25' + Anthropic-Ratelimit-Tokens-Limit: + - '2400000' + Anthropic-Ratelimit-Tokens-Remaining: + - '2400000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T18:18:35Z' + Request-Id: + - req_011CVHi3YRg6Ci88kiLM46kE + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '2614' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a11b5f74c88bffc-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-sonnet-4-20250514","id":"msg_01SX7S6wKVyFVdm4hPJ4Mbjc","type":"message","role":"assistant","content":[{"type":"text","text":"I''d + be happy to help you get the weather information! However, I need to know + which location you''d like the weather for. Could you please tell me the city + and state (or city and country) you''re interested in?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":407,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":50,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 18:18:36 GMT +- request: + method: post + uri: http://127.0.0.1:11434/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"qwen3:latest","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + current weather for a location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"City + and state"},"unit":{"type":"string","enum":["celsius","fahrenheit"]}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - 127.0.0.1:11434 + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ollama + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '387' + response: + status: + code: 200 + message: OK + headers: + Content-Type: + - application/json + Date: + - Wed, 19 Nov 2025 18:18:45 GMT + Content-Length: + - '937' + body: + encoding: UTF-8 + string: '{"id":"chatcmpl-178","object":"chat.completion","created":1763576325,"model":"qwen3:latest","system_fingerprint":"fp_ollama","choices":[{"index":0,"message":{"role":"assistant","content":"I + need more details to check the weather. Could you please tell me the location + (city and state) and whether you want the temperature in Celsius or Fahrenheit?","reasoning":"Okay, + the user asked, \"What''s the weather?\" I need to figure out how to respond. + Let me check the tools provided. There''s a get_weather function, but it requires + parameters like location and unit. The user didn''t specify a location or + unit. I should ask them for more details. Let me confirm if they want the + weather for a specific city and whether they prefer Celsius or Fahrenheit. + That way, I can call the function properly once I have all the necessary information.\n"},"finish_reason":"stop"}],"usage":{"prompt_tokens":161,"completion_tokens":136,"total_tokens":297}} + + ' + recorded_at: Wed, 19 Nov 2025 18:18:45 GMT +- request: + method: post + uri: https://openrouter.ai/api/v1/chat/completions + body: + encoding: UTF-8 + string: '{"model":"google/gemini-2.0-flash-001","messages":[{"role":"user","content":"What''s + the weather?"}],"tools":[{"type":"function","function":{"name":"get_weather","description":"Get + current weather for a location","parameters":{"type":"object","properties":{"location":{"type":"string","description":"City + and state"},"unit":{"type":"string","enum":["celsius","fahrenheit"]}},"required":["location"]}}}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - openrouter.ai + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Authorization: + - Bearer ACCESS_TOKEN + Http-Referer: + - https://example.com + X-Title: + - Dummy + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '402' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 18:18:46 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Access-Control-Allow-Origin: + - "*" + Vary: + - Accept-Encoding + Permissions-Policy: + - payment=(self "https://checkout.stripe.com" "https://connect-js.stripe.com" + "https://js.stripe.com" "https://*.js.stripe.com" "https://hooks.stripe.com") + Referrer-Policy: + - no-referrer, strict-origin-when-cross-origin + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a11b644cedeee17-SJC + body: + encoding: ASCII-8BIT + string: "\n \n{\"id\":\"gen-1763576326-GG7DVgNI8Zxze0XPOD7X\",\"provider\":\"Google\",\"model\":\"google/gemini-2.0-flash-001\",\"object\":\"chat.completion\",\"created\":1763576326,\"choices\":[{\"logprobs\":null,\"finish_reason\":\"stop\",\"native_finish_reason\":\"STOP\",\"index\":0,\"message\":{\"role\":\"assistant\",\"content\":\"To + provide you with the weather, I need to know your location. Could you please + tell me which city and state you're in?\\n\",\"refusal\":null,\"reasoning\":null}}],\"usage\":{\"prompt_tokens\":28,\"completion_tokens\":29,\"total_tokens\":57,\"prompt_tokens_details\":{\"cached_tokens\":0},\"completion_tokens_details\":{\"reasoning_tokens\":0,\"image_tokens\":0}}}" + recorded_at: Wed, 19 Nov 2025 18:18:46 GMT +recorded_with: VCR 6.3.1 From b74d7f77896916d7e0c8a80f613ad57b3a9b61bb Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 10:39:36 -0800 Subject: [PATCH 09/17] Update CHANGELOG --- CHANGELOG.md | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1d112737..71d7740a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -111,6 +111,29 @@ Template paths: ### Added +**Universal Tools Format** +```ruby +# Single format works across all providers (Anthropic, OpenAI, OpenRouter, Ollama, Mock) +tools: [{ + name: "get_weather", + description: "Get current weather", + parameters: { + type: "object", + properties: { + location: { type: "string", description: "City and state" } + }, + required: ["location"] + } +}] + +# Tool choice normalization +tool_choice: "auto" # Let model decide +tool_choice: "required" # Force tool use +tool_choice: { name: "get_weather" } # Force specific tool +``` + +Automatic conversion to provider-specific formats. Old formats still work (backward compatible). + **Mock Provider for Testing** ```ruby class MyAgent < ActiveAgent::Base @@ -196,6 +219,7 @@ response.usage.service_tier # Anthropic - Retry logic moved to provider SDKs (automatic exponential backoff) - Migrated to official SDKs: `openai` gem and `anthropic` gem - Type-safe options with per-provider definitions +- Shared `ToolChoiceClearing` concern eliminates duplication across providers **Configuration** - Options configurable at class level, instance level, or per-call From 9887f25607e0960f7bc38fa4237f57d2d48d4cb5 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 12:36:18 -0800 Subject: [PATCH 10/17] Add Anthropic [Beta] MCP Support --- .../providers/anthropic/options.rb | 10 +- .../providers/anthropic/request.rb | 31 ++- .../providers/anthropic/transforms.rb | 43 ++- .../providers/anthropic_provider.rb | 9 +- .../test_agent_common_format_mixed_auth.yml | 141 ++++++++++ ...t_agent_common_format_multiple_servers.yml | 142 ++++++++++ ...test_agent_common_format_single_server.yml | 110 ++++++++ ..._common_format_single_server_with_auth.yml | 140 ++++++++++ .../test_agent_common_format_sse_server.yml | 109 ++++++++ .../anthropic/common_format/mcp_test.rb | 256 ++++++++++++++++++ test/providers/anthropic/transforms_test.rb | 120 +++++++- test/test_helper.rb | 11 +- 12 files changed, 1102 insertions(+), 20 deletions(-) create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml create mode 100644 test/integration/anthropic/common_format/mcp_test.rb diff --git a/lib/active_agent/providers/anthropic/options.rb b/lib/active_agent/providers/anthropic/options.rb index 898d3027..2160e412 100644 --- a/lib/active_agent/providers/anthropic/options.rb +++ b/lib/active_agent/providers/anthropic/options.rb @@ -28,15 +28,13 @@ def initialize(kwargs = {}) end def serialize - super.except(:anthropic_beta).tap do |hash| - hash[:extra_headers] = extra_headers unless extra_headers.blank? - end + super.except(:anthropic_beta) end + # Anthropic gem handles beta headers differently via client.beta + # rather than via extra_headers in request_options def extra_headers - deep_compact( - "anthropic-beta" => anthropic_beta.presence, - ) + {} end private diff --git a/lib/active_agent/providers/anthropic/request.rb b/lib/active_agent/providers/anthropic/request.rb index 08419c23..ccc33b91 100644 --- a/lib/active_agent/providers/anthropic/request.rb +++ b/lib/active_agent/providers/anthropic/request.rb @@ -67,11 +67,13 @@ class Request < SimpleDelegator # @option params [Array] :messages required # @option params [Integer] :max_tokens (4096) # @option params [Hash] :response_format custom field for JSON mode simulation + # @option params [String] :anthropic_beta beta version for features like MCP # @raise [ArgumentError] when gem model validation fails def initialize(**params) # Step 1: Extract custom fields that gem doesn't support @response_format = params.delete(:response_format) @stream = params.delete(:stream) + anthropic_beta = params.delete(:anthropic_beta) # Step 2: Map common format 'instructions' to Anthropic's 'system' if params.key?(:instructions) @@ -84,10 +86,24 @@ def initialize(**params) # Step 4: Transform params for gem compatibility transformed = Transforms.normalize_params(params) - # Step 5: Create gem model - this validates all parameters! - gem_model = ::Anthropic::Models::MessageCreateParams.new(**transformed) + # Step 5: Determine if we need beta params (for MCP or other beta features) + use_beta = anthropic_beta.present? || transformed[:mcp_servers]&.any? - # Step 6: Delegate all method calls to gem model + # Step 6: Add betas parameter if using beta API + if use_beta + # Default to MCP beta version if not specified + beta_version = anthropic_beta || "mcp-client-2025-04-04" + transformed[:betas] = [ beta_version ] + end + + # Step 7: Create gem model - use Beta version if needed + gem_model = if use_beta + ::Anthropic::Models::Beta::MessageCreateParams.new(**transformed) + else + ::Anthropic::Models::MessageCreateParams.new(**transformed) + end + + # Step 8: Delegate all method calls to gem model super(gem_model) rescue ArgumentError => e # Re-raise with more context @@ -134,6 +150,15 @@ def instructions=(value) self.system = value end + # Accessor for MCP servers. + # + # Safely returns MCP servers array, defaulting to empty array if not set. + # + # @return [Array] + def mcp_servers + __getobj__.instance_variable_get(:@data)[:mcp_servers] || [] + end + # Removes the last message from the messages array. # # Used for JSON format simulation to remove the lead-in assistant message. diff --git a/lib/active_agent/providers/anthropic/transforms.rb b/lib/active_agent/providers/anthropic/transforms.rb index e1c6a736..8214f845 100644 --- a/lib/active_agent/providers/anthropic/transforms.rb +++ b/lib/active_agent/providers/anthropic/transforms.rb @@ -30,6 +30,7 @@ def normalize_params(params) params[:system] = normalize_system(params[:system]) if params[:system] params[:tools] = normalize_tools(params[:tools]) if params[:tools] params[:tool_choice] = normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] + params[:mcp_servers] = normalize_mcp_servers(params[:mcp_servers]) if params[:mcp_servers] params end @@ -58,6 +59,44 @@ def normalize_tools(tools) end end + # Normalizes MCP servers from common format to Anthropic format. + # + # Common format: + # {name: "stripe", url: "https://...", authorization: "token"} + # Anthropic format: + # {type: "url", name: "stripe", url: "https://...", authorization_token: "token"} + # + # @param mcp_servers [Array] + # @return [Array] + def normalize_mcp_servers(mcp_servers) + return mcp_servers unless mcp_servers.is_a?(Array) + + mcp_servers.map do |server| + server_hash = server.is_a?(Hash) ? server.deep_symbolize_keys : server + + # If already in Anthropic format (has type: "url" and authorization_token), return as-is + if server_hash[:type] == "url" && !server_hash[:authorization] + next server_hash + end + + # Convert common format to Anthropic format + result = { + type: "url", + name: server_hash[:name], + url: server_hash[:url] + } + + # Map 'authorization' to 'authorization_token' + if server_hash[:authorization] + result[:authorization_token] = server_hash[:authorization] + elsif server_hash[:authorization_token] + result[:authorization_token] = server_hash[:authorization_token] + end + + result.compact + end + end + # Normalizes tool_choice from common format to Anthropic gem model objects. # # The Anthropic gem expects tool_choice to be a model object (ToolChoiceAuto, @@ -398,9 +437,9 @@ def cleanup_serialized_request(hash, defaults, gem_object = nil) # Apply content compression for API efficiency compress_content(hash) - # Remove provider-internal fields that should not be in API request - hash.delete(:mcp_servers) # Provider-level config, not API param + # Remove provider-internal fields and empty arrays hash.delete(:stop_sequences) if hash[:stop_sequences] == [] + hash.delete(:mcp_servers) if hash[:mcp_servers] == [] hash.delete(:tool_choice) if hash[:tool_choice].nil? # Don't send null tool_choice # Remove default values (except max_tokens which is required by API) diff --git a/lib/active_agent/providers/anthropic_provider.rb b/lib/active_agent/providers/anthropic_provider.rb index b6eb03c4..1bdb0009 100644 --- a/lib/active_agent/providers/anthropic_provider.rb +++ b/lib/active_agent/providers/anthropic_provider.rb @@ -77,9 +77,14 @@ def prepare_prompt_request_response_format end # @see BaseProvider#api_prompt_executer - # @return [Anthropic::Messages] + # @return [Anthropic::Messages, Anthropic::Resources::Beta::Messages] def api_prompt_executer - client.messages + # Use beta API when anthropic_beta option is set or when using MCP servers + if options.anthropic_beta.present? || request.mcp_servers&.any? + client.beta.messages + else + client.messages + end end # @see BaseProvider#api_response_normalize diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml new file mode 100644 index 00000000..6738bbd7 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml @@ -0,0 +1,141 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What + tools do you have available?","role":"user"}],"mcp_servers":[{"name":"cloudflare-demo","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"},{"name":"github-copilot","type":"url","url":"https://api.githubcopilot.com/mcp/","authorization_token":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '425' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:30:50 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '3996000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T20:30:43Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '799000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T20:30:50Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T20:30:42Z' + Retry-After: + - '18' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4795000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T20:30:43Z' + Request-Id: + - req_011CVHt7uDMUSpVrehmUC1kz + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '9375' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a1277811db7aab7-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01R71HsqqyVcvwkVaHYFyu9o","type":"message","role":"assistant","content":[{"type":"text","text":"I + have a comprehensive set of tools available for working with GitHub and Cloudflare. + Here''s what I can do:\n\n## GitHub Tools\n\n### Repository Management\n- + **Create repositories** - Create new repos in your account or an organization\n- + **Fork repositories** - Fork existing projects\n- **List branches** - View + available branches in a repo\n- **List tags** - View git tags in a repository\n- + **List releases** - View releases in a repository\n\n### File Operations\n- + **Get file contents** - Retrieve files or directory contents\n- **Create or + update files** - Create new files or update existing ones in a repo\n- **Delete + files** - Remove files from a repository\n- **Push files** - Push multiple + files in a single commit\n\n### Branch Management\n- **Create branches** - + Create new branches from a source branch\n\n### Commits\n- **Get commit details** + - View specific commit information with diffs\n- **List commits** - View commit + history for a branch\n\n### Issues\n- **Read issues** - Get issue details, + comments, labels, and sub-issues\n- **Create/update issues** - Create new + issues or update existing ones\n- **Search issues** - Search across issues + in repositories\n- **Add issue comments** - Comment on issues\n- **Get labels** + - Retrieve specific label information\n\n### Pull Requests\n- **List pull + requests** - View PRs in a repository\n- **Read PR details** - Get PR information, + diffs, files, status, comments, and reviews\n- **Create pull requests** - + Create new PRs\n- **Update pull requests** - Modify PR title, description, + state, reviewers, etc.\n- **Merge pull requests** - Merge PRs with different + merge strategies\n- **Update PR branch** - Sync PR branch with base branch\n- + **Request Copilot review** - Get automated code review on a PR\n- **Create/submit/delete + reviews** - Manage PR reviews\n- **Add review comments** - Add comments to + pending reviews\n\n### Search\n- **Search code** - Find code across all GitHub + repositories\n- **Search repositories** - Find repos by name, description, + topics, etc.\n- **Search users** - Find GitHub users\n- **Search pull requests** + - Search for PRs\n\n### User & Team\n- **Get authenticated user info** - Get + details about yourself\n- **Get teams** - View teams you''re a member of\n- + **Get team members** - View members of a specific team\n\n### Releases & Tags\n- + **Get latest release** - Retrieve the latest release\n- **Get release by tag** + - Get a specific release by tag name\n- **Get tag details** - Get information + about a specific git tag\n\n### Issue Types\n- **List issue types** - Get + supported issue types for an organization\n\n### Sub-Issues\n- **Add/remove/reprioritize + sub-issues** - Manage parent-child issue relationships\n\n## Cloudflare Tools\n- + **Get MCP Demo Day info** - Information about Cloudflare''s MCP Demo Day\n\nIs + there a specific task you''d like me to help you with?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9642,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":683,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:30:50 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml new file mode 100644 index 00000000..112ceb1d --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml @@ -0,0 +1,142 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What + tools do you have available?","role":"user"}],"mcp_servers":[{"name":"cloudflare-demo","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"},{"name":"github-copilot","type":"url","url":"https://api.githubcopilot.com/mcp/","authorization_token":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '425' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:31:02 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '3996000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T20:30:55Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '799000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T20:31:02Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T20:30:54Z' + Retry-After: + - '5' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4795000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T20:30:55Z' + Request-Id: + - req_011CVHt8papiqK5VdNm4gH1C + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '9014' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a1277cf2b87238d-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01FWVyG5U1wAWwTpCxgRAC3t","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to a comprehensive set of tools for working with GitHub and Cloudflare. + Here''s an overview:\n\n## GitHub Tools\n\n### Repository Management\n- **Create + repository** - Create a new GitHub repository in your account or organization\n- + **Fork repository** - Fork a GitHub repository to your account or organization\n- + **List branches** - List branches in a repository\n- **Create branch** - Create + a new branch\n\n### File Operations\n- **Get file contents** - Retrieve contents + of files or directories\n- **Create or update file** - Create or update a + single file in a repository\n- **Delete file** - Delete a file from a repository\n- + **Push files** - Push multiple files in a single commit\n\n### Issue Management\n- + **Create/update issue** - Create new issues or update existing ones\n- **Read + issue** - Get issue details, comments, sub-issues, and labels\n- **List issues** + - List issues in a repository with filtering and sorting\n- **Search issues** + - Search for issues across GitHub\n- **Add issue comment** - Add comments + to issues (including pull requests)\n- **Sub-issue management** - Add, remove, + or reprioritize sub-issues\n\n### Pull Request Management\n- **Create pull + request** - Create new pull requests\n- **Read pull request** - Get PR details, + diffs, status, files, comments, and reviews\n- **List pull requests** - List + pull requests in a repository\n- **Search pull requests** - Search for pull + requests across GitHub\n- **Update pull request** - Update PR title, description, + state, reviewers, etc.\n- **Update PR branch** - Update a PR branch with latest + changes from base\n- **Merge pull request** - Merge a pull request with various + merge methods\n\n### Code Review\n- **Pull request review write** - Create, + submit, or delete PR reviews\n- **Request Copilot review** - Request automated + code review from GitHub Copilot\n- **Add review comment** - Add comments to + pending PR reviews\n\n### Commits & Tags\n- **Get commit** - Get details of + a specific commit with diffs\n- **List commits** - List commits on a branch + with filtering\n- **Get/List tags** - Get details or list git tags in a repository\n- + **Get/List releases** - Get details or list releases in a repository\n\n### + Other GitHub Tools\n- **Get authenticated user** - Get details of your GitHub + profile\n- **Search code** - Search code across all GitHub repositories\n- + **Search repositories** - Find GitHub repositories by various criteria\n- + **Search users** - Find GitHub users\n- **Get team members** - Get members + of a specific team in an organization\n- **Get teams** - Get teams the user + is a member of\n- **Get label** - Get information about a specific label\n\n## + Cloudflare Tools\n\n- **MCP Demo Day info** - Get information about Cloudflare''s + MCP Demo Day\n\nAll these tools allow me to help you manage repositories, + collaborate on code, handle issues and pull requests, perform code reviews, + and work with Git-related operations. What would you like to do?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9642,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":681,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:31:02 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml new file mode 100644 index 00000000..1d6d558c --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml @@ -0,0 +1,110 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What + tools do you have available?","role":"user"}],"mcp_servers":[{"name":"cloudflare-demo","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '225' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:30:53 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T20:30:52Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T20:30:53Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T20:30:51Z' + Retry-After: + - '8' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T20:30:52Z' + Request-Id: + - req_011CVHt8cCh979WafFuEo6qH + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '2719' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a1277bd1d52eb29-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01Ti8oxkwxk7txgLa9bruWyR","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to one tool:\n\n1. **cloudflare-demo_mcp_demo_day_info** - Get + information about Cloudflare''s MCP Demo Day. Use this tool if the user asks + about Cloudflare''s MCP demo day.\n\nThis tool doesn''t require any parameters + and can be used to retrieve details about Cloudflare''s MCP (Model Context + Protocol) Demo Day event.\n\nIs there anything specific you''d like to know + about Cloudflare''s MCP Demo Day?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":114,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:30:53 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml new file mode 100644 index 00000000..a5931830 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml @@ -0,0 +1,140 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What + tools do you have available?","role":"user"}],"mcp_servers":[{"name":"github-copilot","type":"url","url":"https://api.githubcopilot.com/mcp/","authorization_token":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '337' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:31:12 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '3996000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T20:31:06Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '799000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T20:31:12Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T20:31:02Z' + Retry-After: + - '55' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4795000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T20:31:06Z' + Request-Id: + - req_011CVHt9VnkciWVkKLsE3Mro + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '10287' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a1278087e2a50a1-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_011My4uTh5XocRzomwbE1Pz2","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to a comprehensive set of GitHub tools that allow me to interact + with repositories, pull requests, issues, and more. Here''s what I can do:\n\n## + Repository Management\n- **Create repositories** - Set up new public or private + repos\n- **Fork repositories** - Fork existing projects to your account/organization\n- + **Search repositories** - Find repos by name, description, topics, etc.\n- + **Get repository contents** - View files and directory structures\n\n## Branches + & Commits\n- **Create branches** - Make new branches from existing ones\n- + **List branches** - View all branches in a repository\n- **List commits** + - See commit history with filtering options\n- **Get commit details** - View + specific commit information with diffs\n\n## Issues Management\n- **Create/update + issues** - Create new issues or modify existing ones\n- **Read issue details** + - Get issue information, comments, sub-issues, labels\n- **List issues** - + View issues with filtering and sorting\n- **Search issues** - Find issues + across repositories\n- **Manage sub-issues** - Add, remove, or reprioritize + sub-issues\n- **Add issue comments** - Comment on issues\n\n## Pull Requests\n- + **Create pull requests** - Open new PRs with custom titles and descriptions\n- + **Read PR details** - Get PR info, diffs, status, files changed, comments, + reviews\n- **List/search PRs** - Find pull requests with various filters\n- + **Update PRs** - Modify PR title, description, state, reviewers\n- **Update + PR branch** - Sync PR with latest base branch changes\n- **Merge PRs** - Merge + with different strategies (merge, squash, rebase)\n- **PR Reviews** - Create, + submit, or delete reviews; add review comments\n\n## Files\n- **Create/update + files** - Add or modify files in repositories\n- **Delete files** - Remove + files from repositories\n- **Push multiple files** - Commit multiple files + in one go\n\n## Tags & Releases\n- **List tags** - View git tags in a repository\n- + **Get tag details** - Get specific tag information\n- **List releases** - + View releases\n- **Get release details** - Get specific release or latest + release information\n\n## Users & Teams\n- **Get authenticated user info** + - View your own GitHub profile details\n- **Search users** - Find GitHub users + by name or profile info\n- **Get team members** - View members of specific + teams\n- **Get user teams** - See teams a user belongs to\n\n## Code Search\n- + **Search code** - Fast code search across repositories using GitHub''s search + engine\n\n## Additional\n- **Request Copilot code review** - Get automated + feedback on pull requests\n- **Assign Copilot to issues** - Have me work on + resolving issues\n\nIs there anything specific you''d like me to help you + with?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9567,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":631,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:31:12 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml new file mode 100644 index 00000000..14a6fbe6 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml @@ -0,0 +1,109 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What + tools do you have available?","role":"user"}],"mcp_servers":[{"name":"cloudflare-demo","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '225' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:31:16 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T20:31:15Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T20:31:15Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T20:31:14Z' + Retry-After: + - '45' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T20:31:15Z' + Request-Id: + - req_011CVHtAGQsRA3M7DK6ZGMnx + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '3042' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a127849b80ccf13-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01EbbCEVCE1g8MNxSNUyT4Fn","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to one tool:\n\n**cloudflare-demo_mcp_demo_day_info** - This tool + retrieves information about Cloudflare''s MCP Demo Day. You can use it if + you have questions about Cloudflare''s MCP demo day event.\n\nThis is the + only specialized tool I have available. For other questions or tasks, I can + help you directly using my general knowledge and conversational abilities."}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":95,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:31:16 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/anthropic/common_format/mcp_test.rb b/test/integration/anthropic/common_format/mcp_test.rb new file mode 100644 index 00000000..a73d4f6a --- /dev/null +++ b/test/integration/anthropic/common_format/mcp_test.rb @@ -0,0 +1,256 @@ +# frozen_string_literal: true + +require_relative "../../test_helper" + +# NOTE: MCP (Model Context Protocol) implementation is complete and API accepts MCP parameters +# +# The Anthropic gem v1.14.0 includes MCP support via Beta::MessageCreateParams with +# BetaRequestMCPServerURLDefinition models, and our implementation correctly uses these. +# +# The beta parameter (anthropic-beta: mcp-client-2025-04-04) is automatically added +# when mcp_servers are present in the request. No explicit configuration is needed. +# +# To test with real MCP servers: +# 1. Update the URLs to point to accessible MCP servers +# 2. Run tests to record VCR cassettes +# +# Unit tests validate the transformation logic in: +# test/providers/anthropic/transforms_test.rb (7 tests, all passing) + +module Integration + module Anthropic + module CommonFormat + class McpTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :anthropic, + model: "claude-haiku-4-5", + max_tokens: 1024 + + # Common format with single MCP server without authorization + COMMON_FORMAT_SINGLE_SERVER = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What tools do you have available?" + } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://demo-day.mcp.cloudflare.com/sse", + name: "cloudflare-demo" + } + ] + # Note: betas parameter is automatically added and transformed to anthropic-beta HTTP header + } + + def common_format_single_server + prompt( + message: "What tools do you have available?", + mcp_servers: [ + { + name: "cloudflare-demo", + url: "https://demo-day.mcp.cloudflare.com/sse" + } + ] + ) + end + + # Common format with single MCP server with authorization + COMMON_FORMAT_SINGLE_SERVER_WITH_AUTH = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What tools do you have available?" + } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://api.githubcopilot.com/mcp/", + name: "github-copilot", + authorization_token: "GITHUB_MCP_TOKEN" + } + ] + # Note: betas parameter is automatically added and transformed to anthropic-beta HTTP header + } + + def common_format_single_server_with_auth + prompt( + message: "What tools do you have available?", + mcp_servers: [ + { + name: "github-copilot", + url: "https://api.githubcopilot.com/mcp/", + authorization: ENV["GITHUB_MCP_TOKEN"] + } + ] + ) + end + + # Common format with multiple MCP servers + COMMON_FORMAT_MULTIPLE_SERVERS = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What tools do you have available?" + } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://demo-day.mcp.cloudflare.com/sse", + name: "cloudflare-demo" + }, + { + type: "url", + url: "https://api.githubcopilot.com/mcp/", + name: "github-copilot", + authorization_token: "GITHUB_MCP_TOKEN" + } + ] + # Note: betas parameter is automatically added and transformed to anthropic-beta HTTP header + } + + def common_format_multiple_servers + prompt( + message: "What tools do you have available?", + mcp_servers: [ + { + name: "cloudflare-demo", + url: "https://demo-day.mcp.cloudflare.com/sse" + }, + { + name: "github-copilot", + url: "https://api.githubcopilot.com/mcp/", + authorization: ENV["GITHUB_MCP_TOKEN"] + } + ] + ) + end + + # Common format with mixed auth and no auth servers + COMMON_FORMAT_MIXED_AUTH = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What tools do you have available?" + } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://demo-day.mcp.cloudflare.com/sse", + name: "cloudflare-demo" + }, + { + type: "url", + url: "https://api.githubcopilot.com/mcp/", + name: "github-copilot", + authorization_token: "GITHUB_MCP_TOKEN" + } + ] + # Note: betas parameter is automatically added and transformed to anthropic-beta HTTP header + } + + def common_format_mixed_auth + prompt( + message: "What tools do you have available?", + mcp_servers: [ + { + name: "cloudflare-demo", + url: "https://demo-day.mcp.cloudflare.com/sse" + }, + { + name: "github-copilot", + url: "https://api.githubcopilot.com/mcp/", + authorization: ENV["GITHUB_MCP_TOKEN"] + } + ] + ) + end + + # Common format with SSE endpoint (Cloudflare demo server) + COMMON_FORMAT_SSE_SERVER = { + model: "claude-haiku-4-5", + messages: [ + { + role: "user", + content: "What tools do you have available?" + } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://demo-day.mcp.cloudflare.com/sse", + name: "cloudflare-demo" + } + ] + # Note: betas parameter is transformed to anthropic-beta HTTP header + } + + def common_format_sse_server + prompt( + message: "What tools do you have available?", + mcp_servers: [ + { + name: "cloudflare-demo", + url: "https://demo-day.mcp.cloudflare.com/sse" + } + ] + ) + end + end + + # Test common format MCP scenarios with Cloudflare (no VCR filtering needed) + [ + :common_format_single_server, + :common_format_sse_server + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase)) + end + + # Test scenarios with VCR-filtered authorization tokens + # These tests verify the cassette recording but skip WebMock replay verification + # because VCR filters the token in the cassette but not in the live WebMock request + [ + :common_format_single_server_with_auth, + :common_format_multiple_servers, + :common_format_mixed_auth + ].each do |action_name| + agent_name = TestAgent.name.demodulize.underscore + expected_body = TestAgent.const_get(action_name.to_s.upcase) + + test "#{agent_name} #{action_name} Request Building" do + cassette_name = [ self.class.name.underscore, "#{agent_name}_#{action_name}" ].join("/") + + # Run once to record response + VCR.use_cassette(cassette_name) do + TestAgent.send(action_name).generate_now + end + + # Validate that the recorded request matches our expectations (with filtered values) + cassette_file = YAML.load_file("test/fixtures/vcr_cassettes/#{cassette_name}.yml") + saved_request_body = JSON.parse(cassette_file.dig("http_interactions", 0, "request", "body", "string"), symbolize_names: true) + + assert_equal expected_body, saved_request_body + + # Note: Skipping WebMock replay verification because VCR filters tokens in cassettes + # but WebMock sees unfiltered tokens in live requests, causing mismatches + end + end + end + end + end +end diff --git a/test/providers/anthropic/transforms_test.rb b/test/providers/anthropic/transforms_test.rb index f9b0af56..737127fd 100644 --- a/test/providers/anthropic/transforms_test.rb +++ b/test/providers/anthropic/transforms_test.rb @@ -460,8 +460,8 @@ def transforms assert_equal "hello", result[:messages][0][:content] end - test "cleanup_serialized_request removes mcp_servers" do - hash = { mcp_servers: { foo: "bar" }, model: "claude-3" } + test "cleanup_serialized_request removes empty mcp_servers" do + hash = { mcp_servers: [], model: "claude-3" } result = transforms.cleanup_serialized_request(hash, {}) @@ -469,6 +469,21 @@ def transforms assert_equal "claude-3", result[:model] end + test "cleanup_serialized_request keeps non-empty mcp_servers" do + hash = { + mcp_servers: [ + { type: "url", name: "stripe", url: "https://mcp.stripe.com" } + ], + model: "claude-3" + } + + result = transforms.cleanup_serialized_request(hash, {}) + + assert_not_nil result[:mcp_servers] + assert_equal 1, result[:mcp_servers].length + assert_equal "stripe", result[:mcp_servers][0][:name] + end + test "cleanup_serialized_request removes empty stop_sequences" do hash = { stop_sequences: [], model: "claude-3" } @@ -535,6 +550,107 @@ def transforms assert_equal "hello", compressed[:messages][0][:content] assert_equal "You are helpful", compressed[:system] end + + # normalize_mcp_servers tests + test "normalize_mcp_servers converts common format to Anthropic format" do + mcp_servers = [ + { + name: "stripe", + url: "https://mcp.stripe.com", + authorization: "sk_test_123" + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal 1, result.size + assert_equal "url", result[0][:type] + assert_equal "stripe", result[0][:name] + assert_equal "https://mcp.stripe.com", result[0][:url] + assert_equal "sk_test_123", result[0][:authorization_token] + end + + test "normalize_mcp_servers handles server without authorization" do + mcp_servers = [ + { + name: "public_api", + url: "https://mcp.public.com" + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal 1, result.size + assert_equal "url", result[0][:type] + assert_equal "public_api", result[0][:name] + assert_equal "https://mcp.public.com", result[0][:url] + assert_nil result[0][:authorization_token] + end + + test "normalize_mcp_servers preserves Anthropic format" do + mcp_servers = [ + { + type: "url", + name: "stripe", + url: "https://mcp.stripe.com" + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal 1, result.size + assert_equal "url", result[0][:type] + assert_equal "stripe", result[0][:name] + end + + test "normalize_mcp_servers handles multiple servers" do + mcp_servers = [ + { + name: "stripe", + url: "https://mcp.stripe.com", + authorization: "key1" + }, + { + name: "sendgrid", + url: "https://mcp.sendgrid.com", + authorization: "key2" + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal 2, result.size + assert_equal "stripe", result[0][:name] + assert_equal "sendgrid", result[1][:name] + assert_equal "key1", result[0][:authorization_token] + assert_equal "key2", result[1][:authorization_token] + end + + test "normalize_mcp_servers accepts authorization_token directly" do + mcp_servers = [ + { + name: "test", + url: "https://test.com", + authorization_token: "token123" + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal "token123", result[0][:authorization_token] + end + + test "normalize_mcp_servers returns nil for nil input" do + result = transforms.normalize_mcp_servers(nil) + + assert_nil result + end + + test "normalize_mcp_servers returns non-array unchanged" do + result = transforms.normalize_mcp_servers("not an array") + + assert_equal "not an array", result + end end end end diff --git a/test/test_helper.rb b/test/test_helper.rb index c6a82463..55047d33 100644 --- a/test/test_helper.rb +++ b/test/test_helper.rb @@ -101,11 +101,12 @@ def doc_example_output(example = nil, test_name = nil) config.cassette_library_dir = "test/fixtures/vcr_cassettes" config.hook_into :webmock - config.filter_sensitive_data("ACCESS_TOKEN") { ENV["OPEN_AI_ACCESS_TOKEN"] } - config.filter_sensitive_data("ORGANIZATION_ID") { ENV["OPEN_AI_ORGANIZATION_ID"] } - config.filter_sensitive_data("PROJECT_ID") { ENV["OPEN_AI_PROJECT_ID"] } - config.filter_sensitive_data("ACCESS_TOKEN") { ENV["OPEN_ROUTER_ACCESS_TOKEN"] } - config.filter_sensitive_data("ACCESS_TOKEN") { ENV["ANTHROPIC_ACCESS_TOKEN"] } + config.filter_sensitive_data("ACCESS_TOKEN") { ENV["OPEN_AI_ACCESS_TOKEN"] } + config.filter_sensitive_data("ORGANIZATION_ID") { ENV["OPEN_AI_ORGANIZATION_ID"] } + config.filter_sensitive_data("PROJECT_ID") { ENV["OPEN_AI_PROJECT_ID"] } + config.filter_sensitive_data("ACCESS_TOKEN") { ENV["OPEN_ROUTER_ACCESS_TOKEN"] } + config.filter_sensitive_data("ACCESS_TOKEN") { ENV["ANTHROPIC_ACCESS_TOKEN"] } + config.filter_sensitive_data("GITHUB_MCP_TOKEN") { ENV["GITHUB_MCP_TOKEN"] } end # Load fixtures from the engine From 96a375b35ce26abcd9a4ac28810e6fd6c571bf55 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 12:47:42 -0800 Subject: [PATCH 11/17] Add OpenAI Response MCP Support --- .../providers/open_ai/responses/request.rb | 12 +- .../providers/open_ai/responses/transforms.rb | 36 + ...gent_common_format_mixed_tools_and_mcp.yml | 433 ++++ ...t_agent_common_format_multiple_servers.yml | 1938 +++++++++++++++++ ...test_agent_common_format_single_server.yml | 204 ++ ..._common_format_single_server_with_auth.yml | 1911 ++++++++++++++++ .../responses/common_format/mcp_test.rb | 189 ++ .../open_ai/responses/transforms_test.rb | 76 + 8 files changed, 4797 insertions(+), 2 deletions(-) create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml create mode 100644 test/integration/open_ai/responses/common_format/mcp_test.rb diff --git a/lib/active_agent/providers/open_ai/responses/request.rb b/lib/active_agent/providers/open_ai/responses/request.rb index f42785b0..dd5cdf57 100644 --- a/lib/active_agent/providers/open_ai/responses/request.rb +++ b/lib/active_agent/providers/open_ai/responses/request.rb @@ -77,10 +77,18 @@ def initialize(**params) params[:tools] = Responses::Transforms.normalize_tools(params[:tools]) if params[:tools] params[:tool_choice] = Responses::Transforms.normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] - # Step 8: Create gem model - delegates to OpenAI gem + # Step 8: Normalize MCP servers from common format + # OpenAI treats MCP servers as a special type of tool in the tools array + if params[:mcp_servers]&.any? + normalized_mcp_tools = Responses::Transforms.normalize_mcp_servers(params.delete(:mcp_servers)) + # Merge MCP servers into tools array + params[:tools] = (params[:tools] || []) + normalized_mcp_tools + end + + # Step 9: Create gem model - delegates to OpenAI gem gem_model = ::OpenAI::Models::Responses::ResponseCreateParams.new(**params) - # Step 8: Delegate all method calls to gem model + # Step 10: Delegate all method calls to gem model super(gem_model) rescue ArgumentError => e # Re-raise with more context diff --git a/lib/active_agent/providers/open_ai/responses/transforms.rb b/lib/active_agent/providers/open_ai/responses/transforms.rb index 71a5874b..fca00a57 100644 --- a/lib/active_agent/providers/open_ai/responses/transforms.rb +++ b/lib/active_agent/providers/open_ai/responses/transforms.rb @@ -69,6 +69,42 @@ def normalize_tools(tools) end end + # Normalizes MCP servers from common format to OpenAI Responses API format. + # + # Common format: + # {name: "stripe", url: "https://...", authorization: "token"} + # OpenAI format: + # {type: "mcp", server_label: "stripe", server_url: "https://...", authorization: "token"} + # + # @param mcp_servers [Array] + # @return [Array] + def normalize_mcp_servers(mcp_servers) + return mcp_servers unless mcp_servers.is_a?(Array) + + mcp_servers.map do |server| + server_hash = server.is_a?(Hash) ? server.deep_symbolize_keys : server + + # If already in OpenAI format (has type: "mcp" and server_label), return as-is + if server_hash[:type] == "mcp" && server_hash[:server_label] + next server_hash + end + + # Convert common format to OpenAI format + result = { + type: "mcp", + server_label: server_hash[:name] || server_hash[:server_label], + server_url: server_hash[:url] || server_hash[:server_url] + } + + # Keep authorization field (OpenAI uses 'authorization', not 'authorization_token') + if server_hash[:authorization] + result[:authorization] = server_hash[:authorization] + end + + result.compact + end + end + # Normalizes tool_choice from common format to OpenAI Responses API format. # # Responses API uses flat format for specific tool choice, unlike Chat API's nested format. diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml new file mode 100644 index 00000000..8fd04a3e --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml @@ -0,0 +1,433 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-5","input":"Get the weather and calculate 5 + 3","tools":[{"type":"function","name":"calculate","description":"Perform + arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string"},"a":{"type":"number"},"b":{"type":"number"}}}},{"type":"mcp","server_label":"weather","server_url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '362' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:46:45 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '15000' + X-Ratelimit-Limit-Tokens: + - '40000000' + X-Ratelimit-Remaining-Requests: + - '14999' + X-Ratelimit-Remaining-Tokens: + - '39999822' + X-Ratelimit-Reset-Requests: + - 4ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_e0aa95a5caec410e9fdb14b57046aba1 + Openai-Processing-Ms: + - '23328' + X-Envoy-Upstream-Service-Time: + - '23335' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=l.Szu4p9ibKe8Xjpg8JeIgoJO2BmN0qqum.io_eHYJ8-1763585205-1.0.1.1-ZGiZ0_Xo1aDdtYB3qkB44ViDrgI7qu8BdwLQnOK5CftkO8jzvZIIQUL5.eJaV4D.PdpGKSjKLor2ulCYJg43wkZscXqbaK1n2adT9TkTwI4; + path=/; expires=Wed, 19-Nov-25 21:16:45 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=E0ZLmHrKaPmXjjSHFozFViW1pOnY2ZekrM2gBygn7oU-1763585205740-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a128e77fc64f973-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_08f5a80549ac3c9800691e2c9e74dc819baa3f1f2aa0b04c81", + "object": "response", + "created_at": 1763585182, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-5-2025-08-07", + "output": [ + { + "id": "mcpl_08f5a80549ac3c9800691e2c9ea194819ba35edc1fb8e5166a", + "type": "mcp_list_tools", + "server_label": "weather", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Get information about Cloudflare's MCP Demo Day. Use this tool if the user asks about Cloudflare's MCP demo day", + "input_schema": { + "type": "object", + "properties": {} + }, + "name": "mcp_demo_day_info" + } + ] + }, + { + "id": "rs_08f5a80549ac3c9800691e2ca18a10819b8ff9312fac3298a4", + "type": "reasoning", + "summary": [] + }, + { + "id": "fc_08f5a80549ac3c9800691e2cb313f4819b85f889bbc12d4f91", + "type": "function_call", + "status": "completed", + "arguments": "{\"operation\":\"add\",\"a\":5,\"b\":3}", + "call_id": "call_KPprJEhF7cJMtOWar14nESCk", + "name": "calculate" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": "medium", + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Perform arithmetic", + "name": "calculate", + "parameters": { + "type": "object", + "properties": { + "operation": { + "type": "string" + }, + "a": { + "type": "number" + }, + "b": { + "type": "number" + } + }, + "additionalProperties": false, + "required": [ + "operation", + "a", + "b" + ] + }, + "strict": true + }, + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "weather", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 110, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 667, + "output_tokens_details": { + "reasoning_tokens": 640 + }, + "total_tokens": 777 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 20:46:45 GMT +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-5","input":[{"role":"user","content":"Get the weather + and calculate 5 + 3"},{"id":"mcpl_08f5a80549ac3c9800691e2c9ea194819ba35edc1fb8e5166a","server_label":"weather","tools":[{"input_schema":{"type":"object","properties":{}},"name":"mcp_demo_day_info","annotations":{"read_only":false},"description":"Get + information about Cloudflare''s MCP Demo Day. Use this tool if the user asks + about Cloudflare''s MCP demo day"}],"type":"mcp_list_tools"},{"id":"rs_08f5a80549ac3c9800691e2ca18a10819b8ff9312fac3298a4","summary":[],"type":"reasoning"},{"arguments":"{\"operation\":\"add\",\"a\":5,\"b\":3}","call_id":"call_KPprJEhF7cJMtOWar14nESCk","name":"calculate","type":"function_call","id":"fc_08f5a80549ac3c9800691e2cb313f4819b85f889bbc12d4f91","status":"completed"},{"call_id":"call_KPprJEhF7cJMtOWar14nESCk","output":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","type":"function_call_output"}],"tools":[{"type":"function","name":"calculate","description":"Perform + arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string"},"a":{"type":"number"},"b":{"type":"number"}}}},{"type":"mcp","server_label":"weather","server_url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '1209' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:46:48 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '15000' + X-Ratelimit-Limit-Tokens: + - '40000000' + X-Ratelimit-Remaining-Requests: + - '14999' + X-Ratelimit-Remaining-Tokens: + - '40000000' + X-Ratelimit-Reset-Requests: + - 4ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_0ddd0d9d12ce4075959cef66d2c5c81c + Openai-Processing-Ms: + - '2549' + X-Envoy-Upstream-Service-Time: + - '2552' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=0h51r0xUubUKzwauERdI6XJWkiJg99OXCnCKwE.9Nhw-1763585208-1.0.1.1-wxkThT5uZxbRZGaQirsI7hQYnLxmumvJkmMoaU8p.i6_icyR0tHG6vTMnAJWPUjPvWtpnSM.G4qAmBlJv8qcNtR2gqAQA8a8Cn.tIMFCF0U; + path=/; expires=Wed, 19-Nov-25 21:16:48 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=kOCSRE9Q5scpXc5F2umRR8XlhV2_.BNe2DBNJvPnSUo-1763585208942-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a128f103b7dcfa8-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_08f5a80549ac3c9800691e2cb66654819b9c1c0dc1f10d9118", + "object": "response", + "created_at": 1763585206, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-5-2025-08-07", + "output": [ + { + "id": "msg_08f5a80549ac3c9800691e2cb768e8819bac854261a573efc4", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "- 5 + 3 = 8\n- Weather: I don\u2019t have live weather access here. Tell me the city/ZIP and date (e.g., \u201ctoday in Seattle\u201d), and I can suggest likely conditions based on climate norms or show you how to check quickly." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": "medium", + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Perform arithmetic", + "name": "calculate", + "parameters": { + "type": "object", + "properties": { + "operation": { + "type": "string" + }, + "a": { + "type": "number" + }, + "b": { + "type": "number" + } + }, + "additionalProperties": false, + "required": [ + "operation", + "a", + "b" + ] + }, + "strict": true + }, + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "weather", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 819, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 61, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 880 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 20:46:48 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml new file mode 100644 index 00000000..e3113440 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml @@ -0,0 +1,1938 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-5","input":"Get the weather and repository information","tools":[{"type":"mcp","server_label":"weather","server_url":"https://demo-day.mcp.cloudflare.com/sse"},{"type":"mcp","server_label":"github_copilot","server_url":"https://api.githubcopilot.com/mcp/","authorization":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '384' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:47:01 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '15000' + X-Ratelimit-Limit-Tokens: + - '40000000' + X-Ratelimit-Remaining-Requests: + - '14999' + X-Ratelimit-Remaining-Tokens: + - '39999742' + X-Ratelimit-Reset-Requests: + - 4ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_3a6c3c814288492cb4c9bcffda99d81e + Openai-Processing-Ms: + - '12146' + X-Envoy-Upstream-Service-Time: + - '12150' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=j2.cfQDVPxlsLwWrqPmnoLgRx6FjOvdhDAdZeNoiwws-1763585221-1.0.1.1-xMHbVgX_aU0ZBtQgMiIqXH.XN1a9SRt24.IvJaTK_G54jLIwVPqeBROke.LVRN2VAjY0eGnsP9daFP_4kSIvr6jIFCR0jOKkDvFXsM7Usgo; + path=/; expires=Wed, 19-Nov-25 21:17:01 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=HfhDIJwloIhF0Ex4_rIqS8SBlLrJguf2kwC9GiP7aHE-1763585221220-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a128f247e6715e0-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_01142e73b6db117e00691e2cb911f88198b67e43d7dc3c2291", + "object": "response", + "created_at": 1763585209, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-5-2025-08-07", + "output": [ + { + "id": "mcpl_01142e73b6db117e00691e2cb968608198a41622b3b25afc00", + "type": "mcp_list_tools", + "server_label": "weather", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Get information about Cloudflare's MCP Demo Day. Use this tool if the user asks about Cloudflare's MCP demo day", + "input_schema": { + "type": "object", + "properties": {} + }, + "name": "mcp_demo_day_info" + } + ] + }, + { + "id": "mcpl_01142e73b6db117e00691e2cb968f081989cfd717a72971239", + "type": "mcp_list_tools", + "server_label": "github_copilot", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Add review comment to the requester's latest pending pull request review. A pending review needs to already exist to call this (check with the user if not sure).", + "input_schema": { + "properties": { + "body": { + "description": "The text of the review comment", + "type": "string" + }, + "line": { + "description": "The line of the blob in the pull request diff that the comment applies to. For multi-line comments, the last line of the range", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "path": { + "description": "The relative path to the file that necessitates a comment", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "side": { + "description": "The side of the diff to comment on. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "startLine": { + "description": "For multi-line comments, the first line of the range that the comment applies to", + "type": "number" + }, + "startSide": { + "description": "For multi-line comments, the starting side of the diff that the comment applies to. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "subjectType": { + "description": "The level at which the comment is targeted", + "enum": [ + "FILE", + "LINE" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber", + "path", + "body", + "subjectType" + ], + "type": "object" + }, + "name": "add_comment_to_pending_review" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a comment to a specific issue in a GitHub repository. Use this tool to add comments to pull requests as well (in this case pass pull request number as issue_number), but only if user is not asking specifically to add review comments.", + "input_schema": { + "properties": { + "body": { + "description": "Comment content", + "type": "string" + }, + "issue_number": { + "description": "Issue number to comment on", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issue_number", + "body" + ], + "type": "object" + }, + "name": "add_issue_comment" + }, + { + "annotations": { + "read_only": false + }, + "description": "Assign Copilot to a specific issue in a GitHub repository.\n\nThis tool can help with the following outcomes:\n- a Pull Request created with source code changes to resolve the issue\n\n\nMore information can be found at:\n- https://docs.github.com/en/copilot/using-github-copilot/using-copilot-coding-agent-to-work-on-tasks/about-assigning-tasks-to-copilot\n", + "input_schema": { + "properties": { + "issueNumber": { + "description": "Issue number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issueNumber" + ], + "type": "object" + }, + "name": "assign_copilot_to_issue" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new branch in a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Name for new branch", + "type": "string" + }, + "from_branch": { + "description": "Source branch (defaults to repo default)", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch" + ], + "type": "object" + }, + "name": "create_branch" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create or update a single file in a GitHub repository. If updating, you must provide the SHA of the file you want to update. Use this tool to create or update a file in a GitHub repository remotely; do not use it for local file operations.", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to create/update the file in", + "type": "string" + }, + "content": { + "description": "Content of the file", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path where to create/update the file", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Required if updating an existing file. The blob SHA of the file being replaced.", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "content", + "message", + "branch" + ], + "type": "object" + }, + "name": "create_or_update_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "Branch to merge into", + "type": "string" + }, + "body": { + "description": "PR description", + "type": "string" + }, + "draft": { + "description": "Create as draft PR", + "type": "boolean" + }, + "head": { + "description": "Branch containing changes", + "type": "string" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "title": { + "description": "PR title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "title", + "head", + "base" + ], + "type": "object" + }, + "name": "create_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new GitHub repository in your account or specified organization", + "input_schema": { + "properties": { + "autoInit": { + "description": "Initialize with README", + "type": "boolean" + }, + "description": { + "description": "Repository description", + "type": "string" + }, + "name": { + "description": "Repository name", + "type": "string" + }, + "organization": { + "description": "Organization to create the repository in (omit to create in your personal account)", + "type": "string" + }, + "private": { + "description": "Whether repo should be private", + "type": "boolean" + } + }, + "required": [ + "name" + ], + "type": "object" + }, + "name": "create_repository" + }, + { + "annotations": { + "read_only": false + }, + "description": "Delete a file from a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to delete the file from", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path to the file to delete", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "message", + "branch" + ], + "type": "object" + }, + "name": "delete_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Fork a GitHub repository to your account or specified organization", + "input_schema": { + "properties": { + "organization": { + "description": "Organization to fork to", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "fork_repository" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details for a commit from a GitHub repository", + "input_schema": { + "properties": { + "include_diff": { + "default": true, + "description": "Whether to include file diffs and stats in the response. Default is true.", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch name, or tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "sha" + ], + "type": "object" + }, + "name": "get_commit" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the contents of a file or directory from a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "default": "/", + "description": "Path to file/directory (directories must end with a slash '/')", + "type": "string" + }, + "ref": { + "description": "Accepts optional git refs such as `refs/tags/{tag}`, `refs/heads/{branch}` or `refs/pull/{pr_number}/head`", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Accepts optional commit SHA. If specified, it will be used instead of ref", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_file_contents" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific label from a repository.", + "input_schema": { + "properties": { + "name": { + "description": "Label name.", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization name)", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "name" + ], + "type": "object" + }, + "name": "get_label" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the latest release in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_latest_release" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the authenticated GitHub user. Use this when a request is about the user's own profile for GitHub. Or when information is missing to build other tool calls.", + "input_schema": { + "properties": {}, + "type": "object" + }, + "name": "get_me" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific release by its tag name in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name (e.g., 'v1.0.0')", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_release_by_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details about a specific git tag in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get member usernames of a specific team in an organization. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "org": { + "description": "Organization login (owner) that contains the team.", + "type": "string" + }, + "team_slug": { + "description": "Team slug", + "type": "string" + } + }, + "required": [ + "org", + "team_slug" + ], + "type": "object" + }, + "name": "get_team_members" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the teams the user is a member of. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "user": { + "description": "Username to get teams for. If not provided, uses the authenticated user.", + "type": "string" + } + }, + "type": "object" + }, + "name": "get_teams" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information about a specific issue in a GitHub repository.", + "input_schema": { + "properties": { + "issue_number": { + "description": "The number of the issue", + "type": "number" + }, + "method": { + "description": "The read operation to perform on a single issue. \nOptions are: \n1. get - Get details of a specific issue.\n2. get_comments - Get issue comments.\n3. get_sub_issues - Get sub-issues of the issue.\n4. get_labels - Get labels assigned to the issue.\n", + "enum": [ + "get", + "get_comments", + "get_sub_issues", + "get_labels" + ], + "type": "string" + }, + "owner": { + "description": "The owner of the repository", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "The name of the repository", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number" + ], + "type": "object" + }, + "name": "issue_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new or update an existing issue in a GitHub repository.", + "input_schema": { + "properties": { + "assignees": { + "description": "Usernames to assign to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "body": { + "description": "Issue body content", + "type": "string" + }, + "duplicate_of": { + "description": "Issue number that this issue is a duplicate of. Only used when state_reason is 'duplicate'.", + "type": "number" + }, + "issue_number": { + "description": "Issue number to update", + "type": "number" + }, + "labels": { + "description": "Labels to apply to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "method": { + "description": "Write operation to perform on a single issue.\nOptions are: \n- 'create' - creates a new issue. \n- 'update' - updates an existing issue.\n", + "enum": [ + "create", + "update" + ], + "type": "string" + }, + "milestone": { + "description": "Milestone number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "state_reason": { + "description": "Reason for the state change. Ignored unless state is changed.", + "enum": [ + "completed", + "not_planned", + "duplicate" + ], + "type": "string" + }, + "title": { + "description": "Issue title", + "type": "string" + }, + "type": { + "description": "Type of this issue. Only use if the repository has issue types configured. Use list_issue_types tool to get valid type values for the organization. If the repository doesn't support issue types, omit this parameter.", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo" + ], + "type": "object" + }, + "name": "issue_write" + }, + { + "annotations": { + "read_only": true + }, + "description": "List branches in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_branches" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get list of commits of a branch in a GitHub repository. Returns at least 30 results per page by default, but can return more if specified using the perPage parameter (up to 100).", + "input_schema": { + "properties": { + "author": { + "description": "Author username or email address to filter commits by", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch or tag name to list commits of. If not provided, uses the default branch of the repository. If a commit SHA is provided, will list commits up to that SHA.", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_commits" + }, + { + "annotations": { + "read_only": true + }, + "description": "List supported issue types for repository owner (organization).", + "input_schema": { + "properties": { + "owner": { + "description": "The organization owner of the repository", + "type": "string" + } + }, + "required": [ + "owner" + ], + "type": "object" + }, + "name": "list_issue_types" + }, + { + "annotations": { + "read_only": true + }, + "description": "List issues in a GitHub repository. For pagination, use the 'endCursor' from the previous response's 'pageInfo' in the 'after' parameter.", + "input_schema": { + "properties": { + "after": { + "description": "Cursor for pagination. Use the endCursor from the previous page's PageInfo for GraphQL APIs.", + "type": "string" + }, + "direction": { + "description": "Order direction. If provided, the 'orderBy' also needs to be provided.", + "enum": [ + "ASC", + "DESC" + ], + "type": "string" + }, + "labels": { + "description": "Filter by labels", + "items": { + "type": "string" + }, + "type": "array" + }, + "orderBy": { + "description": "Order issues by field. If provided, the 'direction' also needs to be provided.", + "enum": [ + "CREATED_AT", + "UPDATED_AT", + "COMMENTS" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "since": { + "description": "Filter by date (ISO 8601 timestamp)", + "type": "string" + }, + "state": { + "description": "Filter by state, by default both open and closed issues are returned when not provided", + "enum": [ + "OPEN", + "CLOSED" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "List pull requests in a GitHub repository. If the user specifies an author, then DO NOT use this tool and use the search_pull_requests tool instead.", + "input_schema": { + "properties": { + "base": { + "description": "Filter by base branch", + "type": "string" + }, + "direction": { + "description": "Sort direction", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "head": { + "description": "Filter by head user/org and branch", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sort": { + "description": "Sort by", + "enum": [ + "created", + "updated", + "popularity", + "long-running" + ], + "type": "string" + }, + "state": { + "description": "Filter by state", + "enum": [ + "open", + "closed", + "all" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "List releases in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_releases" + }, + { + "annotations": { + "read_only": true + }, + "description": "List git tags in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_tags" + }, + { + "annotations": { + "read_only": false + }, + "description": "Merge a pull request in a GitHub repository.", + "input_schema": { + "properties": { + "commit_message": { + "description": "Extra detail for merge commit", + "type": "string" + }, + "commit_title": { + "description": "Title for merge commit", + "type": "string" + }, + "merge_method": { + "description": "Merge method", + "enum": [ + "merge", + "squash", + "rebase" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "merge_pull_request" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information on a specific pull request in GitHub repository.", + "input_schema": { + "properties": { + "method": { + "description": "Action to specify what pull request data needs to be retrieved from GitHub. \nPossible options: \n 1. get - Get details of a specific pull request.\n 2. get_diff - Get the diff of a pull request.\n 3. get_status - Get status of a head commit in a pull request. This reflects status of builds and checks.\n 4. get_files - Get the list of files changed in a pull request. Use with pagination parameters to control the number of results returned.\n 5. get_review_comments - Get the review comments on a pull request. They are comments made on a portion of the unified diff during a pull request review. Use with pagination parameters to control the number of results returned.\n 6. get_reviews - Get the reviews on a pull request. When asked for review comments, use get_review_comments method.\n 7. get_comments - Get comments on a pull request. Use this if user doesn't specifically want review comments. Use with pagination parameters to control the number of results returned.\n", + "enum": [ + "get", + "get_diff", + "get_status", + "get_files", + "get_review_comments", + "get_reviews", + "get_comments" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create and/or submit, delete review of a pull request.\n\nAvailable methods:\n- create: Create a new review of a pull request. If \"event\" parameter is provided, the review is submitted. If \"event\" is omitted, a pending review is created.\n- submit_pending: Submit an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request. The \"body\" and \"event\" parameters are used when submitting the review.\n- delete_pending: Delete an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request.\n", + "input_schema": { + "properties": { + "body": { + "description": "Review comment text", + "type": "string" + }, + "commitID": { + "description": "SHA of commit to review", + "type": "string" + }, + "event": { + "description": "Review action to perform.", + "enum": [ + "APPROVE", + "REQUEST_CHANGES", + "COMMENT" + ], + "type": "string" + }, + "method": { + "description": "The write operation to perform on pull request review.", + "enum": [ + "create", + "submit_pending", + "delete_pending" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_review_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Push multiple files to a GitHub repository in a single commit", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to push to", + "type": "string" + }, + "files": { + "description": "Array of file objects to push, each object with path (string) and content (string)", + "items": { + "additionalProperties": false, + "properties": { + "content": { + "description": "file content", + "type": "string" + }, + "path": { + "description": "path to the file", + "type": "string" + } + }, + "required": [ + "path", + "content" + ], + "type": "object" + }, + "type": "array" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch", + "files", + "message" + ], + "type": "object" + }, + "name": "push_files" + }, + { + "annotations": { + "read_only": false + }, + "description": "Request a GitHub Copilot code review for a pull request. Use this for automated feedback on pull requests, usually before requesting a human reviewer.", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "request_copilot_review" + }, + { + "annotations": { + "read_only": true + }, + "description": "Fast and precise code search across ALL GitHub repositories using GitHub's native search engine. Best for finding exact symbols, functions, classes, or specific code patterns.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order for results", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub's powerful code search syntax. Examples: 'content:Skill language:Java org:github', 'NOT is:archived language:Python OR language:go', 'repo:github/github-mcp-server'. Supports exact matching, language filters, path filters, and more.", + "type": "string" + }, + "sort": { + "description": "Sort field ('indexed' only)", + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_code" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for issues in GitHub repositories using issues search syntax already scoped to is:issue", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only issues for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub issues search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only issues for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for pull requests in GitHub repositories using issues search syntax already scoped to is:pr", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only pull requests for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub pull request search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only pull requests for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub repositories by name, description, readme, topics, or other metadata. Perfect for discovering projects, finding examples, or locating specific repositories across GitHub.", + "input_schema": { + "properties": { + "minimal_output": { + "default": true, + "description": "Return minimal repository information (default: true). When false, returns full GitHub API repository objects.", + "type": "boolean" + }, + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Repository search query. Examples: 'machine learning in:name stars:>1000 language:python', 'topic:react', 'user:facebook'. Supports advanced search syntax for precise filtering.", + "type": "string" + }, + "sort": { + "description": "Sort repositories by field, defaults to best match", + "enum": [ + "stars", + "forks", + "help-wanted-issues", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_repositories" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub users by username, real name, or other profile information. Useful for locating developers, contributors, or team members.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "User search query. Examples: 'john smith', 'location:seattle', 'followers:>100'. Search is automatically scoped to type:user.", + "type": "string" + }, + "sort": { + "description": "Sort users by number of followers or repositories, or when the person joined GitHub.", + "enum": [ + "followers", + "repositories", + "joined" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_users" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a sub-issue to a parent issue in a GitHub repository.", + "input_schema": { + "properties": { + "after_id": { + "description": "The ID of the sub-issue to be prioritized after (either after_id OR before_id should be specified)", + "type": "number" + }, + "before_id": { + "description": "The ID of the sub-issue to be prioritized before (either after_id OR before_id should be specified)", + "type": "number" + }, + "issue_number": { + "description": "The number of the parent issue", + "type": "number" + }, + "method": { + "description": "The action to perform on a single sub-issue\nOptions are:\n- 'add' - add a sub-issue to a parent issue in a GitHub repository.\n- 'remove' - remove a sub-issue from a parent issue in a GitHub repository.\n- 'reprioritize' - change the order of sub-issues within a parent issue in a GitHub repository. Use either 'after_id' or 'before_id' to specify the new position.\n\t\t\t\t", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "replace_parent": { + "description": "When true, replaces the sub-issue's current parent issue. Use with 'add' method only.", + "type": "boolean" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sub_issue_id": { + "description": "The ID of the sub-issue to add. ID is not the same as issue number", + "type": "number" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number", + "sub_issue_id" + ], + "type": "object" + }, + "name": "sub_issue_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update an existing pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "New base branch name", + "type": "string" + }, + "body": { + "description": "New description", + "type": "string" + }, + "draft": { + "description": "Mark pull request as draft (true) or ready for review (false)", + "type": "boolean" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number to update", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "reviewers": { + "description": "GitHub usernames to request reviews from", + "items": { + "type": "string" + }, + "type": "array" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "title": { + "description": "New title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update the branch of a pull request with the latest changes from the base branch.", + "input_schema": { + "properties": { + "expectedHeadSha": { + "description": "The expected SHA of the pull request's HEAD ref", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request_branch" + } + ] + }, + { + "id": "rs_01142e73b6db117e00691e2cbb47308198b2c8f0e0c04ed9e7", + "type": "reasoning", + "summary": [] + }, + { + "id": "msg_01142e73b6db117e00691e2cc3337c8198a4606717b4ac6447", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Happy to help! To get started, I need a couple details:\n\n- Weather:\n - Which location (city and country or ZIP/postcode)?\n - Do you want current conditions, today\u2019s forecast, or a multi\u2011day forecast?\n\n- Repository information:\n - Which GitHub repository (owner/name or a URL)?\n - What details do you want (e.g., branches, latest commits, open issues/PRs, releases, tags)?" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": "medium", + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "weather", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + }, + { + "type": "mcp", + "allowed_tools": null, + "authorization": "", + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "github_copilot", + "server_url": "https://api.githubcopilot.com/mcp/" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 5369, + "input_tokens_details": { + "cached_tokens": 5248 + }, + "output_tokens": 482, + "output_tokens_details": { + "reasoning_tokens": 384 + }, + "total_tokens": 5851 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 20:47:01 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml new file mode 100644 index 00000000..c4e4aa8f --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml @@ -0,0 +1,204 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-5","input":"Get the current weather","tools":[{"type":"mcp","server_label":"weather","server_url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '156' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:46:21 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '15000' + X-Ratelimit-Limit-Tokens: + - '40000000' + X-Ratelimit-Remaining-Requests: + - '14999' + X-Ratelimit-Remaining-Tokens: + - '39999994' + X-Ratelimit-Reset-Requests: + - 4ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_a0c5f02486ed497abb5d00386934bdd8 + Openai-Processing-Ms: + - '8667' + X-Envoy-Upstream-Service-Time: + - '8670' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=hEMCBTQW2kbtAW3WZlboz0.6poiAhEfPOmz0KuCKxBA-1763585181-1.0.1.1-6CSuuuouS3h6lNKBeSKHyL02OMvCAYjGjtPdJVJAAIVrsQ8HAk2vSUyOuXNN17MJ1eEXsBSQFV4QuH._fMIpsYxFwn.DZjjH9MjdBYkZb8o; + path=/; expires=Wed, 19-Nov-25 21:16:21 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=kWh83s9g4apTauc3btcfmRN7gISFSYbxm6AUx7mq0_c-1763585181322-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a128e40ad027aaf-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_0a61b801b444d08000691e2c94a29c8197912046d330286604", + "object": "response", + "created_at": 1763585172, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-5-2025-08-07", + "output": [ + { + "id": "mcpl_0a61b801b444d08000691e2c94e7e48197b1c74f8cfdb06d07", + "type": "mcp_list_tools", + "server_label": "weather", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Get information about Cloudflare's MCP Demo Day. Use this tool if the user asks about Cloudflare's MCP demo day", + "input_schema": { + "type": "object", + "properties": {} + }, + "name": "mcp_demo_day_info" + } + ] + }, + { + "id": "rs_0a61b801b444d08000691e2c9698588197827178deb303ddb2", + "type": "reasoning", + "summary": [] + }, + { + "id": "msg_0a61b801b444d08000691e2c9c38688197ae98d8ca7b672d05", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Sure\u2014what location should I use? Please share a city and country (or ZIP/postcode or coordinates), and your preferred units (C or F)." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": "medium", + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "weather", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 142, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 229, + "output_tokens_details": { + "reasoning_tokens": 192 + }, + "total_tokens": 371 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 20:46:21 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml new file mode 100644 index 00000000..28b1a402 --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml @@ -0,0 +1,1911 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-5","input":"Get repository information","tools":[{"type":"mcp","server_label":"github_copilot","server_url":"https://api.githubcopilot.com/mcp/","authorization":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '273' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:46:12 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '15000' + X-Ratelimit-Limit-Tokens: + - '40000000' + X-Ratelimit-Remaining-Requests: + - '14999' + X-Ratelimit-Remaining-Tokens: + - '39999552' + X-Ratelimit-Reset-Requests: + - 4ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_725ba7e71fda420a94b554e9145875e4 + Openai-Processing-Ms: + - '18277' + X-Envoy-Upstream-Service-Time: + - '18280' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=9Y5IzrkvtruiIpBJPjA_YtnMLViW09gnhXpCFU5MIr4-1763585172-1.0.1.1-2BtyiLqLt5Tk7jGO03TuGBSmxSEf9zkw7ePAewveycnA4zEyh1aeBbIl2repFN4fTRTSnVIx4O_4R9qCbIsypuWejk9VM9Z.hxB5QueNYsE; + path=/; expires=Wed, 19-Nov-25 21:16:12 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=XS4xf6HTx0IcH9mL6bPrTEpDK9hIAo3tN57JX3FLGBk-1763585172364-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a128dcc7dd0fa62-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_0bbb974cbbc378f500691e2c8216c0819891982946cb6cf0ad", + "object": "response", + "created_at": 1763585154, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-5-2025-08-07", + "output": [ + { + "id": "mcpl_0bbb974cbbc378f500691e2c82402c8198bdf001518e0d9a7e", + "type": "mcp_list_tools", + "server_label": "github_copilot", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Add review comment to the requester's latest pending pull request review. A pending review needs to already exist to call this (check with the user if not sure).", + "input_schema": { + "properties": { + "body": { + "description": "The text of the review comment", + "type": "string" + }, + "line": { + "description": "The line of the blob in the pull request diff that the comment applies to. For multi-line comments, the last line of the range", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "path": { + "description": "The relative path to the file that necessitates a comment", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "side": { + "description": "The side of the diff to comment on. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "startLine": { + "description": "For multi-line comments, the first line of the range that the comment applies to", + "type": "number" + }, + "startSide": { + "description": "For multi-line comments, the starting side of the diff that the comment applies to. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "subjectType": { + "description": "The level at which the comment is targeted", + "enum": [ + "FILE", + "LINE" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber", + "path", + "body", + "subjectType" + ], + "type": "object" + }, + "name": "add_comment_to_pending_review" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a comment to a specific issue in a GitHub repository. Use this tool to add comments to pull requests as well (in this case pass pull request number as issue_number), but only if user is not asking specifically to add review comments.", + "input_schema": { + "properties": { + "body": { + "description": "Comment content", + "type": "string" + }, + "issue_number": { + "description": "Issue number to comment on", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issue_number", + "body" + ], + "type": "object" + }, + "name": "add_issue_comment" + }, + { + "annotations": { + "read_only": false + }, + "description": "Assign Copilot to a specific issue in a GitHub repository.\n\nThis tool can help with the following outcomes:\n- a Pull Request created with source code changes to resolve the issue\n\n\nMore information can be found at:\n- https://docs.github.com/en/copilot/using-github-copilot/using-copilot-coding-agent-to-work-on-tasks/about-assigning-tasks-to-copilot\n", + "input_schema": { + "properties": { + "issueNumber": { + "description": "Issue number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issueNumber" + ], + "type": "object" + }, + "name": "assign_copilot_to_issue" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new branch in a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Name for new branch", + "type": "string" + }, + "from_branch": { + "description": "Source branch (defaults to repo default)", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch" + ], + "type": "object" + }, + "name": "create_branch" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create or update a single file in a GitHub repository. If updating, you must provide the SHA of the file you want to update. Use this tool to create or update a file in a GitHub repository remotely; do not use it for local file operations.", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to create/update the file in", + "type": "string" + }, + "content": { + "description": "Content of the file", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path where to create/update the file", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Required if updating an existing file. The blob SHA of the file being replaced.", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "content", + "message", + "branch" + ], + "type": "object" + }, + "name": "create_or_update_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "Branch to merge into", + "type": "string" + }, + "body": { + "description": "PR description", + "type": "string" + }, + "draft": { + "description": "Create as draft PR", + "type": "boolean" + }, + "head": { + "description": "Branch containing changes", + "type": "string" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "title": { + "description": "PR title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "title", + "head", + "base" + ], + "type": "object" + }, + "name": "create_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new GitHub repository in your account or specified organization", + "input_schema": { + "properties": { + "autoInit": { + "description": "Initialize with README", + "type": "boolean" + }, + "description": { + "description": "Repository description", + "type": "string" + }, + "name": { + "description": "Repository name", + "type": "string" + }, + "organization": { + "description": "Organization to create the repository in (omit to create in your personal account)", + "type": "string" + }, + "private": { + "description": "Whether repo should be private", + "type": "boolean" + } + }, + "required": [ + "name" + ], + "type": "object" + }, + "name": "create_repository" + }, + { + "annotations": { + "read_only": false + }, + "description": "Delete a file from a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to delete the file from", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path to the file to delete", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "message", + "branch" + ], + "type": "object" + }, + "name": "delete_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Fork a GitHub repository to your account or specified organization", + "input_schema": { + "properties": { + "organization": { + "description": "Organization to fork to", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "fork_repository" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details for a commit from a GitHub repository", + "input_schema": { + "properties": { + "include_diff": { + "default": true, + "description": "Whether to include file diffs and stats in the response. Default is true.", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch name, or tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "sha" + ], + "type": "object" + }, + "name": "get_commit" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the contents of a file or directory from a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "default": "/", + "description": "Path to file/directory (directories must end with a slash '/')", + "type": "string" + }, + "ref": { + "description": "Accepts optional git refs such as `refs/tags/{tag}`, `refs/heads/{branch}` or `refs/pull/{pr_number}/head`", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Accepts optional commit SHA. If specified, it will be used instead of ref", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_file_contents" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific label from a repository.", + "input_schema": { + "properties": { + "name": { + "description": "Label name.", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization name)", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "name" + ], + "type": "object" + }, + "name": "get_label" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the latest release in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_latest_release" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the authenticated GitHub user. Use this when a request is about the user's own profile for GitHub. Or when information is missing to build other tool calls.", + "input_schema": { + "properties": {}, + "type": "object" + }, + "name": "get_me" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific release by its tag name in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name (e.g., 'v1.0.0')", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_release_by_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details about a specific git tag in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get member usernames of a specific team in an organization. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "org": { + "description": "Organization login (owner) that contains the team.", + "type": "string" + }, + "team_slug": { + "description": "Team slug", + "type": "string" + } + }, + "required": [ + "org", + "team_slug" + ], + "type": "object" + }, + "name": "get_team_members" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the teams the user is a member of. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "user": { + "description": "Username to get teams for. If not provided, uses the authenticated user.", + "type": "string" + } + }, + "type": "object" + }, + "name": "get_teams" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information about a specific issue in a GitHub repository.", + "input_schema": { + "properties": { + "issue_number": { + "description": "The number of the issue", + "type": "number" + }, + "method": { + "description": "The read operation to perform on a single issue. \nOptions are: \n1. get - Get details of a specific issue.\n2. get_comments - Get issue comments.\n3. get_sub_issues - Get sub-issues of the issue.\n4. get_labels - Get labels assigned to the issue.\n", + "enum": [ + "get", + "get_comments", + "get_sub_issues", + "get_labels" + ], + "type": "string" + }, + "owner": { + "description": "The owner of the repository", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "The name of the repository", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number" + ], + "type": "object" + }, + "name": "issue_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new or update an existing issue in a GitHub repository.", + "input_schema": { + "properties": { + "assignees": { + "description": "Usernames to assign to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "body": { + "description": "Issue body content", + "type": "string" + }, + "duplicate_of": { + "description": "Issue number that this issue is a duplicate of. Only used when state_reason is 'duplicate'.", + "type": "number" + }, + "issue_number": { + "description": "Issue number to update", + "type": "number" + }, + "labels": { + "description": "Labels to apply to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "method": { + "description": "Write operation to perform on a single issue.\nOptions are: \n- 'create' - creates a new issue. \n- 'update' - updates an existing issue.\n", + "enum": [ + "create", + "update" + ], + "type": "string" + }, + "milestone": { + "description": "Milestone number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "state_reason": { + "description": "Reason for the state change. Ignored unless state is changed.", + "enum": [ + "completed", + "not_planned", + "duplicate" + ], + "type": "string" + }, + "title": { + "description": "Issue title", + "type": "string" + }, + "type": { + "description": "Type of this issue. Only use if the repository has issue types configured. Use list_issue_types tool to get valid type values for the organization. If the repository doesn't support issue types, omit this parameter.", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo" + ], + "type": "object" + }, + "name": "issue_write" + }, + { + "annotations": { + "read_only": true + }, + "description": "List branches in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_branches" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get list of commits of a branch in a GitHub repository. Returns at least 30 results per page by default, but can return more if specified using the perPage parameter (up to 100).", + "input_schema": { + "properties": { + "author": { + "description": "Author username or email address to filter commits by", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch or tag name to list commits of. If not provided, uses the default branch of the repository. If a commit SHA is provided, will list commits up to that SHA.", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_commits" + }, + { + "annotations": { + "read_only": true + }, + "description": "List supported issue types for repository owner (organization).", + "input_schema": { + "properties": { + "owner": { + "description": "The organization owner of the repository", + "type": "string" + } + }, + "required": [ + "owner" + ], + "type": "object" + }, + "name": "list_issue_types" + }, + { + "annotations": { + "read_only": true + }, + "description": "List issues in a GitHub repository. For pagination, use the 'endCursor' from the previous response's 'pageInfo' in the 'after' parameter.", + "input_schema": { + "properties": { + "after": { + "description": "Cursor for pagination. Use the endCursor from the previous page's PageInfo for GraphQL APIs.", + "type": "string" + }, + "direction": { + "description": "Order direction. If provided, the 'orderBy' also needs to be provided.", + "enum": [ + "ASC", + "DESC" + ], + "type": "string" + }, + "labels": { + "description": "Filter by labels", + "items": { + "type": "string" + }, + "type": "array" + }, + "orderBy": { + "description": "Order issues by field. If provided, the 'direction' also needs to be provided.", + "enum": [ + "CREATED_AT", + "UPDATED_AT", + "COMMENTS" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "since": { + "description": "Filter by date (ISO 8601 timestamp)", + "type": "string" + }, + "state": { + "description": "Filter by state, by default both open and closed issues are returned when not provided", + "enum": [ + "OPEN", + "CLOSED" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "List pull requests in a GitHub repository. If the user specifies an author, then DO NOT use this tool and use the search_pull_requests tool instead.", + "input_schema": { + "properties": { + "base": { + "description": "Filter by base branch", + "type": "string" + }, + "direction": { + "description": "Sort direction", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "head": { + "description": "Filter by head user/org and branch", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sort": { + "description": "Sort by", + "enum": [ + "created", + "updated", + "popularity", + "long-running" + ], + "type": "string" + }, + "state": { + "description": "Filter by state", + "enum": [ + "open", + "closed", + "all" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "List releases in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_releases" + }, + { + "annotations": { + "read_only": true + }, + "description": "List git tags in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_tags" + }, + { + "annotations": { + "read_only": false + }, + "description": "Merge a pull request in a GitHub repository.", + "input_schema": { + "properties": { + "commit_message": { + "description": "Extra detail for merge commit", + "type": "string" + }, + "commit_title": { + "description": "Title for merge commit", + "type": "string" + }, + "merge_method": { + "description": "Merge method", + "enum": [ + "merge", + "squash", + "rebase" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "merge_pull_request" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information on a specific pull request in GitHub repository.", + "input_schema": { + "properties": { + "method": { + "description": "Action to specify what pull request data needs to be retrieved from GitHub. \nPossible options: \n 1. get - Get details of a specific pull request.\n 2. get_diff - Get the diff of a pull request.\n 3. get_status - Get status of a head commit in a pull request. This reflects status of builds and checks.\n 4. get_files - Get the list of files changed in a pull request. Use with pagination parameters to control the number of results returned.\n 5. get_review_comments - Get the review comments on a pull request. They are comments made on a portion of the unified diff during a pull request review. Use with pagination parameters to control the number of results returned.\n 6. get_reviews - Get the reviews on a pull request. When asked for review comments, use get_review_comments method.\n 7. get_comments - Get comments on a pull request. Use this if user doesn't specifically want review comments. Use with pagination parameters to control the number of results returned.\n", + "enum": [ + "get", + "get_diff", + "get_status", + "get_files", + "get_review_comments", + "get_reviews", + "get_comments" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create and/or submit, delete review of a pull request.\n\nAvailable methods:\n- create: Create a new review of a pull request. If \"event\" parameter is provided, the review is submitted. If \"event\" is omitted, a pending review is created.\n- submit_pending: Submit an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request. The \"body\" and \"event\" parameters are used when submitting the review.\n- delete_pending: Delete an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request.\n", + "input_schema": { + "properties": { + "body": { + "description": "Review comment text", + "type": "string" + }, + "commitID": { + "description": "SHA of commit to review", + "type": "string" + }, + "event": { + "description": "Review action to perform.", + "enum": [ + "APPROVE", + "REQUEST_CHANGES", + "COMMENT" + ], + "type": "string" + }, + "method": { + "description": "The write operation to perform on pull request review.", + "enum": [ + "create", + "submit_pending", + "delete_pending" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_review_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Push multiple files to a GitHub repository in a single commit", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to push to", + "type": "string" + }, + "files": { + "description": "Array of file objects to push, each object with path (string) and content (string)", + "items": { + "additionalProperties": false, + "properties": { + "content": { + "description": "file content", + "type": "string" + }, + "path": { + "description": "path to the file", + "type": "string" + } + }, + "required": [ + "path", + "content" + ], + "type": "object" + }, + "type": "array" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch", + "files", + "message" + ], + "type": "object" + }, + "name": "push_files" + }, + { + "annotations": { + "read_only": false + }, + "description": "Request a GitHub Copilot code review for a pull request. Use this for automated feedback on pull requests, usually before requesting a human reviewer.", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "request_copilot_review" + }, + { + "annotations": { + "read_only": true + }, + "description": "Fast and precise code search across ALL GitHub repositories using GitHub's native search engine. Best for finding exact symbols, functions, classes, or specific code patterns.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order for results", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub's powerful code search syntax. Examples: 'content:Skill language:Java org:github', 'NOT is:archived language:Python OR language:go', 'repo:github/github-mcp-server'. Supports exact matching, language filters, path filters, and more.", + "type": "string" + }, + "sort": { + "description": "Sort field ('indexed' only)", + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_code" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for issues in GitHub repositories using issues search syntax already scoped to is:issue", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only issues for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub issues search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only issues for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for pull requests in GitHub repositories using issues search syntax already scoped to is:pr", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only pull requests for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub pull request search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only pull requests for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub repositories by name, description, readme, topics, or other metadata. Perfect for discovering projects, finding examples, or locating specific repositories across GitHub.", + "input_schema": { + "properties": { + "minimal_output": { + "default": true, + "description": "Return minimal repository information (default: true). When false, returns full GitHub API repository objects.", + "type": "boolean" + }, + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Repository search query. Examples: 'machine learning in:name stars:>1000 language:python', 'topic:react', 'user:facebook'. Supports advanced search syntax for precise filtering.", + "type": "string" + }, + "sort": { + "description": "Sort repositories by field, defaults to best match", + "enum": [ + "stars", + "forks", + "help-wanted-issues", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_repositories" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub users by username, real name, or other profile information. Useful for locating developers, contributors, or team members.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "User search query. Examples: 'john smith', 'location:seattle', 'followers:>100'. Search is automatically scoped to type:user.", + "type": "string" + }, + "sort": { + "description": "Sort users by number of followers or repositories, or when the person joined GitHub.", + "enum": [ + "followers", + "repositories", + "joined" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_users" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a sub-issue to a parent issue in a GitHub repository.", + "input_schema": { + "properties": { + "after_id": { + "description": "The ID of the sub-issue to be prioritized after (either after_id OR before_id should be specified)", + "type": "number" + }, + "before_id": { + "description": "The ID of the sub-issue to be prioritized before (either after_id OR before_id should be specified)", + "type": "number" + }, + "issue_number": { + "description": "The number of the parent issue", + "type": "number" + }, + "method": { + "description": "The action to perform on a single sub-issue\nOptions are:\n- 'add' - add a sub-issue to a parent issue in a GitHub repository.\n- 'remove' - remove a sub-issue from a parent issue in a GitHub repository.\n- 'reprioritize' - change the order of sub-issues within a parent issue in a GitHub repository. Use either 'after_id' or 'before_id' to specify the new position.\n\t\t\t\t", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "replace_parent": { + "description": "When true, replaces the sub-issue's current parent issue. Use with 'add' method only.", + "type": "boolean" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sub_issue_id": { + "description": "The ID of the sub-issue to add. ID is not the same as issue number", + "type": "number" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number", + "sub_issue_id" + ], + "type": "object" + }, + "name": "sub_issue_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update an existing pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "New base branch name", + "type": "string" + }, + "body": { + "description": "New description", + "type": "string" + }, + "draft": { + "description": "Mark pull request as draft (true) or ready for review (false)", + "type": "boolean" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number to update", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "reviewers": { + "description": "GitHub usernames to request reviews from", + "items": { + "type": "string" + }, + "type": "array" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "title": { + "description": "New title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update the branch of a pull request with the latest changes from the base branch.", + "input_schema": { + "properties": { + "expectedHeadSha": { + "description": "The expected SHA of the pull request's HEAD ref", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request_branch" + } + ] + }, + { + "id": "rs_0bbb974cbbc378f500691e2c841370819897cf8a919e277e6a", + "type": "reasoning", + "summary": [] + }, + { + "id": "msg_0bbb974cbbc378f500691e2c925fe4819892ab87007a8a0f3a", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Which repository do you want info for? Please provide the owner and repo name (e.g., owner/repo) or paste the GitHub URL.\n\nAlso, what details would you like? I can fetch things like:\n- Description, license, topics, default branch\n- Stars/forks/watchers\n- Latest release/tags\n- Branches and recent commits\n- Open issues/PRs and their statuses\n- Languages and file structure" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": "medium", + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "authorization": "", + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "github_copilot", + "server_url": "https://api.githubcopilot.com/mcp/" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 5312, + "input_tokens_details": { + "cached_tokens": 5248 + }, + "output_tokens": 671, + "output_tokens_details": { + "reasoning_tokens": 576 + }, + "total_tokens": 5983 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 20:46:12 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/open_ai/responses/common_format/mcp_test.rb b/test/integration/open_ai/responses/common_format/mcp_test.rb new file mode 100644 index 00000000..ad6d4cb6 --- /dev/null +++ b/test/integration/open_ai/responses/common_format/mcp_test.rb @@ -0,0 +1,189 @@ +# frozen_string_literal: true + +require_relative "../../../test_helper" + +module Integration + module OpenAI + module Responses + module CommonFormat + class McpTest < ActiveSupport::TestCase + include Integration::TestHelper + + class TestAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-5" + + # Single MCP server without authentication + COMMON_FORMAT_SINGLE_SERVER = { + model: "gpt-5", + input: "Get the current weather", + tools: [ + { + type: "mcp", + server_label: "weather", + server_url: "https://demo-day.mcp.cloudflare.com/sse" + } + ] + } + def common_format_single_server + prompt( + input: "Get the current weather", + mcp_servers: [ + { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" } + ] + ) + end + + # Single MCP server with authentication + COMMON_FORMAT_SINGLE_SERVER_WITH_AUTH = { + model: "gpt-5", + input: "Get repository information", + tools: [ + { + type: "mcp", + server_label: "github_copilot", + server_url: "https://api.githubcopilot.com/mcp/", + authorization: "GITHUB_MCP_TOKEN" + } + ] + } + def common_format_single_server_with_auth + prompt( + input: "Get repository information", + mcp_servers: [ + { name: "github_copilot", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } + ] + ) + end + + # Multiple MCP servers + COMMON_FORMAT_MULTIPLE_SERVERS = { + model: "gpt-5", + input: "Get the weather and repository information", + tools: [ + { + type: "mcp", + server_label: "weather", + server_url: "https://demo-day.mcp.cloudflare.com/sse" + }, + { + type: "mcp", + server_label: "github_copilot", + server_url: "https://api.githubcopilot.com/mcp/", + authorization: "GITHUB_MCP_TOKEN" + } + ] + } + def common_format_multiple_servers + prompt( + input: "Get the weather and repository information", + mcp_servers: [ + { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" }, + { name: "github_copilot", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } + ] + ) + end + + # MCP servers mixed with regular tools + COMMON_FORMAT_MIXED_TOOLS_AND_MCP = { + model: "gpt-5", + input: "Get the weather and calculate 5 + 3", + tools: [ + { + type: "function", + name: "calculate", + description: "Perform arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string" }, + a: { type: "number" }, + b: { type: "number" } + } + } + }, + { + type: "mcp", + server_label: "weather", + server_url: "https://demo-day.mcp.cloudflare.com/sse" + } + ] + } + def common_format_mixed_tools_and_mcp + prompt( + input: "Get the weather and calculate 5 + 3", + tools: [ + { + name: "calculate", + description: "Perform arithmetic", + parameters: { + type: "object", + properties: { + operation: { type: "string" }, + a: { type: "number" }, + b: { type: "number" } + } + } + } + ], + mcp_servers: [ + { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" } + ] + ) + end + + def calculate(operation:, a:, b:) + result = case operation + when "add" then a + b + when "subtract" then a - b + when "multiply" then a * b + when "divide" then a / b + end + { operation: operation, a: a, b: b, result: result } + end + end + + ################################################################################ + # This automatically runs all the tests for the test actions + ################################################################################ + + # Tests without sensitive tokens (can use standard test_request_builder) + [ + :common_format_single_server, + :common_format_mixed_tools_and_mcp + ].each do |action_name| + test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) + end + + # Tests with sensitive tokens need custom handling + # These tests verify the cassette recording but skip WebMock replay verification + # because VCR filters the token in the cassette but not in the live WebMock request + [ + :common_format_single_server_with_auth, + :common_format_multiple_servers + ].each do |action_name| + agent_name = TestAgent.name.demodulize.underscore + expected_body = TestAgent.const_get(action_name.to_s.upcase) + + test "#{agent_name} #{action_name} Request Building" do + cassette_name = [ self.class.name.underscore, "#{agent_name}_#{action_name}" ].join("/") + + # Run once to record response + VCR.use_cassette(cassette_name) do + TestAgent.send(action_name).generate_now + end + + # Validate that the recorded request matches our expectations (with filtered values) + cassette_file = YAML.load_file("test/fixtures/vcr_cassettes/#{cassette_name}.yml") + saved_request_body = JSON.parse(cassette_file.dig("http_interactions", 0, "request", "body", "string"), symbolize_names: true) + + assert_equal expected_body, saved_request_body + + # Note: Skipping WebMock replay verification because VCR filters tokens in cassettes + # but WebMock sees unfiltered tokens in live requests, causing mismatches + end + end + end + end + end + end +end diff --git a/test/providers/open_ai/responses/transforms_test.rb b/test/providers/open_ai/responses/transforms_test.rb index c289e903..fe3d5df0 100644 --- a/test/providers/open_ai/responses/transforms_test.rb +++ b/test/providers/open_ai/responses/transforms_test.rb @@ -384,6 +384,82 @@ def serializable.serialize assert_equal "response", hash[:format][:name] assert hash[:format][:strict] end + + # normalize_mcp_servers tests + test "normalize_mcp_servers converts common format to OpenAI format" do + servers = [ + { name: "stripe", url: "https://mcp.stripe.com", authorization: "sk_test_123" } + ] + + result = transforms.normalize_mcp_servers(servers) + + assert_equal 1, result.size + assert_equal "mcp", result[0][:type] + assert_equal "stripe", result[0][:server_label] + assert_equal "https://mcp.stripe.com", result[0][:server_url] + assert_equal "sk_test_123", result[0][:authorization] + end + + test "normalize_mcp_servers handles multiple servers" do + servers = [ + { name: "stripe", url: "https://mcp.stripe.com", authorization: "token1" }, + { name: "github", url: "https://api.githubcopilot.com/mcp/", authorization: "token2" } + ] + + result = transforms.normalize_mcp_servers(servers) + + assert_equal 2, result.size + assert_equal "stripe", result[0][:server_label] + assert_equal "github", result[1][:server_label] + end + + test "normalize_mcp_servers handles server without authorization" do + servers = [ + { name: "public", url: "https://demo.mcp.example.com" } + ] + + result = transforms.normalize_mcp_servers(servers) + + assert_equal 1, result.size + assert_equal "mcp", result[0][:type] + assert_equal "public", result[0][:server_label] + assert_equal "https://demo.mcp.example.com", result[0][:server_url] + assert_nil result[0][:authorization] + end + + test "normalize_mcp_servers returns already normalized servers as-is" do + servers = [ + { type: "mcp", server_label: "stripe", server_url: "https://mcp.stripe.com", authorization: "token" } + ] + + result = transforms.normalize_mcp_servers(servers) + + assert_equal servers, result + end + + test "normalize_mcp_servers handles alternative field names" do + servers = [ + { server_label: "stripe", server_url: "https://mcp.stripe.com", authorization: "token" } + ] + + result = transforms.normalize_mcp_servers(servers) + + assert_equal 1, result.size + assert_equal "mcp", result[0][:type] + assert_equal "stripe", result[0][:server_label] + end + + test "normalize_mcp_servers returns nil for nil input" do + result = transforms.normalize_mcp_servers(nil) + + assert_nil result + end + + test "normalize_mcp_servers returns empty array for empty array" do + result = transforms.normalize_mcp_servers([]) + + assert_equal [], result + end end end end From e1e1a32cbfcea554a851055995c175444271d9d9 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 12:50:48 -0800 Subject: [PATCH 12/17] Add Native Format MCP Tests --- .../test_agent_mcp_server.yml | 108 ++++++++++ .../test_agent_mcp_server.yml | 199 ++++++++++++++++++ .../anthropic/native_format_test.rb | 39 +++- .../open_ai/responses/native_format_test.rb | 25 +++ 4 files changed, 370 insertions(+), 1 deletion(-) create mode 100644 test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml create mode 100644 test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml b/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml new file mode 100644 index 00000000..a0c3a0cb --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml @@ -0,0 +1,108 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-sonnet-4-5-20250929","messages":[{"content":"What + tools do you have available?","role":"user"}],"max_tokens":1024,"mcp_servers":[{"name":"cloudflare-demo","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '235' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:50:18 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '2000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '2000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T20:50:17Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '400000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '400000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T20:50:18Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T20:50:15Z' + Retry-After: + - '44' + Anthropic-Ratelimit-Tokens-Limit: + - '2400000' + Anthropic-Ratelimit-Tokens-Remaining: + - '2400000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T20:50:17Z' + Request-Id: + - req_011CVHucNXpL4ZPbJzgT2Z69 + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '4865' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a1294248fef31f4-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-sonnet-4-5-20250929","id":"msg_01SoVDRqmrw4zWsNMChbZ9dZ","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to one tool:\n\n**Cloudflare MCP Demo Day Info** - This tool gets + information about Cloudflare''s MCP Demo Day. I can use it to answer questions + about Cloudflare''s MCP demo day event.\n\nIf you''d like to know more about + Cloudflare''s MCP Demo Day, I''d be happy to fetch that information for you!"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":87,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:50:18 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml new file mode 100644 index 00000000..088e263d --- /dev/null +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml @@ -0,0 +1,199 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4.1","input":"What tools do you have available?","tools":[{"type":"mcp","server_label":"cloudflare-demo","server_url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '176' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 20:50:30 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999906' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_f930de4b1fef42efbd671c8f50d718d5 + Openai-Processing-Ms: + - '6126' + X-Envoy-Upstream-Service-Time: + - '6130' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=CSKnilMCYDGbIJen9mOqPAZ7DhPigg8Pt9lbuzpb4C0-1763585430-1.0.1.1-elHDnP58qXxQEsSrnvucnenaNLCDZx7DPQ.RgpWQl00bHT8O.GMC2dbL7nYk0AcuEUiE0o3B_2_3Fw2X1jP5rCtTjCg9XwrOBh9NALSeSiM; + path=/; expires=Wed, 19-Nov-25 21:20:30 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=UCeoHiVlR_TvsF53s39eQ1irF_vsGIxk5sC3B_i1KBM-1763585430500-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a1294661dd3eb25-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_03973314476bd0d900691e2d905f88819bb815ddd4fd9ab654", + "object": "response", + "created_at": 1763585424, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-2025-04-14", + "output": [ + { + "id": "mcpl_03973314476bd0d900691e2d90b224819b91340ffbd1a64e78", + "type": "mcp_list_tools", + "server_label": "cloudflare-demo", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Get information about Cloudflare's MCP Demo Day. Use this tool if the user asks about Cloudflare's MCP demo day", + "input_schema": { + "type": "object", + "properties": {} + }, + "name": "mcp_demo_day_info" + } + ] + }, + { + "id": "msg_03973314476bd0d900691e2d922164819b80e5e14143fd8a75", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Here are the tools I currently have available:\n\n1. **Image Input Capabilities:** I can analyze and interpret images you upload.\n2. **Web Browsing (not available right now):** Sometimes, I have the ability to access the web for real-time information, but it's currently disabled.\n3. **Plugin for Cloudflare's MCP Demo Day:** \n - I can fetch information about Cloudflare\u2019s Managed Components Platform (MCP) Demo Day using a dedicated plugin.\n\nIf you need information from these areas or want to try the MCP Demo Day plugin, let me know!" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "cloudflare-demo", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 75, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 119, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 194 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 20:50:30 GMT +recorded_with: VCR 6.3.1 diff --git a/test/integration/anthropic/native_format_test.rb b/test/integration/anthropic/native_format_test.rb index 02681cda..8ea0ee34 100644 --- a/test/integration/anthropic/native_format_test.rb +++ b/test/integration/anthropic/native_format_test.rb @@ -620,6 +620,42 @@ def stop_sequences stop_sequences: [ "}" ] ) end + + ############################################################### + # Native Format MCP Server + ############################################################### + MCP_SERVER = { + model: "claude-sonnet-4-5-20250929", + messages: [ + { + role: "user", + content: "What tools do you have available?" + } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://demo-day.mcp.cloudflare.com/sse", + name: "cloudflare-demo" + } + ] + } + def mcp_server + prompt( + messages: [ + { role: "user", content: "What tools do you have available?" } + ], + max_tokens: 1024, + mcp_servers: [ + { + type: "url", + url: "https://demo-day.mcp.cloudflare.com/sse", + name: "cloudflare-demo" + } + ] + ) + end end ################################################################################ @@ -642,7 +678,8 @@ def stop_sequences :streaming, :tools_with_streaming, :sampling_parameters, - :stop_sequences + :stop_sequences, + :mcp_server ].each do |action_name| test_request_builder(TestAgent, action_name, :generate_now, TestAgent.const_get(action_name.to_s.upcase, true)) end diff --git a/test/integration/open_ai/responses/native_format_test.rb b/test/integration/open_ai/responses/native_format_test.rb index 2de74388..b89148c8 100644 --- a/test/integration/open_ai/responses/native_format_test.rb +++ b/test/integration/open_ai/responses/native_format_test.rb @@ -183,6 +183,30 @@ def get_current_weather(location:, unit: "fahrenheit") { location:, unit:, temperature: "22" } end + MCP_SERVER = { + model: "gpt-4.1", + input: "What tools do you have available?", + tools: [ + { + type: "mcp", + server_label: "cloudflare-demo", + server_url: "https://demo-day.mcp.cloudflare.com/sse" + } + ] + } + def mcp_server + prompt( + input: "What tools do you have available?", + tools: [ + { + type: "mcp", + server_label: "cloudflare-demo", + server_url: "https://demo-day.mcp.cloudflare.com/sse" + } + ] + ) + end + REASONING = { model: "o3-mini", input: "How much wood would a woodchuck chuck?", @@ -243,6 +267,7 @@ def functions_with_streaming # :file_search, :streaming, :functions, + :mcp_server, :reasoning, :functions_with_streaming ].each do |action_name| From d67d5cc2d8b3c28bc94d4854894a7ba1a7ec89b2 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 13:22:15 -0800 Subject: [PATCH 13/17] Update Docs for MCPs Common Format --- docs/.vitepress/config.mts | 1 + docs/actions.md | 6 + docs/actions/mcps.md | 88 + docs/actions/tools.md | 35 +- docs/examples/research-agent.md | 8 +- docs/providers/anthropic.md | 2 +- docs/providers/open_ai.md | 2 +- .../providers/anthropic/transforms.rb | 9 +- .../providers/open_ai/responses/request.rb | 9 +- test/docs/actions/mcps_examples_test.rb | 257 +++ test/docs/actions_examples_test.rb | 25 + .../docs/actions/mcps/anthropic_basic.yml | 110 + .../docs/actions/mcps/multiple_servers.yml | 1933 +++++++++++++++++ .../actions/mcps/native_format_anthropic.yml | 109 + .../actions/mcps/native_format_openai.yml | 1906 ++++++++++++++++ .../actions/mcps/openai_custom_servers.yml | 1906 ++++++++++++++++ .../docs/actions/mcps/quick_start_weather.yml | 112 + .../docs/actions/mcps/single_server.yml | 112 + .../docs/actions/mcps/with_function_tools.yml | 226 ++ .../docs/actions_examples/mcps.yml | 104 + .../test_agent_common_format_mixed_auth.yml | 86 +- ...t_agent_common_format_multiple_servers.yml | 91 +- ...test_agent_common_format_single_server.yml | 32 +- ..._common_format_single_server_with_auth.yml | 95 +- .../test_agent_common_format_sse_server.yml | 32 +- .../test_agent_mcp_server.yml | 32 +- ...gent_common_format_mixed_tools_and_mcp.yml | 74 +- ...t_agent_common_format_multiple_servers.yml | 42 +- ...test_agent_common_format_single_server.yml | 38 +- ..._common_format_single_server_with_auth.yml | 40 +- .../test_agent_mcp_server.yml | 32 +- .../responses/common_format/mcp_test.rb | 8 +- 32 files changed, 7214 insertions(+), 348 deletions(-) create mode 100644 docs/actions/mcps.md create mode 100644 test/docs/actions/mcps_examples_test.rb create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/anthropic_basic.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/multiple_servers.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_anthropic.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_openai.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/openai_custom_servers.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/quick_start_weather.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/single_server.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions/mcps/with_function_tools.yml create mode 100644 test/fixtures/vcr_cassettes/docs/actions_examples/mcps.yml diff --git a/docs/.vitepress/config.mts b/docs/.vitepress/config.mts index 2b878f84..98b8a317 100644 --- a/docs/.vitepress/config.mts +++ b/docs/.vitepress/config.mts @@ -116,6 +116,7 @@ export default defineConfig({ { text: 'Messages', link: '/actions/messages' }, { text: 'Embeddings', link: '/actions/embeddings' }, { text: 'Tools', link: '/actions/tools' }, + { text: 'MCPs', link: '/actions/mcps' }, { text: 'Structured Output', link: '/actions/structured_output' }, { text: 'Usage', link: '/actions/usage' }, ] diff --git a/docs/actions.md b/docs/actions.md index 22e6bc5d..c2e774d8 100644 --- a/docs/actions.md +++ b/docs/actions.md @@ -31,6 +31,12 @@ Let AI call Ruby methods during generation: <<< @/../test/docs/actions_examples_test.rb#tools_weather_agent{ruby:line-numbers} +### [MCPs](/actions/mcps) + +Connect to external services via Model Context Protocol: + +<<< @/../test/docs/actions_examples_test.rb#mcps_research_agent{ruby:line-numbers} + ### [Structured Output](/actions/structured_output) Enforce JSON responses with schemas: diff --git a/docs/actions/mcps.md b/docs/actions/mcps.md new file mode 100644 index 00000000..0ccff44e --- /dev/null +++ b/docs/actions/mcps.md @@ -0,0 +1,88 @@ +--- +title: Model Context Protocols (MCP) +description: Connect agents to external services and APIs using the Model Context Protocol. Universal integration for tools and data sources. +--- +# {{ $frontmatter.title }} + +Connect agents to external services via [Model Context Protocol](https://modelcontextprotocol.io/) servers. MCP servers expose tools and data sources that agents can use automatically. + +## Quick Start + +<<< @/../test/docs/actions/mcps_examples_test.rb#quick_start_weather_agent {ruby:line-numbers} + +## Provider Support + +| Provider | Support | Notes | +|:---------------|:-------:|:------| +| **OpenAI** | ✅ | Via Responses API | +| **Anthropic** | ⚠️ | Beta | +| **OpenRouter** | 🚧 | In development | +| **Ollama** | ❌ | Not supported | +| **Mock** | ❌ | Not supported | + +## MCP Format + +```ruby +{ + name: "server_name", # Required: server identifier + url: "https://server.url", # Required: MCP endpoint + authorization: "token" # Optional: auth token +} +``` + +### Single Server + +<<< @/../test/docs/actions/mcps_examples_test.rb#single_server_data_agent {ruby:line-numbers} + +### Multiple Servers + +<<< @/../test/docs/actions/mcps_examples_test.rb#multiple_servers_integrated_agent {ruby:line-numbers} + +### With Function Tools + +<<< @/../test/docs/actions/mcps_examples_test.rb#hybrid_agent_with_tools {ruby:line-numbers} + +## OpenAI + +OpenAI supports MCP via the Responses API with pre-built connectors and custom servers. + +### Pre-built Connectors + +<<< @/../test/docs/actions/mcps_examples_test.rb#openai_prebuilt_connectors {ruby:line-numbers} + +Available: Dropbox, Google Drive, GitHub, Slack, and more. See [OpenAI's MCP docs](https://platform.openai.com/docs/guides/mcp) for the full list. + +### Custom Servers + +<<< @/../test/docs/actions/mcps_examples_test.rb#openai_custom_servers {ruby:line-numbers} + +## Anthropic + +Anthropic supports MCP servers via the `mcp_servers` parameter (beta). Up to 20 servers per request. + +<<< @/../test/docs/actions/mcps_examples_test.rb#anthropic_basic_mcp {ruby:line-numbers} + +See [Anthropic's MCP docs](https://docs.anthropic.com/en/docs/build-with-claude/mcp) for details. + +## Native Formats + +ActiveAgent converts the common format to provider-specific formats automatically. Use native formats only if needed for provider-specific features. + +::: code-group +<<< @/../test/docs/actions/mcps_examples_test.rb#native_formats_openai {ruby:line-numbers} [OpenAI] +<<< @/../test/docs/actions/mcps_examples_test.rb#native_formats_anthropic {ruby:line-numbers} [Anthropic] +::: + +## Troubleshooting + +**Server not responding:** Verify the URL is correct and accessible from your environment. + +**Authorization failures:** Check token validity, permissions, and expiration. + +**Tools not available:** Ensure the server implements MCP correctly and returns valid tool definitions. + +## Related + +- [Tools](/actions/tools) - Function tools and tool choice +- [OpenAI Provider](/providers/open_ai) - OpenAI-specific features +- [Anthropic Provider](/providers/anthropic) - Anthropic-specific features diff --git a/docs/actions/tools.md b/docs/actions/tools.md index 0a040282..61f51435 100644 --- a/docs/actions/tools.md +++ b/docs/actions/tools.md @@ -17,13 +17,15 @@ The LLM calls `get_weather` automatically when it needs weather data, and uses t ## Provider Support Matrix -| Provider | Functions | Server-side Tools | MCP Support | Notes | -|:---------------|:---------:|:-----------------:|:-----------:|:------| -| **OpenAI** | 🟩 | 🟩 | 🟩 | Server-side tools and MCP require Responses API | -| **Anthropic** | 🟩 | 🟩 | 🟨 | MCP in beta | -| **OpenRouter** | 🟩 | ❌ | 🟦 | MCP via converted tool definitions; model-dependent capabilities | -| **Ollama** | 🟩 | ❌ | ❌ | Model-dependent capabilities | -| **Mock** | 🟦 | ❌ | ❌ | Accepted but not enforced | +| Provider | Functions | Server-side Tools | Notes | +|:---------------|:---------:|:-----------------:|:------| +| **OpenAI** | 🟩 | 🟩 | Server-side tools require Responses API | +| **Anthropic** | 🟩 | 🟩 | Full support for built-in tools | +| **OpenRouter** | 🟩 | ❌ | Model-dependent capabilities | +| **Ollama** | 🟩 | ❌ | Model-dependent capabilities | +| **Mock** | 🟦 | ❌ | Accepted but not enforced | + +For **MCP (Model Context Protocol)** support, see the [MCP documentation](/actions/mcps). ## Functions @@ -131,24 +133,6 @@ OpenAI's **Responses API** provides several built-in tools (requires GPT-5, GPT- Anthropic provides web access and specialized capabilities including Web Search for real-time information, Web Fetch (Beta) for specific URLs, Extended Thinking to show reasoning processes, and Computer Use (Beta) for interface interaction. For complete details and examples, see [Anthropic's tool use documentation](https://docs.claude.com/en/docs/agents-and-tools/tool-use/overview). -## Model Context Protocol (MCP) - -MCP (Model Context Protocol) enables agents to connect to external services and APIs. Think of it as a universal adapter for integrating tools and data sources. - -### OpenAI MCP Integration - -OpenAI supports MCP through their Responses API in two ways: pre-built connectors for popular services (Dropbox, Google Drive, GitHub, Slack, and more) and custom MCP servers. For complete details on OpenAI's MCP support, connector IDs, and configuration options, see [OpenAI's MCP documentation](https://platform.openai.com/docs/guides/mcp). - -### Anthropic MCP Integration - -Anthropic supports MCP servers via the `mcp_servers` parameter (beta feature). You can connect up to 20 MCP servers per request. For the latest on Anthropic's MCP implementation and configuration, see [Anthropic's MCP documentation](https://docs.anthropic.com/en/docs/build-with-claude/mcp). - -### OpenRouter MCP Integration - -::: info Coming Soon -MCP support for OpenRouter is currently under development and will be available in a future release. -::: - ## Troubleshooting ### Tool Not Being Called @@ -161,6 +145,7 @@ If the LLM passes unexpected parameters, add detailed parameter descriptions wit ## Related Documentation +- [MCP (Model Context Protocol)](/actions/mcps) - Connect to external services via MCP - [Agents](/agents) - Understand the agent lifecycle and callbacks - [Generation](/agents/generation) - Execute tool-enabled generations - [Messages](/actions/messages) - Learn about conversation structure diff --git a/docs/examples/research-agent.md b/docs/examples/research-agent.md index a7984633..3f33d9c7 100644 --- a/docs/examples/research-agent.md +++ b/docs/examples/research-agent.md @@ -43,7 +43,7 @@ class ResearchAgent < ApplicationAgent # Configure research tools at the class level configure_research_tools( enable_web_search: true, - mcp_servers: ["arxiv", "github"], + mcps: ["arxiv", "github"], default_search_context: "high" ) @@ -253,7 +253,7 @@ Configure default research settings: class ResearchAgent < ApplicationAgent configure_research_tools( enable_web_search: true, - mcp_servers: ["arxiv", "github", "pubmed"], + mcps: ["arxiv", "github", "pubmed"], default_search_context: "high", enable_visualizations: true ) @@ -399,7 +399,7 @@ class ResearchAgent < ApplicationAgent configure_research_tools( enable_web_search: true, - mcp_servers: ["arxiv"] + mcps: ["arxiv"] ) end @@ -408,7 +408,7 @@ class AcademicAgent < ApplicationAgent configure_research_tools( enable_web_search: false, - mcp_servers: ["arxiv", "pubmed"] + mcps: ["arxiv", "pubmed"] ) end ``` diff --git a/docs/providers/anthropic.md b/docs/providers/anthropic.md index c1d82e5c..c5698fbd 100644 --- a/docs/providers/anthropic.md +++ b/docs/providers/anthropic.md @@ -107,7 +107,7 @@ Anthropic provides access to the Claude model family. For the complete list of a - **`thinking`** - Enable Claude's thinking mode for complex reasoning - **`context_management`** - Configure context window management - **`service_tier`** - Select service tier ("auto", "standard_only") -- **`mcp_servers`** - Array of MCP server definitions (max 20) +- **`mcps`** - Array of MCP server definitions (max 20) ### Client Configuration diff --git a/docs/providers/open_ai.md b/docs/providers/open_ai.md index ce88dd34..2033bff8 100644 --- a/docs/providers/open_ai.md +++ b/docs/providers/open_ai.md @@ -120,7 +120,7 @@ Search-preview models in Chat API provide web search but with different configur - `mcp` - Enable MCP integration - **`tool_choice`** - Control tool usage ("auto", "required", "none", or specific tool) - **`parallel_tool_calls`** - Allow parallel tool execution (boolean) -- **`mcp_servers`** - Array of MCP server configurations (max 20) +- **`mcps`** - Array of MCP server configurations (max 20) ### Embeddings diff --git a/lib/active_agent/providers/anthropic/transforms.rb b/lib/active_agent/providers/anthropic/transforms.rb index 8214f845..a2b2aead 100644 --- a/lib/active_agent/providers/anthropic/transforms.rb +++ b/lib/active_agent/providers/anthropic/transforms.rb @@ -30,7 +30,14 @@ def normalize_params(params) params[:system] = normalize_system(params[:system]) if params[:system] params[:tools] = normalize_tools(params[:tools]) if params[:tools] params[:tool_choice] = normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] - params[:mcp_servers] = normalize_mcp_servers(params[:mcp_servers]) if params[:mcp_servers] + + # Handle mcps parameter (common format) -> transforms to mcp_servers (provider format) + if params[:mcps] + params[:mcp_servers] = normalize_mcp_servers(params.delete(:mcps)) + elsif params[:mcp_servers] + params[:mcp_servers] = normalize_mcp_servers(params[:mcp_servers]) + end + params end diff --git a/lib/active_agent/providers/open_ai/responses/request.rb b/lib/active_agent/providers/open_ai/responses/request.rb index dd5cdf57..4566c83d 100644 --- a/lib/active_agent/providers/open_ai/responses/request.rb +++ b/lib/active_agent/providers/open_ai/responses/request.rb @@ -77,10 +77,13 @@ def initialize(**params) params[:tools] = Responses::Transforms.normalize_tools(params[:tools]) if params[:tools] params[:tool_choice] = Responses::Transforms.normalize_tool_choice(params[:tool_choice]) if params[:tool_choice] - # Step 8: Normalize MCP servers from common format + # Step 8: Normalize MCP servers from common format (mcps parameter) # OpenAI treats MCP servers as a special type of tool in the tools array - if params[:mcp_servers]&.any? - normalized_mcp_tools = Responses::Transforms.normalize_mcp_servers(params.delete(:mcp_servers)) + mcp_param = params[:mcps] || params[:mcp_servers] + if mcp_param&.any? + normalized_mcp_tools = Responses::Transforms.normalize_mcp_servers(mcp_param) + params.delete(:mcps) + params.delete(:mcp_servers) # Merge MCP servers into tools array params[:tools] = (params[:tools] || []) + normalized_mcp_tools end diff --git a/test/docs/actions/mcps_examples_test.rb b/test/docs/actions/mcps_examples_test.rb new file mode 100644 index 00000000..2a4d959b --- /dev/null +++ b/test/docs/actions/mcps_examples_test.rb @@ -0,0 +1,257 @@ +require "test_helper" + +module Docs + module Actions + class McpsExamplesTest < ActiveSupport::TestCase + class QuickStartExample < ActiveSupport::TestCase + # region quick_start_weather_agent + class WeatherAgent < ActiveAgent::Base + generate_with :anthropic, model: "claude-haiku-4-5" + + def forecast + prompt( + message: "What's the weather like?", + mcps: [ { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" } ] + ) + end + end + # endregion quick_start_weather_agent + + test "quick start weather agent with mcps" do + VCR.use_cassette("docs/actions/mcps/quick_start_weather") do + response = WeatherAgent.forecast.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + + class SingleServerExample < ActiveSupport::TestCase + # region single_server_data_agent + class DataAgent < ActiveAgent::Base + generate_with :anthropic, model: "claude-haiku-4-5" + + def analyze + prompt( + message: "Analyze the latest data", + mcps: [ { name: "cloudflare-demo", url: "https://demo-day.mcp.cloudflare.com/sse" } ] + ) + end + end + # endregion single_server_data_agent + + test "single server MCP connection" do + VCR.use_cassette("docs/actions/mcps/single_server") do + response = DataAgent.analyze.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + + class MultipleServersExample < ActiveSupport::TestCase + # region multiple_servers_integrated_agent + class IntegratedAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4o" + + def research + prompt( + input: "Research the latest AI developments", + mcps: [ + { name: "cloudflare", url: "https://demo-day.mcp.cloudflare.com/sse" }, + { name: "github", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } + ] + ) + end + end + # endregion multiple_servers_integrated_agent + + test "multiple MCP servers connection" do + VCR.use_cassette("docs/actions/mcps/multiple_servers") do + response = IntegratedAgent.research.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + + class WithFunctionToolsExample < ActiveSupport::TestCase + # region hybrid_agent_with_tools + class HybridAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4o" + + def analyze_data + prompt( + input: "Calculate and fetch data", + tools: [ { + name: "calculate", + description: "Perform calculations", + parameters: { + type: "object", + properties: { + operation: { type: "string" }, + a: { type: "number" }, + b: { type: "number" } + } + } + } ], + mcps: [ { name: "data-service", url: "https://demo-day.mcp.cloudflare.com/sse" } ] + ) + end + + def calculate(operation:, a:, b:) + case operation + when "add" then a + b + when "subtract" then a - b + end + end + end + # endregion hybrid_agent_with_tools + + test "MCP with function tools" do + VCR.use_cassette("docs/actions/mcps/with_function_tools") do + response = HybridAgent.analyze_data.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + + class OpenAIPrebuiltConnectorsExample < ActiveSupport::TestCase + # region openai_prebuilt_connectors + class FileAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4o" + + def search_files + prompt( + input: "Find documents about Q4 revenue", + mcps: [ { name: "dropbox", url: "mcp://dropbox" } ] # Pre-built connector + ) + end + end + # endregion openai_prebuilt_connectors + + test "OpenAI pre-built connectors" do + skip "Pre-built connectors require real OAuth tokens" + # This example is for documentation purposes only + # Real testing would require actual Dropbox OAuth setup + end + end + + class OpenAICustomServersExample < ActiveSupport::TestCase + # region openai_custom_servers + class CustomAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4o" + + def custom_tools + prompt( + input: "Use custom tools", + mcps: [ { name: "github_copilot", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } ] + ) + end + end + # endregion openai_custom_servers + + test "OpenAI custom MCP servers" do + VCR.use_cassette("docs/actions/mcps/openai_custom_servers") do + response = CustomAgent.custom_tools.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + + class AnthropicBasicExample < ActiveSupport::TestCase + # region anthropic_basic_mcp + class ClaudeAgent < ActiveAgent::Base + generate_with :anthropic, model: "claude-haiku-4-5" + + def use_mcp + prompt( + message: "What tools are available?", + mcps: [ { name: "demo-server", url: "https://demo-day.mcp.cloudflare.com/sse" } ] + ) + end + end + # endregion anthropic_basic_mcp + + test "Anthropic basic MCP usage" do + VCR.use_cassette("docs/actions/mcps/anthropic_basic") do + response = ClaudeAgent.use_mcp.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + + class NativeFormatsExample < ActiveSupport::TestCase + # region native_formats_openai + class OpenAINativeAgent < ActiveAgent::Base + generate_with :openai, model: "gpt-4o" + + def native_format + prompt( + input: "What can you do?", + tools: [ { + type: "mcp", + server_label: "github", + server_url: "https://api.githubcopilot.com/mcp/", + authorization: ENV["GITHUB_MCP_TOKEN"] + } ] + ) + end + end + # endregion native_formats_openai + + # region native_formats_anthropic + class AnthropicNativeAgent < ActiveAgent::Base + generate_with :anthropic, model: "claude-haiku-4-5" + + def native_format + prompt( + message: "What can you do?", + mcp_servers: [ { + type: "url", + name: "cloudflare", + url: "https://demo-day.mcp.cloudflare.com/sse" + } ] + ) + end + end + # endregion native_formats_anthropic + + test "native format OpenAI" do + VCR.use_cassette("docs/actions/mcps/native_format_openai") do + response = OpenAINativeAgent.native_format.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + + test "native format Anthropic" do + VCR.use_cassette("docs/actions/mcps/native_format_anthropic") do + response = AnthropicNativeAgent.native_format.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + end + end +end diff --git a/test/docs/actions_examples_test.rb b/test/docs/actions_examples_test.rb index f9d0b590..ad439fdc 100644 --- a/test/docs/actions_examples_test.rb +++ b/test/docs/actions_examples_test.rb @@ -92,6 +92,31 @@ def get_current_weather(location:, unit: "fahrenheit") end end + class McpsExample < ActiveSupport::TestCase + # region mcps_research_agent + class ResearchAgent < ApplicationAgent + generate_with :anthropic, model: "claude-haiku-4-5" + + def research + prompt( + message: "Research AI developments", + mcps: [ { name: "github", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } ] + ) + end + end + # endregion mcps_research_agent + + test "MCP connection" do + VCR.use_cassette("docs/actions_examples/mcps") do + response = ResearchAgent.research.generate_now + + assert response.message.content.present? + + doc_example_output(response) + end + end + end + # region structured_output_extract class DataExtractionAgent < ApplicationAgent generate_with :openai, model: "gpt-4o" diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/anthropic_basic.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/anthropic_basic.yml new file mode 100644 index 00000000..a36ba591 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/anthropic_basic.yml @@ -0,0 +1,110 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","messages":[{"content":"What tools are + available?","role":"user"}],"max_tokens":4096,"mcp_servers":[{"name":"demo-server","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '213' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:18:51 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T21:18:50Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T21:18:51Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T21:18:50Z' + Retry-After: + - '10' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T21:18:50Z' + Request-Id: + - req_011CVHwnnL48nmgHJDW47Snx + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '3100' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a12be02bb591f2f-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01EAYgadXov5fnUqr13pDWuT","type":"message","role":"assistant","content":[{"type":"text","text":"Based + on the tools I have access to, here''s what''s available:\n\n1. **demo-server_mcp_demo_day_info** + - Get information about Cloudflare''s MCP Demo Day\n - This tool allows + you to retrieve details about Cloudflare''s MCP (Model Context Protocol) demo + day event\n - No parameters required\n\nThat''s the tool currently available + in my environment. If you''d like to learn more about Cloudflare''s MCP Demo + Day, I can fetch that information for you!"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":581,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":117,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 21:18:52 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/multiple_servers.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/multiple_servers.yml new file mode 100644 index 00000000..d69c8d1d --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/multiple_servers.yml @@ -0,0 +1,1933 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4o","input":"Research the latest AI developments","tools":[{"type":"mcp","server_label":"cloudflare","server_url":"https://demo-day.mcp.cloudflare.com/sse"},{"type":"mcp","server_label":"github","server_url":"https://api.githubcopilot.com/mcp/","authorization":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '373' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:19:03 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29994688' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 10ms + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_098e4aaa1133481fb16e82940c2d1558 + Openai-Processing-Ms: + - '11336' + X-Envoy-Upstream-Service-Time: + - '11339' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=VjjuHDR.PNUCn8rM14xq7LsIIVAUuX1qazRUHp7vdKo-1763587143-1.0.1.1-6rMSTzbv6J_dtUYCiXPVoNUnqfa504.fh1huG7a8Z9qbhBG6ZBWlJpJ_osIb5hk9tpEVzq_wsR2wBTOqFGGQRaNXhB4Pcelr8f7oy8MPkOY; + path=/; expires=Wed, 19-Nov-25 21:49:03 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=l5FXBHKyzefvMOtNNobFrdBv5jST10aXOjtIkKwmAJ8-1763587143496-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a12be17982a1986-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_06eb76a2b12d8f5200691e343c273c819b89b7c2e558d3d3df", + "object": "response", + "created_at": 1763587132, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4o-2024-08-06", + "output": [ + { + "id": "mcpl_06eb76a2b12d8f5200691e343c72e8819b88e9e5b814a0a925", + "type": "mcp_list_tools", + "server_label": "cloudflare", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Get information about Cloudflare's MCP Demo Day. Use this tool if the user asks about Cloudflare's MCP demo day", + "input_schema": { + "type": "object", + "properties": {} + }, + "name": "mcp_demo_day_info" + } + ] + }, + { + "id": "mcpl_06eb76a2b12d8f5200691e343c738c819b8ff83ca0c45ebd6f", + "type": "mcp_list_tools", + "server_label": "github", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Add review comment to the requester's latest pending pull request review. A pending review needs to already exist to call this (check with the user if not sure).", + "input_schema": { + "properties": { + "body": { + "description": "The text of the review comment", + "type": "string" + }, + "line": { + "description": "The line of the blob in the pull request diff that the comment applies to. For multi-line comments, the last line of the range", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "path": { + "description": "The relative path to the file that necessitates a comment", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "side": { + "description": "The side of the diff to comment on. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "startLine": { + "description": "For multi-line comments, the first line of the range that the comment applies to", + "type": "number" + }, + "startSide": { + "description": "For multi-line comments, the starting side of the diff that the comment applies to. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "subjectType": { + "description": "The level at which the comment is targeted", + "enum": [ + "FILE", + "LINE" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber", + "path", + "body", + "subjectType" + ], + "type": "object" + }, + "name": "add_comment_to_pending_review" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a comment to a specific issue in a GitHub repository. Use this tool to add comments to pull requests as well (in this case pass pull request number as issue_number), but only if user is not asking specifically to add review comments.", + "input_schema": { + "properties": { + "body": { + "description": "Comment content", + "type": "string" + }, + "issue_number": { + "description": "Issue number to comment on", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issue_number", + "body" + ], + "type": "object" + }, + "name": "add_issue_comment" + }, + { + "annotations": { + "read_only": false + }, + "description": "Assign Copilot to a specific issue in a GitHub repository.\n\nThis tool can help with the following outcomes:\n- a Pull Request created with source code changes to resolve the issue\n\n\nMore information can be found at:\n- https://docs.github.com/en/copilot/using-github-copilot/using-copilot-coding-agent-to-work-on-tasks/about-assigning-tasks-to-copilot\n", + "input_schema": { + "properties": { + "issueNumber": { + "description": "Issue number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issueNumber" + ], + "type": "object" + }, + "name": "assign_copilot_to_issue" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new branch in a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Name for new branch", + "type": "string" + }, + "from_branch": { + "description": "Source branch (defaults to repo default)", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch" + ], + "type": "object" + }, + "name": "create_branch" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create or update a single file in a GitHub repository. If updating, you must provide the SHA of the file you want to update. Use this tool to create or update a file in a GitHub repository remotely; do not use it for local file operations.", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to create/update the file in", + "type": "string" + }, + "content": { + "description": "Content of the file", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path where to create/update the file", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Required if updating an existing file. The blob SHA of the file being replaced.", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "content", + "message", + "branch" + ], + "type": "object" + }, + "name": "create_or_update_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "Branch to merge into", + "type": "string" + }, + "body": { + "description": "PR description", + "type": "string" + }, + "draft": { + "description": "Create as draft PR", + "type": "boolean" + }, + "head": { + "description": "Branch containing changes", + "type": "string" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "title": { + "description": "PR title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "title", + "head", + "base" + ], + "type": "object" + }, + "name": "create_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new GitHub repository in your account or specified organization", + "input_schema": { + "properties": { + "autoInit": { + "description": "Initialize with README", + "type": "boolean" + }, + "description": { + "description": "Repository description", + "type": "string" + }, + "name": { + "description": "Repository name", + "type": "string" + }, + "organization": { + "description": "Organization to create the repository in (omit to create in your personal account)", + "type": "string" + }, + "private": { + "description": "Whether repo should be private", + "type": "boolean" + } + }, + "required": [ + "name" + ], + "type": "object" + }, + "name": "create_repository" + }, + { + "annotations": { + "read_only": false + }, + "description": "Delete a file from a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to delete the file from", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path to the file to delete", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "message", + "branch" + ], + "type": "object" + }, + "name": "delete_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Fork a GitHub repository to your account or specified organization", + "input_schema": { + "properties": { + "organization": { + "description": "Organization to fork to", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "fork_repository" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details for a commit from a GitHub repository", + "input_schema": { + "properties": { + "include_diff": { + "default": true, + "description": "Whether to include file diffs and stats in the response. Default is true.", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch name, or tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "sha" + ], + "type": "object" + }, + "name": "get_commit" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the contents of a file or directory from a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "default": "/", + "description": "Path to file/directory (directories must end with a slash '/')", + "type": "string" + }, + "ref": { + "description": "Accepts optional git refs such as `refs/tags/{tag}`, `refs/heads/{branch}` or `refs/pull/{pr_number}/head`", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Accepts optional commit SHA. If specified, it will be used instead of ref", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_file_contents" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific label from a repository.", + "input_schema": { + "properties": { + "name": { + "description": "Label name.", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization name)", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "name" + ], + "type": "object" + }, + "name": "get_label" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the latest release in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_latest_release" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the authenticated GitHub user. Use this when a request is about the user's own profile for GitHub. Or when information is missing to build other tool calls.", + "input_schema": { + "properties": {}, + "type": "object" + }, + "name": "get_me" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific release by its tag name in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name (e.g., 'v1.0.0')", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_release_by_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details about a specific git tag in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get member usernames of a specific team in an organization. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "org": { + "description": "Organization login (owner) that contains the team.", + "type": "string" + }, + "team_slug": { + "description": "Team slug", + "type": "string" + } + }, + "required": [ + "org", + "team_slug" + ], + "type": "object" + }, + "name": "get_team_members" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the teams the user is a member of. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "user": { + "description": "Username to get teams for. If not provided, uses the authenticated user.", + "type": "string" + } + }, + "type": "object" + }, + "name": "get_teams" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information about a specific issue in a GitHub repository.", + "input_schema": { + "properties": { + "issue_number": { + "description": "The number of the issue", + "type": "number" + }, + "method": { + "description": "The read operation to perform on a single issue. \nOptions are: \n1. get - Get details of a specific issue.\n2. get_comments - Get issue comments.\n3. get_sub_issues - Get sub-issues of the issue.\n4. get_labels - Get labels assigned to the issue.\n", + "enum": [ + "get", + "get_comments", + "get_sub_issues", + "get_labels" + ], + "type": "string" + }, + "owner": { + "description": "The owner of the repository", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "The name of the repository", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number" + ], + "type": "object" + }, + "name": "issue_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new or update an existing issue in a GitHub repository.", + "input_schema": { + "properties": { + "assignees": { + "description": "Usernames to assign to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "body": { + "description": "Issue body content", + "type": "string" + }, + "duplicate_of": { + "description": "Issue number that this issue is a duplicate of. Only used when state_reason is 'duplicate'.", + "type": "number" + }, + "issue_number": { + "description": "Issue number to update", + "type": "number" + }, + "labels": { + "description": "Labels to apply to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "method": { + "description": "Write operation to perform on a single issue.\nOptions are: \n- 'create' - creates a new issue. \n- 'update' - updates an existing issue.\n", + "enum": [ + "create", + "update" + ], + "type": "string" + }, + "milestone": { + "description": "Milestone number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "state_reason": { + "description": "Reason for the state change. Ignored unless state is changed.", + "enum": [ + "completed", + "not_planned", + "duplicate" + ], + "type": "string" + }, + "title": { + "description": "Issue title", + "type": "string" + }, + "type": { + "description": "Type of this issue. Only use if the repository has issue types configured. Use list_issue_types tool to get valid type values for the organization. If the repository doesn't support issue types, omit this parameter.", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo" + ], + "type": "object" + }, + "name": "issue_write" + }, + { + "annotations": { + "read_only": true + }, + "description": "List branches in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_branches" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get list of commits of a branch in a GitHub repository. Returns at least 30 results per page by default, but can return more if specified using the perPage parameter (up to 100).", + "input_schema": { + "properties": { + "author": { + "description": "Author username or email address to filter commits by", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch or tag name to list commits of. If not provided, uses the default branch of the repository. If a commit SHA is provided, will list commits up to that SHA.", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_commits" + }, + { + "annotations": { + "read_only": true + }, + "description": "List supported issue types for repository owner (organization).", + "input_schema": { + "properties": { + "owner": { + "description": "The organization owner of the repository", + "type": "string" + } + }, + "required": [ + "owner" + ], + "type": "object" + }, + "name": "list_issue_types" + }, + { + "annotations": { + "read_only": true + }, + "description": "List issues in a GitHub repository. For pagination, use the 'endCursor' from the previous response's 'pageInfo' in the 'after' parameter.", + "input_schema": { + "properties": { + "after": { + "description": "Cursor for pagination. Use the endCursor from the previous page's PageInfo for GraphQL APIs.", + "type": "string" + }, + "direction": { + "description": "Order direction. If provided, the 'orderBy' also needs to be provided.", + "enum": [ + "ASC", + "DESC" + ], + "type": "string" + }, + "labels": { + "description": "Filter by labels", + "items": { + "type": "string" + }, + "type": "array" + }, + "orderBy": { + "description": "Order issues by field. If provided, the 'direction' also needs to be provided.", + "enum": [ + "CREATED_AT", + "UPDATED_AT", + "COMMENTS" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "since": { + "description": "Filter by date (ISO 8601 timestamp)", + "type": "string" + }, + "state": { + "description": "Filter by state, by default both open and closed issues are returned when not provided", + "enum": [ + "OPEN", + "CLOSED" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "List pull requests in a GitHub repository. If the user specifies an author, then DO NOT use this tool and use the search_pull_requests tool instead.", + "input_schema": { + "properties": { + "base": { + "description": "Filter by base branch", + "type": "string" + }, + "direction": { + "description": "Sort direction", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "head": { + "description": "Filter by head user/org and branch", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sort": { + "description": "Sort by", + "enum": [ + "created", + "updated", + "popularity", + "long-running" + ], + "type": "string" + }, + "state": { + "description": "Filter by state", + "enum": [ + "open", + "closed", + "all" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "List releases in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_releases" + }, + { + "annotations": { + "read_only": true + }, + "description": "List git tags in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_tags" + }, + { + "annotations": { + "read_only": false + }, + "description": "Merge a pull request in a GitHub repository.", + "input_schema": { + "properties": { + "commit_message": { + "description": "Extra detail for merge commit", + "type": "string" + }, + "commit_title": { + "description": "Title for merge commit", + "type": "string" + }, + "merge_method": { + "description": "Merge method", + "enum": [ + "merge", + "squash", + "rebase" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "merge_pull_request" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information on a specific pull request in GitHub repository.", + "input_schema": { + "properties": { + "method": { + "description": "Action to specify what pull request data needs to be retrieved from GitHub. \nPossible options: \n 1. get - Get details of a specific pull request.\n 2. get_diff - Get the diff of a pull request.\n 3. get_status - Get status of a head commit in a pull request. This reflects status of builds and checks.\n 4. get_files - Get the list of files changed in a pull request. Use with pagination parameters to control the number of results returned.\n 5. get_review_comments - Get the review comments on a pull request. They are comments made on a portion of the unified diff during a pull request review. Use with pagination parameters to control the number of results returned.\n 6. get_reviews - Get the reviews on a pull request. When asked for review comments, use get_review_comments method.\n 7. get_comments - Get comments on a pull request. Use this if user doesn't specifically want review comments. Use with pagination parameters to control the number of results returned.\n", + "enum": [ + "get", + "get_diff", + "get_status", + "get_files", + "get_review_comments", + "get_reviews", + "get_comments" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create and/or submit, delete review of a pull request.\n\nAvailable methods:\n- create: Create a new review of a pull request. If \"event\" parameter is provided, the review is submitted. If \"event\" is omitted, a pending review is created.\n- submit_pending: Submit an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request. The \"body\" and \"event\" parameters are used when submitting the review.\n- delete_pending: Delete an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request.\n", + "input_schema": { + "properties": { + "body": { + "description": "Review comment text", + "type": "string" + }, + "commitID": { + "description": "SHA of commit to review", + "type": "string" + }, + "event": { + "description": "Review action to perform.", + "enum": [ + "APPROVE", + "REQUEST_CHANGES", + "COMMENT" + ], + "type": "string" + }, + "method": { + "description": "The write operation to perform on pull request review.", + "enum": [ + "create", + "submit_pending", + "delete_pending" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_review_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Push multiple files to a GitHub repository in a single commit", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to push to", + "type": "string" + }, + "files": { + "description": "Array of file objects to push, each object with path (string) and content (string)", + "items": { + "additionalProperties": false, + "properties": { + "content": { + "description": "file content", + "type": "string" + }, + "path": { + "description": "path to the file", + "type": "string" + } + }, + "required": [ + "path", + "content" + ], + "type": "object" + }, + "type": "array" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch", + "files", + "message" + ], + "type": "object" + }, + "name": "push_files" + }, + { + "annotations": { + "read_only": false + }, + "description": "Request a GitHub Copilot code review for a pull request. Use this for automated feedback on pull requests, usually before requesting a human reviewer.", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "request_copilot_review" + }, + { + "annotations": { + "read_only": true + }, + "description": "Fast and precise code search across ALL GitHub repositories using GitHub's native search engine. Best for finding exact symbols, functions, classes, or specific code patterns.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order for results", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub's powerful code search syntax. Examples: 'content:Skill language:Java org:github', 'NOT is:archived language:Python OR language:go', 'repo:github/github-mcp-server'. Supports exact matching, language filters, path filters, and more.", + "type": "string" + }, + "sort": { + "description": "Sort field ('indexed' only)", + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_code" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for issues in GitHub repositories using issues search syntax already scoped to is:issue", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only issues for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub issues search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only issues for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for pull requests in GitHub repositories using issues search syntax already scoped to is:pr", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only pull requests for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub pull request search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only pull requests for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub repositories by name, description, readme, topics, or other metadata. Perfect for discovering projects, finding examples, or locating specific repositories across GitHub.", + "input_schema": { + "properties": { + "minimal_output": { + "default": true, + "description": "Return minimal repository information (default: true). When false, returns full GitHub API repository objects.", + "type": "boolean" + }, + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Repository search query. Examples: 'machine learning in:name stars:>1000 language:python', 'topic:react', 'user:facebook'. Supports advanced search syntax for precise filtering.", + "type": "string" + }, + "sort": { + "description": "Sort repositories by field, defaults to best match", + "enum": [ + "stars", + "forks", + "help-wanted-issues", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_repositories" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub users by username, real name, or other profile information. Useful for locating developers, contributors, or team members.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "User search query. Examples: 'john smith', 'location:seattle', 'followers:>100'. Search is automatically scoped to type:user.", + "type": "string" + }, + "sort": { + "description": "Sort users by number of followers or repositories, or when the person joined GitHub.", + "enum": [ + "followers", + "repositories", + "joined" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_users" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a sub-issue to a parent issue in a GitHub repository.", + "input_schema": { + "properties": { + "after_id": { + "description": "The ID of the sub-issue to be prioritized after (either after_id OR before_id should be specified)", + "type": "number" + }, + "before_id": { + "description": "The ID of the sub-issue to be prioritized before (either after_id OR before_id should be specified)", + "type": "number" + }, + "issue_number": { + "description": "The number of the parent issue", + "type": "number" + }, + "method": { + "description": "The action to perform on a single sub-issue\nOptions are:\n- 'add' - add a sub-issue to a parent issue in a GitHub repository.\n- 'remove' - remove a sub-issue from a parent issue in a GitHub repository.\n- 'reprioritize' - change the order of sub-issues within a parent issue in a GitHub repository. Use either 'after_id' or 'before_id' to specify the new position.\n\t\t\t\t", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "replace_parent": { + "description": "When true, replaces the sub-issue's current parent issue. Use with 'add' method only.", + "type": "boolean" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sub_issue_id": { + "description": "The ID of the sub-issue to add. ID is not the same as issue number", + "type": "number" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number", + "sub_issue_id" + ], + "type": "object" + }, + "name": "sub_issue_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update an existing pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "New base branch name", + "type": "string" + }, + "body": { + "description": "New description", + "type": "string" + }, + "draft": { + "description": "Mark pull request as draft (true) or ready for review (false)", + "type": "boolean" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number to update", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "reviewers": { + "description": "GitHub usernames to request reviews from", + "items": { + "type": "string" + }, + "type": "array" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "title": { + "description": "New title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update the branch of a pull request with the latest changes from the base branch.", + "input_schema": { + "properties": { + "expectedHeadSha": { + "description": "The expected SHA of the pull request's HEAD ref", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request_branch" + } + ] + }, + { + "id": "msg_06eb76a2b12d8f5200691e343fd2e4819bb70ba9c33446de71", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Here are some of the latest developments in AI:\n\n1. **Advancements in Generative AI**: Techniques like Generative Adversarial Networks (GANs) and diffusion models are producing high-quality images, videos, and audio. Applications in content creation and digital art are expanding rapidly.\n\n2. **AI in Healthcare**: AI is revolutionizing diagnostics, drug discovery, and personalized medicine. Machine learning models are aiding in the early detection of diseases and optimizing treatment plans.\n\n3. **Natural Language Processing (NLP)**: Large language models, such as OpenAI's GPT-3 and GPT-4, continue to advance, improving the ability of machines to understand and generate human-like text. This is impacting customer service, content generation, and more.\n\n4. **AI Ethics and Regulation**: As AI systems become more widespread, there is a growing focus on ethical considerations, including bias, privacy, and transparency. Governments and organizations are working on developing regulations and guidelines.\n\n5. **AI in Autonomous Vehicles**: AI is being integrated into self-driving cars, drones, and other autonomous systems, enhancing navigation, safety, and efficiency.\n\n6. **AI and Quantum Computing**: Research is being conducted on leveraging quantum computing for AI to solve complex problems more efficiently than classical computers.\n\n7. **Computer Vision**: AI models are becoming better at image and video recognition, with applications ranging from security and surveillance to agriculture and industrial inspection.\n\n8. **AI in Finance**: AI is increasingly used for fraud detection, personalized banking, and algorithmic trading, optimizing various financial services.\n\nThese developments represent just a snapshot of the rapidly evolving AI landscape, which continues to impact numerous industries and daily life." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "cloudflare", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + }, + { + "type": "mcp", + "allowed_tools": null, + "authorization": "", + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "github", + "server_url": "https://api.githubcopilot.com/mcp/" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 5293, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 343, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 5636 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 21:19:03 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_anthropic.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_anthropic.yml new file mode 100644 index 00000000..2d0bad85 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_anthropic.yml @@ -0,0 +1,109 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","messages":[{"content":"What can you do?","role":"user"}],"mcp_servers":[{"name":"cloudflare","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}],"max_tokens":4096}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '203' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:18:48 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T21:18:47Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T21:18:48Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T21:18:47Z' + Retry-After: + - '13' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T21:18:47Z' + Request-Id: + - req_011CVHwnXBEM3NrXrHRmWjgL + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '3401' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a12bdec7c19270c-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_011Hz8oWUNyrUP8CRsmtn8eM","type":"message","role":"assistant","content":[{"type":"text","text":"I + can help you with information about **Cloudflare''s MCP Demo Day**! \n\nIf + you have any questions about:\n- What Cloudflare''s MCP Demo Day is\n- When + and where it''s happening\n- What will be showcased\n- How to participate + or attend\n- Any other details about the event\n\nJust ask me, and I''ll retrieve + the information for you using the tools available to me.\n\nIs there anything + specific about Cloudflare''s MCP Demo Day you''d like to know?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":581,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":115,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 21:18:48 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_openai.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_openai.yml new file mode 100644 index 00000000..a5937502 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/native_format_openai.yml @@ -0,0 +1,1906 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4o","input":"What can you do?","tools":[{"type":"mcp","server_label":"github","server_url":"https://api.githubcopilot.com/mcp/","authorization":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '256' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:18:45 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29994744' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 10ms + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_dc8a2eebab2444798f51ec876a94cf44 + Openai-Processing-Ms: + - '4537' + X-Envoy-Upstream-Service-Time: + - '4540' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=c4iTaSV9bL2OmT1GObGllvGoG3URyXubGmh65fQXrKQ-1763587125-1.0.1.1-.Wo_oU2NLP3rae4sgEq5osjfYrd4gftWkXb2dkV59dirBuT4qHGiBWvpK9HrE.aMulBRA6t4q7CD_06WQHJjiwzmq7MDjVvJn6KZhdpG35Y; + path=/; expires=Wed, 19-Nov-25 21:48:45 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=4GMBp3YRCrAwye8TyVJeBl9eKm6TqRjdFn3M4LwZkLg-1763587125106-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a12bdcf39f43ad4-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_05c47f40bf062be000691e343092ac819bb56c281765935f2a", + "object": "response", + "created_at": 1763587120, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4o-2024-08-06", + "output": [ + { + "id": "mcpl_05c47f40bf062be000691e3430c7d0819b96fe9b1a02fecabc", + "type": "mcp_list_tools", + "server_label": "github", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Add review comment to the requester's latest pending pull request review. A pending review needs to already exist to call this (check with the user if not sure).", + "input_schema": { + "properties": { + "body": { + "description": "The text of the review comment", + "type": "string" + }, + "line": { + "description": "The line of the blob in the pull request diff that the comment applies to. For multi-line comments, the last line of the range", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "path": { + "description": "The relative path to the file that necessitates a comment", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "side": { + "description": "The side of the diff to comment on. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "startLine": { + "description": "For multi-line comments, the first line of the range that the comment applies to", + "type": "number" + }, + "startSide": { + "description": "For multi-line comments, the starting side of the diff that the comment applies to. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "subjectType": { + "description": "The level at which the comment is targeted", + "enum": [ + "FILE", + "LINE" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber", + "path", + "body", + "subjectType" + ], + "type": "object" + }, + "name": "add_comment_to_pending_review" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a comment to a specific issue in a GitHub repository. Use this tool to add comments to pull requests as well (in this case pass pull request number as issue_number), but only if user is not asking specifically to add review comments.", + "input_schema": { + "properties": { + "body": { + "description": "Comment content", + "type": "string" + }, + "issue_number": { + "description": "Issue number to comment on", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issue_number", + "body" + ], + "type": "object" + }, + "name": "add_issue_comment" + }, + { + "annotations": { + "read_only": false + }, + "description": "Assign Copilot to a specific issue in a GitHub repository.\n\nThis tool can help with the following outcomes:\n- a Pull Request created with source code changes to resolve the issue\n\n\nMore information can be found at:\n- https://docs.github.com/en/copilot/using-github-copilot/using-copilot-coding-agent-to-work-on-tasks/about-assigning-tasks-to-copilot\n", + "input_schema": { + "properties": { + "issueNumber": { + "description": "Issue number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issueNumber" + ], + "type": "object" + }, + "name": "assign_copilot_to_issue" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new branch in a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Name for new branch", + "type": "string" + }, + "from_branch": { + "description": "Source branch (defaults to repo default)", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch" + ], + "type": "object" + }, + "name": "create_branch" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create or update a single file in a GitHub repository. If updating, you must provide the SHA of the file you want to update. Use this tool to create or update a file in a GitHub repository remotely; do not use it for local file operations.", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to create/update the file in", + "type": "string" + }, + "content": { + "description": "Content of the file", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path where to create/update the file", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Required if updating an existing file. The blob SHA of the file being replaced.", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "content", + "message", + "branch" + ], + "type": "object" + }, + "name": "create_or_update_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "Branch to merge into", + "type": "string" + }, + "body": { + "description": "PR description", + "type": "string" + }, + "draft": { + "description": "Create as draft PR", + "type": "boolean" + }, + "head": { + "description": "Branch containing changes", + "type": "string" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "title": { + "description": "PR title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "title", + "head", + "base" + ], + "type": "object" + }, + "name": "create_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new GitHub repository in your account or specified organization", + "input_schema": { + "properties": { + "autoInit": { + "description": "Initialize with README", + "type": "boolean" + }, + "description": { + "description": "Repository description", + "type": "string" + }, + "name": { + "description": "Repository name", + "type": "string" + }, + "organization": { + "description": "Organization to create the repository in (omit to create in your personal account)", + "type": "string" + }, + "private": { + "description": "Whether repo should be private", + "type": "boolean" + } + }, + "required": [ + "name" + ], + "type": "object" + }, + "name": "create_repository" + }, + { + "annotations": { + "read_only": false + }, + "description": "Delete a file from a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to delete the file from", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path to the file to delete", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "message", + "branch" + ], + "type": "object" + }, + "name": "delete_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Fork a GitHub repository to your account or specified organization", + "input_schema": { + "properties": { + "organization": { + "description": "Organization to fork to", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "fork_repository" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details for a commit from a GitHub repository", + "input_schema": { + "properties": { + "include_diff": { + "default": true, + "description": "Whether to include file diffs and stats in the response. Default is true.", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch name, or tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "sha" + ], + "type": "object" + }, + "name": "get_commit" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the contents of a file or directory from a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "default": "/", + "description": "Path to file/directory (directories must end with a slash '/')", + "type": "string" + }, + "ref": { + "description": "Accepts optional git refs such as `refs/tags/{tag}`, `refs/heads/{branch}` or `refs/pull/{pr_number}/head`", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Accepts optional commit SHA. If specified, it will be used instead of ref", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_file_contents" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific label from a repository.", + "input_schema": { + "properties": { + "name": { + "description": "Label name.", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization name)", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "name" + ], + "type": "object" + }, + "name": "get_label" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the latest release in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_latest_release" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the authenticated GitHub user. Use this when a request is about the user's own profile for GitHub. Or when information is missing to build other tool calls.", + "input_schema": { + "properties": {}, + "type": "object" + }, + "name": "get_me" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific release by its tag name in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name (e.g., 'v1.0.0')", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_release_by_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details about a specific git tag in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get member usernames of a specific team in an organization. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "org": { + "description": "Organization login (owner) that contains the team.", + "type": "string" + }, + "team_slug": { + "description": "Team slug", + "type": "string" + } + }, + "required": [ + "org", + "team_slug" + ], + "type": "object" + }, + "name": "get_team_members" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the teams the user is a member of. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "user": { + "description": "Username to get teams for. If not provided, uses the authenticated user.", + "type": "string" + } + }, + "type": "object" + }, + "name": "get_teams" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information about a specific issue in a GitHub repository.", + "input_schema": { + "properties": { + "issue_number": { + "description": "The number of the issue", + "type": "number" + }, + "method": { + "description": "The read operation to perform on a single issue. \nOptions are: \n1. get - Get details of a specific issue.\n2. get_comments - Get issue comments.\n3. get_sub_issues - Get sub-issues of the issue.\n4. get_labels - Get labels assigned to the issue.\n", + "enum": [ + "get", + "get_comments", + "get_sub_issues", + "get_labels" + ], + "type": "string" + }, + "owner": { + "description": "The owner of the repository", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "The name of the repository", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number" + ], + "type": "object" + }, + "name": "issue_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new or update an existing issue in a GitHub repository.", + "input_schema": { + "properties": { + "assignees": { + "description": "Usernames to assign to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "body": { + "description": "Issue body content", + "type": "string" + }, + "duplicate_of": { + "description": "Issue number that this issue is a duplicate of. Only used when state_reason is 'duplicate'.", + "type": "number" + }, + "issue_number": { + "description": "Issue number to update", + "type": "number" + }, + "labels": { + "description": "Labels to apply to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "method": { + "description": "Write operation to perform on a single issue.\nOptions are: \n- 'create' - creates a new issue. \n- 'update' - updates an existing issue.\n", + "enum": [ + "create", + "update" + ], + "type": "string" + }, + "milestone": { + "description": "Milestone number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "state_reason": { + "description": "Reason for the state change. Ignored unless state is changed.", + "enum": [ + "completed", + "not_planned", + "duplicate" + ], + "type": "string" + }, + "title": { + "description": "Issue title", + "type": "string" + }, + "type": { + "description": "Type of this issue. Only use if the repository has issue types configured. Use list_issue_types tool to get valid type values for the organization. If the repository doesn't support issue types, omit this parameter.", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo" + ], + "type": "object" + }, + "name": "issue_write" + }, + { + "annotations": { + "read_only": true + }, + "description": "List branches in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_branches" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get list of commits of a branch in a GitHub repository. Returns at least 30 results per page by default, but can return more if specified using the perPage parameter (up to 100).", + "input_schema": { + "properties": { + "author": { + "description": "Author username or email address to filter commits by", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch or tag name to list commits of. If not provided, uses the default branch of the repository. If a commit SHA is provided, will list commits up to that SHA.", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_commits" + }, + { + "annotations": { + "read_only": true + }, + "description": "List supported issue types for repository owner (organization).", + "input_schema": { + "properties": { + "owner": { + "description": "The organization owner of the repository", + "type": "string" + } + }, + "required": [ + "owner" + ], + "type": "object" + }, + "name": "list_issue_types" + }, + { + "annotations": { + "read_only": true + }, + "description": "List issues in a GitHub repository. For pagination, use the 'endCursor' from the previous response's 'pageInfo' in the 'after' parameter.", + "input_schema": { + "properties": { + "after": { + "description": "Cursor for pagination. Use the endCursor from the previous page's PageInfo for GraphQL APIs.", + "type": "string" + }, + "direction": { + "description": "Order direction. If provided, the 'orderBy' also needs to be provided.", + "enum": [ + "ASC", + "DESC" + ], + "type": "string" + }, + "labels": { + "description": "Filter by labels", + "items": { + "type": "string" + }, + "type": "array" + }, + "orderBy": { + "description": "Order issues by field. If provided, the 'direction' also needs to be provided.", + "enum": [ + "CREATED_AT", + "UPDATED_AT", + "COMMENTS" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "since": { + "description": "Filter by date (ISO 8601 timestamp)", + "type": "string" + }, + "state": { + "description": "Filter by state, by default both open and closed issues are returned when not provided", + "enum": [ + "OPEN", + "CLOSED" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "List pull requests in a GitHub repository. If the user specifies an author, then DO NOT use this tool and use the search_pull_requests tool instead.", + "input_schema": { + "properties": { + "base": { + "description": "Filter by base branch", + "type": "string" + }, + "direction": { + "description": "Sort direction", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "head": { + "description": "Filter by head user/org and branch", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sort": { + "description": "Sort by", + "enum": [ + "created", + "updated", + "popularity", + "long-running" + ], + "type": "string" + }, + "state": { + "description": "Filter by state", + "enum": [ + "open", + "closed", + "all" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "List releases in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_releases" + }, + { + "annotations": { + "read_only": true + }, + "description": "List git tags in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_tags" + }, + { + "annotations": { + "read_only": false + }, + "description": "Merge a pull request in a GitHub repository.", + "input_schema": { + "properties": { + "commit_message": { + "description": "Extra detail for merge commit", + "type": "string" + }, + "commit_title": { + "description": "Title for merge commit", + "type": "string" + }, + "merge_method": { + "description": "Merge method", + "enum": [ + "merge", + "squash", + "rebase" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "merge_pull_request" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information on a specific pull request in GitHub repository.", + "input_schema": { + "properties": { + "method": { + "description": "Action to specify what pull request data needs to be retrieved from GitHub. \nPossible options: \n 1. get - Get details of a specific pull request.\n 2. get_diff - Get the diff of a pull request.\n 3. get_status - Get status of a head commit in a pull request. This reflects status of builds and checks.\n 4. get_files - Get the list of files changed in a pull request. Use with pagination parameters to control the number of results returned.\n 5. get_review_comments - Get the review comments on a pull request. They are comments made on a portion of the unified diff during a pull request review. Use with pagination parameters to control the number of results returned.\n 6. get_reviews - Get the reviews on a pull request. When asked for review comments, use get_review_comments method.\n 7. get_comments - Get comments on a pull request. Use this if user doesn't specifically want review comments. Use with pagination parameters to control the number of results returned.\n", + "enum": [ + "get", + "get_diff", + "get_status", + "get_files", + "get_review_comments", + "get_reviews", + "get_comments" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create and/or submit, delete review of a pull request.\n\nAvailable methods:\n- create: Create a new review of a pull request. If \"event\" parameter is provided, the review is submitted. If \"event\" is omitted, a pending review is created.\n- submit_pending: Submit an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request. The \"body\" and \"event\" parameters are used when submitting the review.\n- delete_pending: Delete an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request.\n", + "input_schema": { + "properties": { + "body": { + "description": "Review comment text", + "type": "string" + }, + "commitID": { + "description": "SHA of commit to review", + "type": "string" + }, + "event": { + "description": "Review action to perform.", + "enum": [ + "APPROVE", + "REQUEST_CHANGES", + "COMMENT" + ], + "type": "string" + }, + "method": { + "description": "The write operation to perform on pull request review.", + "enum": [ + "create", + "submit_pending", + "delete_pending" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_review_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Push multiple files to a GitHub repository in a single commit", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to push to", + "type": "string" + }, + "files": { + "description": "Array of file objects to push, each object with path (string) and content (string)", + "items": { + "additionalProperties": false, + "properties": { + "content": { + "description": "file content", + "type": "string" + }, + "path": { + "description": "path to the file", + "type": "string" + } + }, + "required": [ + "path", + "content" + ], + "type": "object" + }, + "type": "array" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch", + "files", + "message" + ], + "type": "object" + }, + "name": "push_files" + }, + { + "annotations": { + "read_only": false + }, + "description": "Request a GitHub Copilot code review for a pull request. Use this for automated feedback on pull requests, usually before requesting a human reviewer.", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "request_copilot_review" + }, + { + "annotations": { + "read_only": true + }, + "description": "Fast and precise code search across ALL GitHub repositories using GitHub's native search engine. Best for finding exact symbols, functions, classes, or specific code patterns.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order for results", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub's powerful code search syntax. Examples: 'content:Skill language:Java org:github', 'NOT is:archived language:Python OR language:go', 'repo:github/github-mcp-server'. Supports exact matching, language filters, path filters, and more.", + "type": "string" + }, + "sort": { + "description": "Sort field ('indexed' only)", + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_code" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for issues in GitHub repositories using issues search syntax already scoped to is:issue", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only issues for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub issues search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only issues for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for pull requests in GitHub repositories using issues search syntax already scoped to is:pr", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only pull requests for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub pull request search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only pull requests for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub repositories by name, description, readme, topics, or other metadata. Perfect for discovering projects, finding examples, or locating specific repositories across GitHub.", + "input_schema": { + "properties": { + "minimal_output": { + "default": true, + "description": "Return minimal repository information (default: true). When false, returns full GitHub API repository objects.", + "type": "boolean" + }, + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Repository search query. Examples: 'machine learning in:name stars:>1000 language:python', 'topic:react', 'user:facebook'. Supports advanced search syntax for precise filtering.", + "type": "string" + }, + "sort": { + "description": "Sort repositories by field, defaults to best match", + "enum": [ + "stars", + "forks", + "help-wanted-issues", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_repositories" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub users by username, real name, or other profile information. Useful for locating developers, contributors, or team members.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "User search query. Examples: 'john smith', 'location:seattle', 'followers:>100'. Search is automatically scoped to type:user.", + "type": "string" + }, + "sort": { + "description": "Sort users by number of followers or repositories, or when the person joined GitHub.", + "enum": [ + "followers", + "repositories", + "joined" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_users" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a sub-issue to a parent issue in a GitHub repository.", + "input_schema": { + "properties": { + "after_id": { + "description": "The ID of the sub-issue to be prioritized after (either after_id OR before_id should be specified)", + "type": "number" + }, + "before_id": { + "description": "The ID of the sub-issue to be prioritized before (either after_id OR before_id should be specified)", + "type": "number" + }, + "issue_number": { + "description": "The number of the parent issue", + "type": "number" + }, + "method": { + "description": "The action to perform on a single sub-issue\nOptions are:\n- 'add' - add a sub-issue to a parent issue in a GitHub repository.\n- 'remove' - remove a sub-issue from a parent issue in a GitHub repository.\n- 'reprioritize' - change the order of sub-issues within a parent issue in a GitHub repository. Use either 'after_id' or 'before_id' to specify the new position.\n\t\t\t\t", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "replace_parent": { + "description": "When true, replaces the sub-issue's current parent issue. Use with 'add' method only.", + "type": "boolean" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sub_issue_id": { + "description": "The ID of the sub-issue to add. ID is not the same as issue number", + "type": "number" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number", + "sub_issue_id" + ], + "type": "object" + }, + "name": "sub_issue_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update an existing pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "New base branch name", + "type": "string" + }, + "body": { + "description": "New description", + "type": "string" + }, + "draft": { + "description": "Mark pull request as draft (true) or ready for review (false)", + "type": "boolean" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number to update", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "reviewers": { + "description": "GitHub usernames to request reviews from", + "items": { + "type": "string" + }, + "type": "array" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "title": { + "description": "New title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update the branch of a pull request with the latest changes from the base branch.", + "input_schema": { + "properties": { + "expectedHeadSha": { + "description": "The expected SHA of the pull request's HEAD ref", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request_branch" + } + ] + }, + { + "id": "msg_05c47f40bf062be000691e3432aad0819b95d41198acf32dfd", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "I can help with a variety of tasks, such as:\n\n1. **Answering Questions**: Providing information on a wide range of topics.\n2. **Text Analysis**: Summarizing, translating, or rephrasing text.\n3. **Technical Support**: Offering programming help, code review, and technical explanations.\n4. **GitHub Management**: Assisting with issues, pull requests, and repository management directly on GitHub.\n5. **Image Analysis**: Describing and interpreting images that you provide.\n\nFeel free to ask about anything specific you need!" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "authorization": "", + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "github", + "server_url": "https://api.githubcopilot.com/mcp/" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 5237, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 118, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 5355 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 21:18:45 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/openai_custom_servers.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/openai_custom_servers.yml new file mode 100644 index 00000000..fbd78f87 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/openai_custom_servers.yml @@ -0,0 +1,1906 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4o","input":"Use custom tools","tools":[{"type":"mcp","server_label":"github_copilot","server_url":"https://api.githubcopilot.com/mcp/","authorization":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '264' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:17:46 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29994737' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 10ms + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_39630111118f41dfa03a34e3067af926 + Openai-Processing-Ms: + - '3750' + X-Envoy-Upstream-Service-Time: + - '3755' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=i5A0axULRCd4mwO13MroyHyKJMLB_m9yEujHaNOqjas-1763587066-1.0.1.1-QkxtATDzVMy9l.m1N0mvECilaMuCkUoYNn3TSB9tbMdjfpGUco103.RwSpq3TVLql50YF2GOBM.FFYEFzSwM4g8NcoYdx6DdsMd3IlYuMNc; + path=/; expires=Wed, 19-Nov-25 21:47:46 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=aJjPuz.w9X21XPKVf1c3EYfGWQlmaxNlzZ_HhafeMjM-1763587066219-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a12bc640dd1c487-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_0fcc7d91cd79b3ca00691e33f675148197b0046f47eabad5a6", + "object": "response", + "created_at": 1763587062, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4o-2024-08-06", + "output": [ + { + "id": "mcpl_0fcc7d91cd79b3ca00691e33f6c8788197b26a74c889b5d877", + "type": "mcp_list_tools", + "server_label": "github_copilot", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Add review comment to the requester's latest pending pull request review. A pending review needs to already exist to call this (check with the user if not sure).", + "input_schema": { + "properties": { + "body": { + "description": "The text of the review comment", + "type": "string" + }, + "line": { + "description": "The line of the blob in the pull request diff that the comment applies to. For multi-line comments, the last line of the range", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "path": { + "description": "The relative path to the file that necessitates a comment", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "side": { + "description": "The side of the diff to comment on. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "startLine": { + "description": "For multi-line comments, the first line of the range that the comment applies to", + "type": "number" + }, + "startSide": { + "description": "For multi-line comments, the starting side of the diff that the comment applies to. LEFT indicates the previous state, RIGHT indicates the new state", + "enum": [ + "LEFT", + "RIGHT" + ], + "type": "string" + }, + "subjectType": { + "description": "The level at which the comment is targeted", + "enum": [ + "FILE", + "LINE" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber", + "path", + "body", + "subjectType" + ], + "type": "object" + }, + "name": "add_comment_to_pending_review" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a comment to a specific issue in a GitHub repository. Use this tool to add comments to pull requests as well (in this case pass pull request number as issue_number), but only if user is not asking specifically to add review comments.", + "input_schema": { + "properties": { + "body": { + "description": "Comment content", + "type": "string" + }, + "issue_number": { + "description": "Issue number to comment on", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issue_number", + "body" + ], + "type": "object" + }, + "name": "add_issue_comment" + }, + { + "annotations": { + "read_only": false + }, + "description": "Assign Copilot to a specific issue in a GitHub repository.\n\nThis tool can help with the following outcomes:\n- a Pull Request created with source code changes to resolve the issue\n\n\nMore information can be found at:\n- https://docs.github.com/en/copilot/using-github-copilot/using-copilot-coding-agent-to-work-on-tasks/about-assigning-tasks-to-copilot\n", + "input_schema": { + "properties": { + "issueNumber": { + "description": "Issue number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "issueNumber" + ], + "type": "object" + }, + "name": "assign_copilot_to_issue" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new branch in a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Name for new branch", + "type": "string" + }, + "from_branch": { + "description": "Source branch (defaults to repo default)", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch" + ], + "type": "object" + }, + "name": "create_branch" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create or update a single file in a GitHub repository. If updating, you must provide the SHA of the file you want to update. Use this tool to create or update a file in a GitHub repository remotely; do not use it for local file operations.", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to create/update the file in", + "type": "string" + }, + "content": { + "description": "Content of the file", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path where to create/update the file", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Required if updating an existing file. The blob SHA of the file being replaced.", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "content", + "message", + "branch" + ], + "type": "object" + }, + "name": "create_or_update_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "Branch to merge into", + "type": "string" + }, + "body": { + "description": "PR description", + "type": "string" + }, + "draft": { + "description": "Create as draft PR", + "type": "boolean" + }, + "head": { + "description": "Branch containing changes", + "type": "string" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "title": { + "description": "PR title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "title", + "head", + "base" + ], + "type": "object" + }, + "name": "create_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new GitHub repository in your account or specified organization", + "input_schema": { + "properties": { + "autoInit": { + "description": "Initialize with README", + "type": "boolean" + }, + "description": { + "description": "Repository description", + "type": "string" + }, + "name": { + "description": "Repository name", + "type": "string" + }, + "organization": { + "description": "Organization to create the repository in (omit to create in your personal account)", + "type": "string" + }, + "private": { + "description": "Whether repo should be private", + "type": "boolean" + } + }, + "required": [ + "name" + ], + "type": "object" + }, + "name": "create_repository" + }, + { + "annotations": { + "read_only": false + }, + "description": "Delete a file from a GitHub repository", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to delete the file from", + "type": "string" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "description": "Path to the file to delete", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "path", + "message", + "branch" + ], + "type": "object" + }, + "name": "delete_file" + }, + { + "annotations": { + "read_only": false + }, + "description": "Fork a GitHub repository to your account or specified organization", + "input_schema": { + "properties": { + "organization": { + "description": "Organization to fork to", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "fork_repository" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details for a commit from a GitHub repository", + "input_schema": { + "properties": { + "include_diff": { + "default": true, + "description": "Whether to include file diffs and stats in the response. Default is true.", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch name, or tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "sha" + ], + "type": "object" + }, + "name": "get_commit" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the contents of a file or directory from a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner (username or organization)", + "type": "string" + }, + "path": { + "default": "/", + "description": "Path to file/directory (directories must end with a slash '/')", + "type": "string" + }, + "ref": { + "description": "Accepts optional git refs such as `refs/tags/{tag}`, `refs/heads/{branch}` or `refs/pull/{pr_number}/head`", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Accepts optional commit SHA. If specified, it will be used instead of ref", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_file_contents" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific label from a repository.", + "input_schema": { + "properties": { + "name": { + "description": "Label name.", + "type": "string" + }, + "owner": { + "description": "Repository owner (username or organization name)", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "name" + ], + "type": "object" + }, + "name": "get_label" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get the latest release in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "get_latest_release" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the authenticated GitHub user. Use this when a request is about the user's own profile for GitHub. Or when information is missing to build other tool calls.", + "input_schema": { + "properties": {}, + "type": "object" + }, + "name": "get_me" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get a specific release by its tag name in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name (e.g., 'v1.0.0')", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_release_by_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details about a specific git tag in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "tag": { + "description": "Tag name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "tag" + ], + "type": "object" + }, + "name": "get_tag" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get member usernames of a specific team in an organization. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "org": { + "description": "Organization login (owner) that contains the team.", + "type": "string" + }, + "team_slug": { + "description": "Team slug", + "type": "string" + } + }, + "required": [ + "org", + "team_slug" + ], + "type": "object" + }, + "name": "get_team_members" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get details of the teams the user is a member of. Limited to organizations accessible with current credentials", + "input_schema": { + "properties": { + "user": { + "description": "Username to get teams for. If not provided, uses the authenticated user.", + "type": "string" + } + }, + "type": "object" + }, + "name": "get_teams" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information about a specific issue in a GitHub repository.", + "input_schema": { + "properties": { + "issue_number": { + "description": "The number of the issue", + "type": "number" + }, + "method": { + "description": "The read operation to perform on a single issue. \nOptions are: \n1. get - Get details of a specific issue.\n2. get_comments - Get issue comments.\n3. get_sub_issues - Get sub-issues of the issue.\n4. get_labels - Get labels assigned to the issue.\n", + "enum": [ + "get", + "get_comments", + "get_sub_issues", + "get_labels" + ], + "type": "string" + }, + "owner": { + "description": "The owner of the repository", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "The name of the repository", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number" + ], + "type": "object" + }, + "name": "issue_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create a new or update an existing issue in a GitHub repository.", + "input_schema": { + "properties": { + "assignees": { + "description": "Usernames to assign to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "body": { + "description": "Issue body content", + "type": "string" + }, + "duplicate_of": { + "description": "Issue number that this issue is a duplicate of. Only used when state_reason is 'duplicate'.", + "type": "number" + }, + "issue_number": { + "description": "Issue number to update", + "type": "number" + }, + "labels": { + "description": "Labels to apply to this issue", + "items": { + "type": "string" + }, + "type": "array" + }, + "method": { + "description": "Write operation to perform on a single issue.\nOptions are: \n- 'create' - creates a new issue. \n- 'update' - updates an existing issue.\n", + "enum": [ + "create", + "update" + ], + "type": "string" + }, + "milestone": { + "description": "Milestone number", + "type": "number" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "state_reason": { + "description": "Reason for the state change. Ignored unless state is changed.", + "enum": [ + "completed", + "not_planned", + "duplicate" + ], + "type": "string" + }, + "title": { + "description": "Issue title", + "type": "string" + }, + "type": { + "description": "Type of this issue. Only use if the repository has issue types configured. Use list_issue_types tool to get valid type values for the organization. If the repository doesn't support issue types, omit this parameter.", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo" + ], + "type": "object" + }, + "name": "issue_write" + }, + { + "annotations": { + "read_only": true + }, + "description": "List branches in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_branches" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get list of commits of a branch in a GitHub repository. Returns at least 30 results per page by default, but can return more if specified using the perPage parameter (up to 100).", + "input_schema": { + "properties": { + "author": { + "description": "Author username or email address to filter commits by", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sha": { + "description": "Commit SHA, branch or tag name to list commits of. If not provided, uses the default branch of the repository. If a commit SHA is provided, will list commits up to that SHA.", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_commits" + }, + { + "annotations": { + "read_only": true + }, + "description": "List supported issue types for repository owner (organization).", + "input_schema": { + "properties": { + "owner": { + "description": "The organization owner of the repository", + "type": "string" + } + }, + "required": [ + "owner" + ], + "type": "object" + }, + "name": "list_issue_types" + }, + { + "annotations": { + "read_only": true + }, + "description": "List issues in a GitHub repository. For pagination, use the 'endCursor' from the previous response's 'pageInfo' in the 'after' parameter.", + "input_schema": { + "properties": { + "after": { + "description": "Cursor for pagination. Use the endCursor from the previous page's PageInfo for GraphQL APIs.", + "type": "string" + }, + "direction": { + "description": "Order direction. If provided, the 'orderBy' also needs to be provided.", + "enum": [ + "ASC", + "DESC" + ], + "type": "string" + }, + "labels": { + "description": "Filter by labels", + "items": { + "type": "string" + }, + "type": "array" + }, + "orderBy": { + "description": "Order issues by field. If provided, the 'direction' also needs to be provided.", + "enum": [ + "CREATED_AT", + "UPDATED_AT", + "COMMENTS" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "since": { + "description": "Filter by date (ISO 8601 timestamp)", + "type": "string" + }, + "state": { + "description": "Filter by state, by default both open and closed issues are returned when not provided", + "enum": [ + "OPEN", + "CLOSED" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "List pull requests in a GitHub repository. If the user specifies an author, then DO NOT use this tool and use the search_pull_requests tool instead.", + "input_schema": { + "properties": { + "base": { + "description": "Filter by base branch", + "type": "string" + }, + "direction": { + "description": "Sort direction", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "head": { + "description": "Filter by head user/org and branch", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sort": { + "description": "Sort by", + "enum": [ + "created", + "updated", + "popularity", + "long-running" + ], + "type": "string" + }, + "state": { + "description": "Filter by state", + "enum": [ + "open", + "closed", + "all" + ], + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "List releases in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_releases" + }, + { + "annotations": { + "read_only": true + }, + "description": "List git tags in a GitHub repository", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo" + ], + "type": "object" + }, + "name": "list_tags" + }, + { + "annotations": { + "read_only": false + }, + "description": "Merge a pull request in a GitHub repository.", + "input_schema": { + "properties": { + "commit_message": { + "description": "Extra detail for merge commit", + "type": "string" + }, + "commit_title": { + "description": "Title for merge commit", + "type": "string" + }, + "merge_method": { + "description": "Merge method", + "enum": [ + "merge", + "squash", + "rebase" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "merge_pull_request" + }, + { + "annotations": { + "read_only": true + }, + "description": "Get information on a specific pull request in GitHub repository.", + "input_schema": { + "properties": { + "method": { + "description": "Action to specify what pull request data needs to be retrieved from GitHub. \nPossible options: \n 1. get - Get details of a specific pull request.\n 2. get_diff - Get the diff of a pull request.\n 3. get_status - Get status of a head commit in a pull request. This reflects status of builds and checks.\n 4. get_files - Get the list of files changed in a pull request. Use with pagination parameters to control the number of results returned.\n 5. get_review_comments - Get the review comments on a pull request. They are comments made on a portion of the unified diff during a pull request review. Use with pagination parameters to control the number of results returned.\n 6. get_reviews - Get the reviews on a pull request. When asked for review comments, use get_review_comments method.\n 7. get_comments - Get comments on a pull request. Use this if user doesn't specifically want review comments. Use with pagination parameters to control the number of results returned.\n", + "enum": [ + "get", + "get_diff", + "get_status", + "get_files", + "get_review_comments", + "get_reviews", + "get_comments" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_read" + }, + { + "annotations": { + "read_only": false + }, + "description": "Create and/or submit, delete review of a pull request.\n\nAvailable methods:\n- create: Create a new review of a pull request. If \"event\" parameter is provided, the review is submitted. If \"event\" is omitted, a pending review is created.\n- submit_pending: Submit an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request. The \"body\" and \"event\" parameters are used when submitting the review.\n- delete_pending: Delete an existing pending review of a pull request. This requires that a pending review exists for the current user on the specified pull request.\n", + "input_schema": { + "properties": { + "body": { + "description": "Review comment text", + "type": "string" + }, + "commitID": { + "description": "SHA of commit to review", + "type": "string" + }, + "event": { + "description": "Review action to perform.", + "enum": [ + "APPROVE", + "REQUEST_CHANGES", + "COMMENT" + ], + "type": "string" + }, + "method": { + "description": "The write operation to perform on pull request review.", + "enum": [ + "create", + "submit_pending", + "delete_pending" + ], + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "method", + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "pull_request_review_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Push multiple files to a GitHub repository in a single commit", + "input_schema": { + "properties": { + "branch": { + "description": "Branch to push to", + "type": "string" + }, + "files": { + "description": "Array of file objects to push, each object with path (string) and content (string)", + "items": { + "additionalProperties": false, + "properties": { + "content": { + "description": "file content", + "type": "string" + }, + "path": { + "description": "path to the file", + "type": "string" + } + }, + "required": [ + "path", + "content" + ], + "type": "object" + }, + "type": "array" + }, + "message": { + "description": "Commit message", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "branch", + "files", + "message" + ], + "type": "object" + }, + "name": "push_files" + }, + { + "annotations": { + "read_only": false + }, + "description": "Request a GitHub Copilot code review for a pull request. Use this for automated feedback on pull requests, usually before requesting a human reviewer.", + "input_schema": { + "properties": { + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "request_copilot_review" + }, + { + "annotations": { + "read_only": true + }, + "description": "Fast and precise code search across ALL GitHub repositories using GitHub's native search engine. Best for finding exact symbols, functions, classes, or specific code patterns.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order for results", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub's powerful code search syntax. Examples: 'content:Skill language:Java org:github', 'NOT is:archived language:Python OR language:go', 'repo:github/github-mcp-server'. Supports exact matching, language filters, path filters, and more.", + "type": "string" + }, + "sort": { + "description": "Sort field ('indexed' only)", + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_code" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for issues in GitHub repositories using issues search syntax already scoped to is:issue", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only issues for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub issues search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only issues for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_issues" + }, + { + "annotations": { + "read_only": true + }, + "description": "Search for pull requests in GitHub repositories using issues search syntax already scoped to is:pr", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "owner": { + "description": "Optional repository owner. If provided with repo, only pull requests for this repository are listed.", + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Search query using GitHub pull request search syntax", + "type": "string" + }, + "repo": { + "description": "Optional repository name. If provided with owner, only pull requests for this repository are listed.", + "type": "string" + }, + "sort": { + "description": "Sort field by number of matches of categories, defaults to best match", + "enum": [ + "comments", + "reactions", + "reactions-+1", + "reactions--1", + "reactions-smile", + "reactions-thinking_face", + "reactions-heart", + "reactions-tada", + "interactions", + "created", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_pull_requests" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub repositories by name, description, readme, topics, or other metadata. Perfect for discovering projects, finding examples, or locating specific repositories across GitHub.", + "input_schema": { + "properties": { + "minimal_output": { + "default": true, + "description": "Return minimal repository information (default: true). When false, returns full GitHub API repository objects.", + "type": "boolean" + }, + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "Repository search query. Examples: 'machine learning in:name stars:>1000 language:python', 'topic:react', 'user:facebook'. Supports advanced search syntax for precise filtering.", + "type": "string" + }, + "sort": { + "description": "Sort repositories by field, defaults to best match", + "enum": [ + "stars", + "forks", + "help-wanted-issues", + "updated" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_repositories" + }, + { + "annotations": { + "read_only": true + }, + "description": "Find GitHub users by username, real name, or other profile information. Useful for locating developers, contributors, or team members.", + "input_schema": { + "properties": { + "order": { + "description": "Sort order", + "enum": [ + "asc", + "desc" + ], + "type": "string" + }, + "page": { + "description": "Page number for pagination (min 1)", + "minimum": 1, + "type": "number" + }, + "perPage": { + "description": "Results per page for pagination (min 1, max 100)", + "maximum": 100, + "minimum": 1, + "type": "number" + }, + "query": { + "description": "User search query. Examples: 'john smith', 'location:seattle', 'followers:>100'. Search is automatically scoped to type:user.", + "type": "string" + }, + "sort": { + "description": "Sort users by number of followers or repositories, or when the person joined GitHub.", + "enum": [ + "followers", + "repositories", + "joined" + ], + "type": "string" + } + }, + "required": [ + "query" + ], + "type": "object" + }, + "name": "search_users" + }, + { + "annotations": { + "read_only": false + }, + "description": "Add a sub-issue to a parent issue in a GitHub repository.", + "input_schema": { + "properties": { + "after_id": { + "description": "The ID of the sub-issue to be prioritized after (either after_id OR before_id should be specified)", + "type": "number" + }, + "before_id": { + "description": "The ID of the sub-issue to be prioritized before (either after_id OR before_id should be specified)", + "type": "number" + }, + "issue_number": { + "description": "The number of the parent issue", + "type": "number" + }, + "method": { + "description": "The action to perform on a single sub-issue\nOptions are:\n- 'add' - add a sub-issue to a parent issue in a GitHub repository.\n- 'remove' - remove a sub-issue from a parent issue in a GitHub repository.\n- 'reprioritize' - change the order of sub-issues within a parent issue in a GitHub repository. Use either 'after_id' or 'before_id' to specify the new position.\n\t\t\t\t", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "replace_parent": { + "description": "When true, replaces the sub-issue's current parent issue. Use with 'add' method only.", + "type": "boolean" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "sub_issue_id": { + "description": "The ID of the sub-issue to add. ID is not the same as issue number", + "type": "number" + } + }, + "required": [ + "method", + "owner", + "repo", + "issue_number", + "sub_issue_id" + ], + "type": "object" + }, + "name": "sub_issue_write" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update an existing pull request in a GitHub repository.", + "input_schema": { + "properties": { + "base": { + "description": "New base branch name", + "type": "string" + }, + "body": { + "description": "New description", + "type": "string" + }, + "draft": { + "description": "Mark pull request as draft (true) or ready for review (false)", + "type": "boolean" + }, + "maintainer_can_modify": { + "description": "Allow maintainer edits", + "type": "boolean" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number to update", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + }, + "reviewers": { + "description": "GitHub usernames to request reviews from", + "items": { + "type": "string" + }, + "type": "array" + }, + "state": { + "description": "New state", + "enum": [ + "open", + "closed" + ], + "type": "string" + }, + "title": { + "description": "New title", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request" + }, + { + "annotations": { + "read_only": false + }, + "description": "Update the branch of a pull request with the latest changes from the base branch.", + "input_schema": { + "properties": { + "expectedHeadSha": { + "description": "The expected SHA of the pull request's HEAD ref", + "type": "string" + }, + "owner": { + "description": "Repository owner", + "type": "string" + }, + "pullNumber": { + "description": "Pull request number", + "type": "number" + }, + "repo": { + "description": "Repository name", + "type": "string" + } + }, + "required": [ + "owner", + "repo", + "pullNumber" + ], + "type": "object" + }, + "name": "update_pull_request_branch" + } + ] + }, + { + "id": "msg_0fcc7d91cd79b3ca00691e33f93f808197971dad56d2968538", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Sure, I can use the custom tools available. How may I assist you today? If you need help with GitHub tasks, just let me know what you need!" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "mcp", + "allowed_tools": null, + "authorization": "", + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "github_copilot", + "server_url": "https://api.githubcopilot.com/mcp/" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 5244, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 36, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 5280 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 21:17:46 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/quick_start_weather.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/quick_start_weather.yml new file mode 100644 index 00000000..f4a685d8 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/quick_start_weather.yml @@ -0,0 +1,112 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","max_tokens":1024,"messages":[{"content":"What''s + the weather like?","role":"user"}],"mcp_servers":[{"name":"weather","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '208' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:17:57 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T21:17:56Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T21:17:57Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T21:17:55Z' + Retry-After: + - '4' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T21:17:56Z' + Request-Id: + - req_011CVHwikcWM2EGSQQHpy1ae + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '3652' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a12bcacf9b1ed3f-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01PRRoQY2wwJULDyvV3RCSNm","type":"message","role":"assistant","content":[{"type":"text","text":"I + don''t have access to real-time weather data. However, I do have a tool available + for information about Cloudflare''s MCP Demo Day, which might include weather-related + details for that event.\n\nIf you''re asking about:\n- **Current weather conditions** + - I''d recommend checking a weather service like Weather.com, your local news, + or a weather app on your phone.\n- **Weather for a specific location** - Let + me know the location and I can try to help direct you to weather resources.\n- + **Weather information related to Cloudflare''s MCP Demo Day** - I can look + that up for you!\n\nIs there something specific I can help you with?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":580,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":150,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 21:17:57 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/single_server.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/single_server.yml new file mode 100644 index 00000000..56a94e5a --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/single_server.yml @@ -0,0 +1,112 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","messages":[{"content":"Analyze the latest + data","role":"user"}],"max_tokens":4096,"mcp_servers":[{"name":"cloudflare-demo","type":"url","url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '215' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:19:07 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T21:19:05Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T21:19:07Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T21:19:05Z' + Retry-After: + - '55' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4800000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T21:19:05Z' + Request-Id: + - req_011CVHwot28nNvAPPLxncXvc + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '3500' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a12be5fbd3bd029-SJC + body: + encoding: ASCII-8BIT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01MhxJCr9WsGyJj5w4gagjGb","type":"message","role":"assistant","content":[{"type":"text","text":"I''d + be happy to help you analyze the latest data! However, I need more context + about what data you''d like me to analyze.\n\nCould you please provide:\n\n1. + **What specific data** are you interested in analyzing? (e.g., business metrics, + Cloudflare information, etc.)\n2. **What format is the data in?** (e.g., a + file, numbers, a dataset, etc.)\n3. **What aspects** would you like me to + focus on? (e.g., trends, comparisons, insights, etc.)\n\nAlternatively, if + you''re asking about **Cloudflare''s MCP Demo Day**, I can retrieve information + about that event. Would you like me to do that?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":583,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":159,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 21:19:07 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions/mcps/with_function_tools.yml b/test/fixtures/vcr_cassettes/docs/actions/mcps/with_function_tools.yml new file mode 100644 index 00000000..a303e0cb --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions/mcps/with_function_tools.yml @@ -0,0 +1,226 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.openai.com/v1/responses + body: + encoding: UTF-8 + string: '{"model":"gpt-4o","input":"Calculate and fetch data","tools":[{"type":"function","name":"calculate","description":"Perform + calculations","parameters":{"type":"object","properties":{"operation":{"type":"string"},"a":{"type":"number"},"b":{"type":"number"}}}},{"type":"mcp","server_label":"data-service","server_url":"https://demo-day.mcp.cloudflare.com/sse"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - OpenAI::Client/Ruby 0.35.1 + Host: + - api.openai.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 0.35.1 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + Authorization: + - Bearer ACCESS_TOKEN + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '359' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:18:40 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + X-Ratelimit-Limit-Requests: + - '10000' + X-Ratelimit-Limit-Tokens: + - '30000000' + X-Ratelimit-Remaining-Requests: + - '9999' + X-Ratelimit-Remaining-Tokens: + - '29999877' + X-Ratelimit-Reset-Requests: + - 6ms + X-Ratelimit-Reset-Tokens: + - 0s + Openai-Version: + - '2020-10-01' + Openai-Organization: + - ORGANIZATION_ID + Openai-Project: + - PROJECT_ID + X-Request-Id: + - req_97e30c6e2dd8442fac1d39d983b83032 + Openai-Processing-Ms: + - '3735' + X-Envoy-Upstream-Service-Time: + - '3738' + Cf-Cache-Status: + - DYNAMIC + Set-Cookie: + - __cf_bm=K4Fs.WFgYwgYVtywHwlxG7Ql5TmXfOl.hoVGueJW7rA-1763587120-1.0.1.1-C01.FBW5GdpmCkngSeInqzWaB3a6xl_py8UR0dLo1NBtGJgS7gh7wqp_0C6tc8MbdVEUj5C_XQF7qhBCKV4cNAg.96KyKVChgbTpKn_GHTI; + path=/; expires=Wed, 19-Nov-25 21:48:40 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=MUGXWvk6.0P3MgHKcKKOXHrb1h3yy4gpxxY0SH_TLjs-1763587120453-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + X-Content-Type-Options: + - nosniff + Server: + - cloudflare + Cf-Ray: + - 9a12bdb68d96ebe7-SJC + Alt-Svc: + - h3=":443"; ma=86400 + body: + encoding: ASCII-8BIT + string: |- + { + "id": "resp_0c71d0fe20fc9d9000691e342cb7148199b97b067fb2fe96f9", + "object": "response", + "created_at": 1763587117, + "status": "completed", + "background": false, + "billing": { + "payer": "developer" + }, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4o-2024-08-06", + "output": [ + { + "id": "mcpl_0c71d0fe20fc9d9000691e342d15f881999a7e283f8ab13502", + "type": "mcp_list_tools", + "server_label": "data-service", + "tools": [ + { + "annotations": { + "read_only": false + }, + "description": "Get information about Cloudflare's MCP Demo Day. Use this tool if the user asks about Cloudflare's MCP demo day", + "input_schema": { + "type": "object", + "properties": {} + }, + "name": "mcp_demo_day_info" + } + ] + }, + { + "id": "msg_0c71d0fe20fc9d9000691e342fc7b48199bcf2a22f78a3de03", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "Sure, I can help with that. What calculation would you like to perform, and what specific data are you seeking?" + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "prompt_cache_key": null, + "prompt_cache_retention": null, + "reasoning": { + "effort": null, + "summary": null + }, + "safety_identifier": null, + "service_tier": "default", + "store": true, + "temperature": 1.0, + "text": { + "format": { + "type": "text" + }, + "verbosity": "medium" + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Perform calculations", + "name": "calculate", + "parameters": { + "type": "object", + "properties": { + "operation": { + "type": "string" + }, + "a": { + "type": "number" + }, + "b": { + "type": "number" + } + }, + "additionalProperties": false, + "required": [ + "operation", + "a", + "b" + ] + }, + "strict": true + }, + { + "type": "mcp", + "allowed_tools": null, + "headers": null, + "require_approval": "always", + "server_description": null, + "server_label": "data-service", + "server_url": "https://demo-day.mcp.cloudflare.com/sse" + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 104, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 26, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 130 + }, + "user": null, + "metadata": {} + } + recorded_at: Wed, 19 Nov 2025 21:18:40 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/docs/actions_examples/mcps.yml b/test/fixtures/vcr_cassettes/docs/actions_examples/mcps.yml new file mode 100644 index 00000000..77d3eb26 --- /dev/null +++ b/test/fixtures/vcr_cassettes/docs/actions_examples/mcps.yml @@ -0,0 +1,104 @@ +--- +http_interactions: +- request: + method: post + uri: https://api.anthropic.com/v1/messages?beta=true + body: + encoding: UTF-8 + string: '{"model":"claude-haiku-4-5","messages":[{"content":"Research AI developments","role":"user"}],"max_tokens":4096,"mcp_servers":[{"name":"github","type":"url","url":"https://api.githubcopilot.com/mcp/","authorization_token":"GITHUB_MCP_TOKEN"}]}' + headers: + Accept-Encoding: + - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 + Accept: + - application/json + User-Agent: + - Anthropic::Client/Ruby 1.14.0 + Host: + - api.anthropic.com + X-Stainless-Arch: + - arm64 + X-Stainless-Lang: + - ruby + X-Stainless-Os: + - MacOS + X-Stainless-Package-Version: + - 1.14.0 + X-Stainless-Runtime: + - ruby + X-Stainless-Runtime-Version: + - 3.4.7 + Content-Type: + - application/json + Anthropic-Version: + - '2023-06-01' + X-Api-Key: + - ACCESS_TOKEN + Anthropic-Beta: + - mcp-client-2025-04-04 + X-Stainless-Retry-Count: + - '0' + X-Stainless-Timeout: + - '600.0' + Content-Length: + - '320' + response: + status: + code: 200 + message: OK + headers: + Date: + - Wed, 19 Nov 2025 21:21:54 GMT + Content-Type: + - application/json + Transfer-Encoding: + - chunked + Connection: + - keep-alive + Anthropic-Ratelimit-Input-Tokens-Limit: + - '4000000' + Anthropic-Ratelimit-Input-Tokens-Remaining: + - '3870000' + Anthropic-Ratelimit-Input-Tokens-Reset: + - '2025-11-19T21:21:47Z' + Anthropic-Ratelimit-Output-Tokens-Limit: + - '800000' + Anthropic-Ratelimit-Output-Tokens-Remaining: + - '799000' + Anthropic-Ratelimit-Output-Tokens-Reset: + - '2025-11-19T21:21:54Z' + Anthropic-Ratelimit-Requests-Limit: + - '4000' + Anthropic-Ratelimit-Requests-Remaining: + - '3999' + Anthropic-Ratelimit-Requests-Reset: + - '2025-11-19T21:21:36Z' + Retry-After: + - '16' + Anthropic-Ratelimit-Tokens-Limit: + - '4800000' + Anthropic-Ratelimit-Tokens-Remaining: + - '4669000' + Anthropic-Ratelimit-Tokens-Reset: + - '2025-11-19T21:21:47Z' + Request-Id: + - req_011CVHx196oPGzZRxJJscME5 + Strict-Transport-Security: + - max-age=31536000; includeSubDomains; preload + Anthropic-Organization-Id: + - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b + X-Envoy-Upstream-Service-Time: + - '17621' + Cf-Cache-Status: + - DYNAMIC + X-Robots-Tag: + - none + Server: + - cloudflare + Cf-Ray: + - 9a12c21a7dd9171a-SJC + body: + encoding: ASCII-8BIT + string: !binary |- + {"model":"claude-haiku-4-5-20251001","id":"msg_01XpF3wPD7tP8Q662X44WFRU","type":"message","role":"assistant","content":[{"type":"text","text":"I'll help you research AI developments by searching GitHub for relevant repositories and code. Let me search for AI-related projects and recent developments."},{"type":"mcp_tool_use","id":"mcptoolu_01EX16KjffBhXiSuBhQsSmEa","name":"search_repositories","input":{"query":"AI machine learning deep learning stars:>1000","sort":"stars","perPage":20},"server_name":"github"},{"type":"mcp_tool_use","id":"mcptoolu_01AP6ME8z1AwGCuzW4RkrqNL","name":"search_repositories","input":{"query":"large language model LLM transformer","sort":"updated","perPage":15},"server_name":"github"},{"type":"mcp_tool_use","id":"mcptoolu_014tJigNv1KkKLTDyj6J9Wg5","name":"search_repositories","input":{"query":"neural network AI framework PyTorch TensorFlow","sort":"stars","perPage":15},"server_name":"github"},{"type":"mcp_tool_result","tool_use_id":"mcptoolu_01EX16KjffBhXiSuBhQsSmEa","is_error":false,"content":[{"type":"text","text":"{\"total_count\":14,\"incomplete_results\":false,\"items\":[{\"id\":1198539,\"name\":\"netron\",\"full_name\":\"lutzroeder/netron\",\"description\":\"Visualizer for neural network, deep learning and machine learning models\",\"html_url\":\"https://github.com/lutzroeder/netron\",\"language\":\"JavaScript\",\"stargazers_count\":31822,\"forks_count\":3030,\"open_issues_count\":20,\"updated_at\":\"2025-11-19T17:03:54Z\",\"created_at\":\"2010-12-26T12:53:43Z\",\"topics\":[\"ai\",\"coreml\",\"deep-learning\",\"deeplearning\",\"keras\",\"machine-learning\",\"machinelearning\",\"ml\",\"neural-network\",\"numpy\",\"onnx\",\"pytorch\",\"safetensors\",\"tensorflow\",\"tensorflow-lite\",\"visualizer\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":326174741,\"name\":\"500-AI-Machine-learning-Deep-learning-Computer-vision-NLP-Projects-with-code\",\"full_name\":\"ashishpatel26/500-AI-Machine-learning-Deep-learning-Computer-vision-NLP-Projects-with-code\",\"description\":\"500 AI Machine learning Deep learning Computer vision NLP Projects with code\",\"html_url\":\"https://github.com/ashishpatel26/500-AI-Machine-learning-Deep-learning-Computer-vision-NLP-Projects-with-code\",\"stargazers_count\":28865,\"forks_count\":6445,\"open_issues_count\":59,\"updated_at\":\"2025-11-19T21:19:02Z\",\"created_at\":\"2021-01-02T12:08:37Z\",\"topics\":[\"artificial-intelligence\",\"artificial-intelligence-projects\",\"awesome\",\"computer-vision\",\"computer-vision-project\",\"data-science\",\"deep-learning\",\"deep-learning-project\",\"machine-learning\",\"machine-learning-projects\",\"nlp\",\"nlp-projects\",\"python\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":92287968,\"name\":\"cheatsheets-ai\",\"full_name\":\"kailashahirwar/cheatsheets-ai\",\"description\":\"Essential Cheat Sheets for deep learning and machine learning researchers https://medium.com/@kailashahirwar/essential-cheat-sheets-for-machine-learning-and-deep-learning-researchers-efb6a8ebd2e5\",\"html_url\":\"https://github.com/kailashahirwar/cheatsheets-ai\",\"stargazers_count\":15372,\"forks_count\":3436,\"open_issues_count\":12,\"updated_at\":\"2025-11-18T05:23:30Z\",\"created_at\":\"2017-05-24T12:06:56Z\",\"topics\":[\"artificial-intelligence\",\"deep-learning\",\"keras\",\"machine-learning\",\"matplotlib\",\"numpy\",\"scipy\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":236637683,\"name\":\"Ai-Learn\",\"full_name\":\"tangyudi/Ai-Learn\",\"description\":\"人工智能学习路线图，整理近200个实战案例与项目，免费提供配套教材，零基础入门，就业实战！包括：Python，数学，机器学习，数据分析，深度学习，计算机视觉，自然语言处理，PyTorch tensorflow machine-learning,deep-learning data-analysis data-mining mathematics data-science artificial-intelligence python tensorflow tensorflow2 caffe keras pytorch algorithm numpy pandas matplotlib seaborn nlp cv等热门领域\",\"html_url\":\"https://github.com/tangyudi/Ai-Learn\",\"stargazers_count\":12187,\"forks_count\":2570,\"open_issues_count\":20,\"updated_at\":\"2025-11-19T16:07:31Z\",\"created_at\":\"2020-01-28T01:55:06Z\",\"topics\":[\"algorithm\",\"artificial-intelligence\",\"caffe\",\"cv\",\"data-analysis\",\"data-mining\",\"data-science\",\"deep-learning\",\"keras\",\"machine-learning\",\"mathematics\",\"matplotlib\",\"nlp\",\"numpy\",\"pandas\",\"python\",\"pytorch\",\"seaborn\",\"tensorflow\",\"tensorflow2\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":274587563,\"name\":\"coursera-deep-learning-specialization\",\"full_name\":\"amanchadha/coursera-deep-learning-specialization\",\"description\":\"Notes, programming assignments and quizzes from all courses within the Coursera Deep Learning specialization offered by deeplearning.ai: (i) Neural Networks and Deep Learning; (ii) Improving Deep Neural Networks: Hyperparameter tuning, Regularization and Optimization; (iii) Structuring Machine Learning Projects; (iv) Convolutional Neural Networks; (v) Sequence Models\",\"html_url\":\"https://github.com/amanchadha/coursera-deep-learning-specialization\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":4063,\"forks_count\":2595,\"open_issues_count\":31,\"updated_at\":\"2025-11-19T20:19:38Z\",\"created_at\":\"2020-06-24T05:59:01Z\",\"topics\":[\"andrew-ng\",\"andrew-ng-course\",\"cnns\",\"convolutional-neural-network\",\"convolutional-neural-networks\",\"coursera\",\"coursera-assignment\",\"coursera-machine-learning\",\"coursera-specialization\",\"deep-learning\",\"hyperparameter-optimization\",\"hyperparameter-tuning\",\"neural-machine-translation\",\"neural-network\",\"neural-networks\",\"neural-style-transfer\",\"recurrent-neural-network\",\"recurrent-neural-networks\",\"regularization\",\"rnns\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":97159209,\"name\":\"Artificial-Intelligence-Deep-Learning-Machine-Learning-Tutorials\",\"full_name\":\"TarrySingh/Artificial-Intelligence-Deep-Learning-Machine-Learning-Tutorials\",\"description\":\"A comprehensive list of Deep Learning / Artificial Intelligence and Machine Learning tutorials - rapidly expanding into areas of AI/Deep Learning / Machine Vision / NLP and industry specific areas such as Climate / Energy, Automotives, Retail, Pharma, Medicine, Healthcare, Policy, Ethics and more.\",\"html_url\":\"https://github.com/TarrySingh/Artificial-Intelligence-Deep-Learning-Machine-Learning-Tutorials\",\"language\":\"Python\",\"stargazers_count\":3938,\"forks_count\":1637,\"open_issues_count\":157,\"updated_at\":\"2025-11-19T12:42:51Z\",\"created_at\":\"2017-07-13T19:46:01Z\",\"topics\":[\"artificial-intelligence\",\"aws\",\"capsule-network\",\"convolutional-neural-networks\",\"deep-learning\",\"ipython-notebook\",\"kaggle\",\"keras\",\"lua\",\"machine-learning\",\"matplotlib\",\"neural-network\",\"pandas\",\"python\",\"python-data\",\"pytorch\",\"scikit-learn\",\"tensorflow\",\"tensorflow-tutorials\",\"torch\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":163941732,\"name\":\"Girls-In-AI\",\"full_name\":\"girls-in-ai/Girls-In-AI\",\"description\":\"免费学代码系列：小白python入门、数据分析data analyst、机器学习machine learning、深度学习deep learning、kaggle实战\",\"html_url\":\"https://github.com/girls-in-ai/Girls-In-AI\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":3297,\"forks_count\":496,\"open_issues_count\":33,\"updated_at\":\"2025-11-18T18:38:02Z\",\"created_at\":\"2019-01-03T08:03:13Z\",\"topics\":[\"deep-learning\",\"girl\",\"jupyter-notebook\",\"kaggle\",\"machine-learning\",\"python3\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":124047136,\"name\":\"AlphaTree-graphic-deep-neural-network\",\"full_name\":\"weslynn/AlphaTree-graphic-deep-neural-network\",\"description\":\"AI Roadmap:机器学习(Machine Learning)、深度学习(Deep Learning)、对抗神经网络(GAN），图神经网络（GNN），NLP，大数据相关的发展路书(roadmap), 并附海量源码（python，pytorch）带大家消化基本知识点，突破面试，完成从新手到合格工程师的跨越，其中深度学习相关论文附有tensorflow caffe官方源码，应用部分含推荐算法和知识图谱\",\"html_url\":\"https://github.com/weslynn/AlphaTree-graphic-deep-neural-network\",\"stargazers_count\":2938,\"forks_count\":613,\"open_issues_count\":7,\"updated_at\":\"2025-11-19T07:55:54Z\",\"created_at\":\"2018-03-06T08:37:27Z\",\"topics\":[\"deep-learning\",\"image-classification\",\"machine-learning\",\"neural-network\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":117641180,\"name\":\"Papers-Literature-ML-DL-RL-AI\",\"full_name\":\"tirthajyoti/Papers-Literature-ML-DL-RL-AI\",\"description\":\"Highly cited and useful papers related to machine learning, deep learning, AI, game theory, reinforcement learning\",\"html_url\":\"https://github.com/tirthajyoti/Papers-Literature-ML-DL-RL-AI\",\"stargazers_count\":2744,\"forks_count\":784,\"open_issues_count\":1,\"updated_at\":\"2025-11-18T16:33:24Z\",\"created_at\":\"2018-01-16T06:25:18Z\",\"topics\":[\"artificial-intelligence\",\"data-mining\",\"data-science\",\"deep-learning\",\"game-theory\",\"hardware\",\"learning-theory\",\"literature\",\"machine-learning\",\"machine-learning-algorithms\",\"neural-network\",\"paper\",\"pattern-recognition\",\"reinforcement-learning\",\"silicon\",\"statistical-learning\",\"statistics\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":33331247,\"name\":\"china-dictatorship\",\"full_name\":\"cirosantilli/china-dictatorship\",\"description\":\"反中共政治宣传库。Anti Chinese government propaganda. 住在中国真名用户的网友请别给星星，不然你要被警察请喝茶。常见问答集，新闻集和饭店和音乐建议。卐习万岁卐。冠状病毒审查郝海东新疆改造中心六四事件法轮功 996.ICU709大抓捕巴拿马文件邓家贵低端人口西藏骚乱。Friends who live in China and have real name on account, please don't star this repo, or else the police might pay you a visit.  Home to the mega-FAQ, news compilation, restaurant and music recommendations.Heil Xi 卐. 大陆修宪香港恶法台湾武统朝鲜毁约美中冷战等都是王沪宁愚弄习思想极左命运共同体的大策划中共窃国这半个多世纪所犯下的滔天罪恶，前期是毛泽东策划的，中期6.4前后是邓小平策划的，黄牛数据分析后期是毛的极左追随者三朝罪恶元凶王沪宁策划的。王沪宁高小肆业因文革政治和情报需要保送“学院外语班“红色仕途翻身，所以王的本质是极左的。他是在上海底层弄堂长大的，因其本性也促成其瘪三下三滥个性，所以也都说他有易主“变色龙”哈巴狗“的天性。大陆像王沪宁这样学马列政治所谓\\\"法学\\\"专业的人，在除朝鲜古巴所有国家特别是在文明发达国家是无法找到专业对口工作必定失业，唯独在大陆却是重用的紧缺“人才”，6.4后中共信仰大危机更是最重用的救党“人才”。这也就是像王沪宁此类工农兵假“大学生”平步青云的原因，他们最熟悉毛泽东历次运动的宫庭内斗经验手段和残酷的阶级斗争等暴力恐怖的“政治学”。王沪宁能平步青云靠他这马毛伪“政治学”资本和头衔，不是什么真才实学，能干实事有点真才实学的或许在他手下的谋士及秘书班子中可以找到。王沪宁的“真才实学”只不过是一个只读四年小学的人，大半辈子在社会上磨炼特别是在中共官场滚打炼出的的手段和经验而已，他和习近平等保送的工农兵假“大学生”都一样，无法从事原“专业”都凭红资本而从政。六四学运期间各界一边倒支持学生，王沪宁一度去法国躲避和筹谋，他还加入了反学运签名，成为极少有的反学运者仕途突显，在六四和苏联垮台后中共意识形态危机，江泽民上台看上唯一能应急的王沪宁聚谋士泡制的\\\"稳定统一领导\\\"和之后的\\\"新权威\\\"谬论。左转被邓小平南巡阻止后，王策划顺邓经济改革却将政治改革逐步全面终止和倒退，泡制“三个代表”为极左转建立庞大牢固的红色既得利益集团。因此六四后各重大决策和危机难题都摆在中共中央政策研究室王沪宁桌面上，使王沪宁成了此后中共三朝都无法摆脱的幕后最有决策性实权的人，中共中央政策研究室是王为其野心巨资经营几十年，聚众谋士的间谍情报汇总研究的特务机关和策划制定决策重要机构与基地，王沪宁本人和决定其仕途关键的首任岳父及家属就有情报工作背景。中央政研室重要到王沪宁入常后为了死抓这中共情报与决策大权，宁可放弃国家副主席和中央党校校长。后再加个除习外唯他担任的中共几核心领导小组之一的“不忘初心牢记使命”主题教育工作小组组长。此后他把持的舆论必将以宣传“不忘初心牢记使命”为主，打造众所周知的所谓“习思想”其实是”王思想“。王自从主导中央政研室开始决策后，策划中止邓小平的与美妥协路线回归毛极左的反美路线。帮助前南斯拉夫提供情报打落美机放中使馆引发炸使馆事件，以此掀起六四后唯一的全国大规模游行并借此反美而起家。后又帮江泽民提供法轮功会是超过中共组织的情报，策划决策镇压迫害开始并没有把矛头指向江的法轮功群体，策划决定阻止党内外近三十年来平反六四的呼声。致远黑皮书马拉松程序员易支付英语台词文字匹配美团点评各业务线提供知识库团队共享阿里云高精Excel识别德讯 ·吉特胡布薄熙来黑科技习近平讲话模拟器习近平音源黑马程序员MySQL数据库玉米杂草数据集销售系统开发疫情期间网民情绪识别比赛996icu996 icu学习强国预测结果导出赖伟林刺杀小说家购物商场英语词汇量小程序联级选择器Bitcoin区块链 技术面试必备基础知识 Leetcode 计算机操作系统 计算机网络 系统设计 Java学习 面试指南 一份涵盖大部分 Java 程序员所需要掌握的核心知识 准备 Java 面试 首选 JavaGuide Python 1 天从新手到大师刷算法全靠套路 认准 labuladong 就够了 免费的计算机编程类中文书籍 欢迎投稿用动画的形式呈现解LeetCode题目的思路 互联网 Java 工程师进阶知识完全扫盲 涵盖高并发 分布式 高可用 微服务 海量数据处理等领域知识后端架构师技术图谱mall项目是一套电商系统 包括前台商城系统及后台管理系统 基于SpringBoot MyBatis实现 采用Docker容器化部署 前台商城系统包含首页门户 商品推荐 商品搜索 商品展示 购物车 订单流程 会员中心 客户服务 帮助中心等模块 后台管理系统包含商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等模块 微信小程序开发资源汇总 最全中华古诗词数据库 唐宋两朝近一万四千古诗人 接近5 5万首唐诗加26万宋诗 两宋时期1564位词人 21 5 首词 uni app 是使用 Vue 语法开发小程序 H5 App的统一框架2 21年最新总结 阿里 腾讯 百度 美团 头条等技术面试题目 以及答案 专家出题人分析汇总 科学上网 自由上网 翻墙 软件 方法 一键翻墙浏览器 免费账号 节点分享 vps一键搭建脚本 教程AiLearning 机器学习 MachineLearning ML 深度学习 DeepLearning DL 自然语言处理 NLP123 6智能刷票 订票开放式跨端跨框架解决方案 支持使用 React Vue Nerv 等框架来开发微信 京东 百度 支付宝 字节跳动 QQ 小程序 H5 React Native 等应用 taro zone 掘金翻译计划 可能是世界最大最好的英译中技术社区 最懂读者和译者的翻译平台  no evil 程序员找工作黑名单 换工作和当技术合伙人需谨慎啊 更新有赞 算法面试 算法知识 针对小白的算法训练 还包括 1 阿里 字节 滴滴 百篇大厂面经汇总 2 千本开源电子书 3 百张思维导图 右侧来个 star 吧 English version supported 955 不加班的公司名单 工作 955 work–life balance 工作与生活的平衡 诊断利器Arthas The Way to Go 中文译本 中文正式名 Go 入门指南  Java面试 Java学习指南 一份涵盖大部分Java程序员所需要掌握的核心知识 教程 技术栈示例代码 快速简单上手教程 2 17年买房经历总结出来的买房购房知识分享给大家 希望对大家有所帮助 买房不易 且买且珍惜http下载工具 基于http代理 支持多连接分块下载 动手学深度学习 面向中文读者 能运行 可讨论 中英文版被全球175所大学采用教学 阿里云计算平台团队出品 为监控而生的数据库连接池程序员简历模板系列 包括PHP程序员简历模板 iOS程序员简历模板 Android程序员简历模板 Web前端程序员简历模板 Java程序员简历模板 C C 程序员简历模板 NodeJS程序员简历模板 架构师简历模板以及通用程序员简历模板采用自身模块规范编写的前端 UI 框架 遵循原生 HTML CSS JS 的书写形式 极低门槛 拿来即用 贵校课程资料民间整理 企业级低代码平台 前后端分离架构强大的代码生成器让前后端代码一键生成 无需写任何代码 引领新的开发模式OnlineCoding 代码生成 手工MERGE 帮助Java项目解决7 %重复工作 让开发更关注业务 既能快速提高效率 帮助公司节省成本 同时又不失灵活性 我是依扬 木易杨 公众号 高级前端进阶 作者 每天搞定一道前端大厂面试题 祝大家天天进步 一年后会看到不一样的自己 冴羽写博客的地方 预计写四个系列 JavaScript深入系列 JavaScript专题系列 ES6系列 React系列 中文分词 词性标注 命名实体识别 依存句法分析 语义依存分析 新词发现 关键词短语提取 自动摘要 文本分类聚类 拼音简繁转换 自然语言处理flutter 开发者帮助 APP 包含 flutter 常用 14 组件的demo 演示与中文文档 下拉刷新 上拉加载 二级刷新 淘宝二楼智能下拉刷新框架 支持越界回弹 越界拖动 具有极强的扩展性 集成了几十种炫酷的Header和 Footer 该项目已成功集成 actuator 监控 admin 可视化监控 logback 日志 aopLog 通过AOP记录web请求日志 统一异常处理 json级别和页面级别 freemarker 模板引擎 thymeleaf 模板引擎 Beetl 模板引擎 Enjoy 模板引擎 JdbcTemplate 通用JDBC操作数据库 JPA 强大的ORM框架 mybatis 强大的ORM框架 通用Mapper 快速操作Mybatis PageHelper 通用的Mybatis分页插件 mybatis plus 快速操作Mybatis BeetlSQL 强大的ORM框架 u Python资源大全中文版 包括 Web框架 网络爬虫 模板引擎 数据库 数据可视化 图片处理等 由 开源前哨 和 Python开发者 微信公号团队维护更新 吴恩达老师的机器学习课程个人笔记To Be Top Javaer Java工程师成神之路循序渐进 学习博客Spring系列源码 mrbird cc谢谢可能是让你受益匪浅的英语进阶指南镜像网易云音乐 Node js API service快速 简单避免OOM的java处理Excel工具基于 Vue js 的小程序开发框架 从底层支持 Vue js 语法和构建工具体系 中文版 Apple 官方 Swift 教程本项目曾冲到全球第一 干货集锦见本页面最底部 另完整精致的纸质版 编程之法 面试和算法心得 已在京东 当当上销售好耶 是女装Security Guide for Developers 实用性开发人员安全须知 阿里巴巴 MySQL binlog 增量订阅 消费组件  ECMAScript 6入门 是一本开源的 JavaScript 语言教程 全面介绍 ECMAScript 6 新增的语法特性  C C 技术面试基础知识总结 包括语言 程序库 数据结构 算法 系统 网络 链接装载库等知识及面试经验 招聘 内推等信息 一款优秀的开源博客发布应用  Solutions to LeetCode by Go 1 % test coverage runtime beats 1 % LeetCode 题解分布式任务调度平台XXL JOB  谷粒 Chrome插件英雄榜 为优秀的Chrome插件写一本中文说明书 让Chrome插件英雄们造福人类公众号 加1 同步更新互联网公司技术架构 微信 淘宝 微博 腾讯 阿里 美团点评 百度 Google Facebook Amazon eBay的架构 欢迎PR补充IntelliJ IDEA 简体中文专题教程程序员技能图谱前端面试每日 3 1 以面试题来驱动学习 提倡每日学习与思考 每天进步一点 每天早上5点纯手工发布面试题 死磕自己 愉悦大家 4 道前端面试题全面覆盖小程序 软技能 华为鸿蒙操作系统 互联网首份程序员考公指南 由3位已经进入体制内的前大厂程序员联合献上 Mac微信功能拓展 微信插件 微信小助手 A plugin for Mac WeChat  机器学习 西瓜书 公式推导解析 在线阅读地址一款轻量级 高性能 功能强大的内网穿透代理服务器 支持tcp udp socks5 http等几乎所有流量转发 可用来访问内网网站 本地支付接口调试 ssh访问 远程桌面 内网dns解析 内网socks5代理等等 并带有功能强大的web管理端一款面向泛前端产品研发全生命周期的效率平台 文言文編程語言清华大学计算机系课程攻略面向云原生微服务的高可用流控防护组件  On Java 8 中文版 本文原文由知名 Hacker Eric S Raymond 所撰寫 教你如何正確的提出技術問題並獲得你滿意的答案 React Native指南汇集了各类react native学习资源 开源App和组件1 Days Of ML Code中文版千古前端图文教程 超详细的前端入门到进阶学习笔记 从零开始学前端 做一名精致优雅的前端工程师 公众号 千古壹号 作者 基于 React 的渐进式研发框架 ice work视频播放器支持弹幕 外挂字幕 支持滤镜 水印 gif截图 片头广告 中间广告 多个同时播放 支持基本的拖动 声音 亮度调节 支持边播边缓存 支持视频自带rotation的旋转 9 27 之类 重力旋转与手动旋转的同步支持 支持列表播放 列表全屏动画 视频加载速度 列表小窗口支持拖动 动画效果 调整比例 多分辨率切换 支持切换播放器 进度条小窗口预览 列表切换详情页面无缝播放 rtsp concat mpeg JumpServer 是全球首款开源的堡垒机 是符合 4A 的专业运维安全审计系统 Linux命令大全搜索工具 内容包含Linux命令手册 详解 学习 搜集 git io linux book Node js 包教不包会 by alsotang又一个小商城 litemall Spring Boot后端 Vue管理员前端 微信小程序用户前端 Vue用户移动端微信 跳一跳 Python 辅助Java资源大全中文版 包括开发库 开发工具 网站 博客 微信 微博等 由伯乐在线持续更新  python模拟登陆一些大型网站 还有一些简单的爬虫 希望对你们有所帮助 ️ 如果喜欢记得给个star哦 C 那些事 网络爬虫实战 淘宝 京东 网易云 B站 123 6 抖音 笔趣阁 漫画小说下载 音乐电影下载等deeplearning ai 吴恩达老师的深度学习课程笔记及资源 Spring Boot基础教程 Spring Boot 2 x版本连载中 帮助 Android App 进行组件化改造的路由框架 最接近原生APP体验的高性能框架基于Vue3 Element Plus 的后台管理系统解决方案程序员如何优雅的挣零花钱 2 版 升级为小书了 从Java基础 JavaWeb基础到常用的框架再到面试题都有完整的教程 几乎涵盖了Java后端必备的知识点spring boot 实践学习案例 是 spring boot 初学者及核心技术巩固的最佳实践 另外写博客 用 OpenWrite 最好用的 V2Ray 一键安装脚本 管理脚本中国程序员容易发音错误的单词 统计学习方法 的代码实现关于Python的面试题本项目将 动手学深度学习 Dive into Deep Learning 原书中的MXNet实现改为PyTorch实现 提高 Android UI 开发效率的 UI 库前端精读周刊 帮你理解最前沿 实用的技术  的奇技淫巧时间选择器 省市区三级联动 Python爬虫代理IP池 proxy pool LeetCode 刷题攻略 2 道经典题目刷题顺序 共6 w字的详细图解 视频难点剖析 5 余张思维导图 从此算法学习不再迷茫 来看看 你会发现相见恨晚 一个基于 electron 的音乐软件Flutter 超完整的开源项目 功能丰富 适合学习和日常使用 GSYGithubApp系列的优势 我们目前已经拥有四个版本 功能齐全 项目框架内技术涉及面广 完成度高 持续维护 配套文章 适合全面学习 对比参考 跨平台的开源Github客户端App 更好的体验 更丰富的功能 旨在更好的日常管理和维护个人Github 提供更好更方便的驾车体验Σ 同款Weex版本同款React Native版本 https g 这是一个用于显示当前网速 CPU及内存利用率的桌面悬浮窗软件 并支持任务栏显示 支持更换皮肤 是一个跨平台的强加密无特征的代理软件 零配置 V2rayU 基于v2ray核心的mac版客户端 用于科学上网 使用swift编写 支持vmess shadowsocks socks5等服务协议 支持订阅 支持二维码 剪贴板导入 手动配置 二维码分享等算法模板 最科学的刷题方式 最快速的刷题路径 你值得拥有 经典编程书籍大全 涵盖 计算机系统与网络 系统架构 算法与数据结构 前端开发 后端开发 移动开发 数据库 测试 项目与团队 程序员职业修炼 求职面试等wangEditor 轻量级web富文本框前端跨框架跨平台框架 每个 JavaScript 工程师都应懂的33个概念 leonardomso一个可以观看国内主流视频平台所有视频的客户端Android开发人员不得不收集的工具类集合 支付宝支付 微信支付 统一下单 微信分享 Zip4j压缩 支持分卷压缩与加密 一键集成UCrop选择圆形头像 一键集成二维码和条形码的扫描与生成 常用Dialog WebView的封装可播放视频 仿斗鱼滑动验证码 Toast封装 震动 GPS Location定位 图片缩放 Exif 图片添加地理位置信息 经纬度 蛛网等级 颜色选择器 ArcGis VTPK 编译运行一下说不定会找到惊喜 123 6 购票助手 支持集群 多账号 多任务购票以及 Web 页面管理  编程随想 收藏的电子书清单 多个学科 含下载链接  Banner 2 来了 Android广告图片轮播控件 内部基于ViewPager2实现 Indicator和UI都可以自定义  零代码 热更新 自动化 ORM 库 后端接口和文档零代码 前端 客户端 定制返回 JSON 的数据和结构 Linux Windows macOS 跨平台 V2Ray 客户端 支持使用 C Qt 开发 可拓展插件式设计 walle 瓦力 Devops开源项目代码部署平台基于 node js Mongodb 构建的后台系统 js 源码解析一个涵盖六个专栏分布式消息队列 分布式事务的仓库 希望胖友小手一抖 右上角来个 Star 感恩 1 24基于 vue element ui 的后台管理系统磁力链接聚合搜索中华人民共和国行政区划 省级 省份直辖市自治区 地级 城市 县级 区县 乡级 乡镇街道 村级 村委会居委会 中国省市区镇村二级三级四级五级联动地址数据 iOS开发常用三方库 插件 知名博客等等LeetCode题解 151道题完整版／中文文案排版指北最良心的 Python 教程 业内为数不多致力于极致体验的超强全自研跨平台 windows android iOS 流媒体内核 通过模块化自由组合 支持实时RTMP推流 RTSP推流 RTMP播放器 RTSP播放器 录像 多路流媒体转发 音视频导播 动态视频合成 音频混音 直播互动 内置轻量级RTSP服务等 比快更快 业界真正靠谱的超低延迟直播SDK 1秒内 低延迟模式下2 4 ms  一个 PHP 微信 SDK ️ 跨平台桌面端视频资源播放器 简洁无广告 免费高颜值 后台管理主线版本基于三者并行开发维护 同时支持电脑 手机 平板 切换分支查看不同的vue版本 element plus版本已发布 vue3 vue3 vue vue3 x vue js 程序无国界 但程序员有国界 中国国家尊严不容挑衅 如果您在特殊时期 此项目是机器学习 Machine Learning 深度学习 Deep Learning NLP面试中常考到的知识点和代码实现 也是作为一个算法工程师必会的理论基础知识 夜读 通过 bilibili 在线直播的方式分享 Go 相关的技术话题 每天大家在微信 telegram Slack 上及时沟通交流编程技术话题 GitHubDaily 分享内容定期整理与分类 欢迎推荐 自荐项目 让更多人知道你的项目  支持多家云存储的云盘系统机器学习相关教程DataX是阿里云DataWorks数据集成的开源版本  这里是写博客的地方 Halfrost Field 冰霜之地mall学习教程 架构 业务 技术要点全方位解析 mall项目 4 k star 是一套电商系统 使用现阶段主流技术实现 涵盖了等技术 采用Docker容器化部署  chick 是使用 Node js 和 MongoDB 开发的社区系统一个非常适合IT团队的在线API文档 技术文档工具汇总各大互联网公司容易考察的高频leetcode题 1 Chinese Word Vectors 上百种预训练中文词向量 Android开源弹幕引擎 烈焰弹幕使 ～深度学习框架PyTorch 入门与实战 网易云音乐命令行版本 对开发人员有用的定律 理论 原则和模式TeachYourselfCS 的中文翻译高颜值的第三方网易云播放器 支持 Windows macOS Linux spring cloud vue oAuth2 全家桶实战 前后端分离模拟商城 完整的购物流程 后端运营平台 可以实现快速搭建企业级微服务项目 支持微信登录等三方登录  Chinese sticker pack More joy 表情包的博物馆 Github最有毒的仓库 中国表情包大集合 聚欢乐 Lantern官方版本下载 蓝灯 翻墙 代理 科学上网 外网 加速器 梯子 路由一款入门级的人脸 视频 文字检测以及识别的项目 vue2 vue router vuex 入门项目PanDownload的个人维护版本 一个基于Spring Boot MyBatis的种子项目 用于快速构建中小型API RESTful API项目 iOS interview questions iOS面试题集锦 附答案 学习qq群或 Telegram 群交流为互联网IT人打造的中文版awesome go强大 可定制 易扩展的 ViewPager 指示器框架 是的最佳替代品 支持角标 更支持在非ViewPager场景下使用 使用hide show 切换Fragment或使用se Kubernetes中文指南 云原生应用架构实践手册For macOS 百度网盘 破解SVIP 下载速度限制 架构师技术图谱 助你早日成为架构师mall admin web是一个电商后台管理系统的前端项目 基于Vue Element实现 主要包括商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等功能 网易云音乐第三方 编程随想 整理的 太子党关系网络 专门揭露赵国的权贵基于gin vue搭建的后台管理系统框架 集成jwt鉴权 权限管理 动态路由 分页封装 多点登录拦截 资源权限 上传下载 代码生成器 表单生成器 通用工作流等基础功能 五分钟一套CURD前后端代码 目VUE3版本正在重构 欢迎issue和pr 27天成为Java大神一个基于浏览器端 JS 实现的在线代理编程电子书 电子书 编程书籍 包括人工智能 大数据类 并发编程 数据库类 数据挖掘 新面试题 架构设计 算法系列 计算机类 设计模式 软件测试 重构优化 等更多分类ADB Usage Complete ADB 用法大全二维码生成器 支持 gif 动态图片二维码 Vim 从入门到精通阿布量化交易系统 股票 期权 期货 比特币 机器学习 基于python的开源量化交易 量化投资架构一个简洁优雅的hexo主题 Wiki of OI ICPC for everyone 某大型游戏线上攻略 内含炫酷算术魔法 Google 开源项目风格指南 中文版  Git AWS Google 镜像 SS SSR VMESS节点行业研究报告的知识储备库 cim cross IM 适用于开发者的分布式即时通讯系统微信小程序开源项目库汇总每天更新 全网热门 BT Tracker 列表 天用Go动手写 从零实现系列强大的哔哩哔哩增强脚本 下载视频 音乐 封面 弹幕 简化直播间 评论区 首页 自定义顶栏 删除广告 夜间模式 触屏设备支持Evil Huawei 华为作过的恶Android上一个优雅 万能自定义UI 仿iOS 支持垂直 水平方向切换 支持周视图 自定义周起始 性能高效的日历控件 支持热插拔实现的UI定制 支持标记 自定义颜色 农历 自定义月视图各种显示模式等 Canvas绘制 速度快 占用内存低 你真的想不到日历居然还可以如此优雅已不再维护科学上网插件的离线安装包储存在这里ThinkPHP Framework 十年匠心的高性能PHP框架 Java 程序员眼中的 Linux 一个支持多选 选原图和视频的图片选择器 同时有预览 裁剪功能 支持hsweb haʊs wɛb 是一个基于spring boot 2 x开发 首个使用全响应式编程的企业级后台管理系统基础项目 学习强国 懒人刷分工具 自动学习wxParse 微信小程序富文本解析自定义组件 支持HTML及markdown解析 newbee mall 项目 新蜂商城 是一套电商系统 包括 newbee mall 商城系统及 newbee mall admin 商城后台管理系统 基于 Spring Boot 2 X 及相关技术栈开发 前台商城系统包含首页门户 商品分类 新品上线 首页轮播 商品推荐 商品搜索 商品展示 购物车 订单结算 订单流程 个人订单管理 会员中心 帮助中心等模块 后台管理系统包含数据面板 轮播图管理 商品管理 订单管理 会员管理 分类管理 设置等模块  最全的前端资源汇总仓库 包括前端学习 开发资源 求职面试等 中文翻译手写实现李航 统计学习方法 书中全部算法 Python 抖音机器人 论如何在抖音上找到漂亮小姐姐？  ️A static blog writing client 一个静态博客写作客户端 超级速查表 编程语言 框架和开发工具的速查表 单个文件包含一切你需要知道的东西 迁移学习前端低代码框架 通过 JSON 配置就能生成各种页面 技术面试最后反问面试官的话Machine Learning Yearning 中文版 机器学习训练秘籍 Andrew Ng 著越来越多的网站具有反爬虫特性 有的用图片隐藏关键数据 有的使用反人类的验证码 建立反反爬虫的代码仓库 通过与不同特性的网站做斗争 无恶意 提高技术 欢迎提交难以采集的网站 因工作原因 项目暂停 本项目收藏这些年来看过或者听过的一些不错的常用的上千本书籍 没准你想找的书就在这里呢 包含了互联网行业大多数书籍和面试经验题目等等 有人工智能系列 常用深度学习框架TensorFlow pytorch keras NLP 机器学习 深度学习等等 大数据系列 Spark Hadoop Scala kafka等 程序员必修系列 C C java 数据结构 linux 设计模式 数据库等等  人人影视bot 完全对接人人影视全部无删减资源Spring Cloud基础教程 持续连载更新中一个用于在 macOS 上平滑你的鼠标滚动效果或单独设置滚动方向的小工具 让你的滚轮爽如触控板阿里妈妈前端团队出品的开源接口管理工具RAP第二代超轻量级中文ocr 支持竖排文字识别 支持ncnn mnn tnn推理总模型仅4 7M 微信全平台 SDK Senparc Weixin for C 支持 NET Framework 及 NET Core NET 6 已支持微信公众号 小程序 小游戏 企业号 企业微信 开放平台 微信支付 JSSDK 微信周边等全平台 WeChat SDK for C 中文独立博客列表高效率 QQ 机器人支持库支持定制任何播放器SDK和控制层 OpenPower工作组收集汇总的医院开放数据Xray 基于 Nginx 的 VLESS XTLS 一键安装脚本 FlutterDemo合集 今天你fu了吗莫烦Python 中文AI教学中国特色 TabBar 一行代码实现 Lottie 动画TabBar 支持中间带 号的TabBar样式 自带红点角标 支持动态刷新 Flutter豆瓣客户端 Awesome Flutter Project 全网最1 %还原豆瓣客户端 首页 书影音 小组 市集及个人中心 一个不拉 img xuvip top douyademo mp4 基于SpringCloud2 1的微服务开发脚手架 整合了等 服务治理方面引入等 让项目开发快速进入业务开发 而不需过多时间花费在架构搭建上 持续更新中基于 Vue2 和 ECharts 封装的图表组件 SSR 去广告ACL规则 SS完整GFWList规则 Clash规则碎片 Telegram频道订阅地址和我一步步部署 kubernetes 集群搜集 整理 维护实用规则 中文自然语言处理相关资料基于SOA架构的分布式电商购物商城 前后端分离 前台商城 全家桶 后台管理系统等What happens when 的中文翻译 原仓库QMUI iOS 致力于提高项目 UI 开发效率的解决方案新型冠状病毒防疫信息收集平台告别枯燥 致力于打造 Python 实用小例子在线制作 sorry 为所欲为 的gifNodejs学习笔记以及经验总结 公众号 程序猿小卡 李宏毅 机器学习 笔记 在线阅读地址 Vue js 源码分析V部落 Vue SpringBoot实现的多用户博客管理平台 Android Signature V2 Scheme签名下的新一代渠道包打包神器Autoscroll Banner 无限循环图片 文字轮播器 多种编程语言实现 LeetCode 剑指 Offer 第 2 版 程序员面试金典 第 6 版 题解一套高质量的微信小程序 UI 组件库飞桨 官方模型库 包含多种学术前沿和工业场景验证的深度学习模型 中文 Python 笔记专门为刚开始刷题的同学准备的算法基地 没有最细只有更细 立志用动画将晦涩难懂的算法说的通俗易懂 版入门实例代码 实战教程 是一个高性能且低损耗的 goroutine 池 CVPR 2 21 论文和开源项目合集有 有  Python进阶 Intermediate Python 中文版 机器人视觉 移动机器人 VS SLAM ORB SLAM2 深度学习目标检测 yolov3 行为检测 opencv PCL 机器学习 无人驾驶后台管理系统解决方案创建在线课程 学术简历或初创网站  Chrome插件开发全攻略 配套完整Demo 欢迎clone体验QUANTAXIS 支持任务调度 分布式部署的 股票 期货 期权 港股 虚拟货币 数据 回测 模拟 交易 可视化 多账户 纯本地量化解决方案微信调试 各种WebView样式调试 手机浏览器的页面真机调试 便捷的远程调试手机页面 抓包工具 支持 HTTPS 无需USB连接设备 rich text 富文本编辑器 汉字拼音 hàn zì pīn yīn面向开发人员梳理的代码安全指南以撸代码的形式学习Python提供同花顺客户端 国金 华泰客户端 雪球的基金 股票自动程序化交易以及自动打新 支持跟踪 joinquant ricequant 模拟交易 和 实盘雪球组合 量化交易组件搜狐视频 sohu tv Redis私有云平台spring boot打造文件文档在线预览项目计算机基础 计算机网络 操作系统 数据库 Git 面试问题全面总结 包含详细的follow up question以及答案 全部采用 问题 追问 答案 的形式 即拿即用 直击互联网大厂面试 可用于模拟面试 面试前复习 短期内快速备战面试 首款微信 macOS 客户端撤回拦截与多开windows kernel exploits Windows平台提权漏洞集合权限管理系统 预览地址 47 1 4 7 138 loginpkuseg多领域中文分词工具一款完善的安全评估工具 支持常见 web 安全问题扫描和自定义 poc 使用之前务必先阅读文档零反射全动态Android插件框架Python入门网络爬虫之精华版分布式配置管理平台 中文 iOS Mac 开发博客列表周志华 机器学习 又称西瓜书是一本较为全面的书籍 书中详细介绍了机器学习领域不同类型的算法 例如 监督学习 无监督学习 半监督学习 强化学习 集成降维 特征选择等 记录了本人在学习过程中的理解思路与扩展知识点 希望对新人阅读西瓜书有所帮助  国内首个Spring Cloud微服务化RBAC的管理平台 核心采用前端采用d2 admin中台框架 记得上边点个star 关注更新Apache ECharts incubating 的微信小程序版本C 资源大全中文版 标准库 Web应用框架 人工智能 数据库 图片处理 机器学习 日志 代码分析等 由 开源前哨 和 CPP开发者 微信公号团队维护更新 stackoverflow上Java相关回答整理翻译 基于Google Flutter的WanAndroid客户端 支持Android和iOS 包括BLoC RxDart 国际化 主题色 启动页 引导页  本代码库是作者小傅哥多年从事一线互联网 Java 开发的学习历程技术汇总 旨在为大家提供一个清晰详细的学习教程 侧重点更倾向编写Java核心内容 如果本仓库能为您提供帮助 请给予支持 关注 点赞 分享 C 资源大全中文版 包括了 构建系统 编译器 数据库 加密 初中高的教程 指南 书籍 库等  NET m3u8 downloader 开源的命令行m3u8 HLS dash下载器 支持普通AES 128 CBC解密 多线程 自定义请求头等 支持简体中文 繁体中文和英文 English Supported 国内低代码平台从业者交流tcc transaction是TCC型事务java实现设计模式 Golang实现 研磨设计模式 读书笔记Vue数据可视化组件库 类似阿里DataV 大屏数据展示 提供SVG的边框及装饰 图表 水位图 飞线图等组件 简单易用 长期更新 React版已发布 自己动手做聊天机器人教程 RecyclerView侧滑菜单 Item拖拽 滑动删除Item 自动加载更多 HeaderView FooterView Item分组黏贴 腾讯物联网终端操作系统一个小巧 轻量的浏览器内核 用来取代wke和libcef包含美颜等4 余种实时滤镜相机 可拍照 录像 图片修改springboot 框架与其它组件结合如等用深度学习对对联  技术面试必备基础知识 Leetcode 计算机操作系统 计算机网络 系统设计 Java学习 面试指南 一份涵盖大部分 Java 程序员所需要掌握的核心知识 准备 Java 面试 首选 JavaGuide 用动画的形式呈现解LeetCode题目的思路 互联网 Java 工程师进阶知识完全扫盲 涵盖高并发 分布式 高可用 微服务 海量数据处理等领域知识mall项目是一套电商系统 包括前台商城系统及后台管理系统 基于SpringBoot MyBatis实现 采用Docker容器化部署 前台商城系统包含首页门户 商品推荐 商品搜索 商品展示 购物车 订单流程 会员中心 客户服务 帮助中心等模块 后台管理系统包含商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等模块  GitHub中文排行榜 帮助你发现高分优秀中文项目 更高效地吸收国人的优秀经验成果 榜单每周更新一次 敬请关注  算法面试 算法知识 针对小白的算法训练 还包括 1 阿里 字节 滴滴 百篇大厂面经汇总 2 千本开源电子书 3 百张思维导图 右侧来个 star 吧 English version supported 诊断利器Arthas教程 技术栈示例代码 快速简单上手教程 http下载工具 基于http代理 支持多连接分块下载阿里云计算平台团队出品 为监控而生的数据库连接池 企业级低代码平台 前后端分离架构强大的代码生成器让前后端代码一键生成 无需写任何代码 引领新的开发模式OnlineCoding 代码生成 手工MERGE 帮助Java项目解决7 %重复工作 让开发更关注业务 既能快速提高效率 帮助公司节省成本 同时又不失灵活性  下拉刷新 上拉加载 二级刷新 淘宝二楼智能下拉刷新框架 支持越界回弹 越界拖动 具有极强的扩展性 集成了几十种炫酷的Header和 Footer 该项目已成功集成 actuator 监控 admin 可视化监控 logback 日志 aopLog 通过AOP记录web请求日志 统一异常处理 json级别和页面级别 freemarker 模板引擎 thymeleaf 模板引擎 Beetl 模板引擎 Enjoy 模板引擎 JdbcTemplate 通用JDBC操作数据库 JPA 强大的ORM框架 mybatis 强大的ORM框架 通用Mapper 快速操作Mybatis PageHelper 通用的Mybatis分页插件 mybatis plus 快速操作Mybatis BeetlSQL 强大的ORM框架 u 微人事是一个前后端分离的人力资源管理系统 项目采用SpringBoot Vue开发  秒杀系统设计与实现 互联网工程师进阶与分析 To Be Top Javaer Java工程师成神之路循序渐进 学习博客Spring系列源码 mrbird cc快速 简单避免OOM的java处理Excel工具阿里巴巴 MySQL binlog 增量订阅 消费组件  一款优秀的开源博客发布应用 分布式任务调度平台XXL JOB 一款面向泛前端产品研发全生命周期的效率平台 面向云原生微服务的高可用流控防护组件 视频播放器支持弹幕 外挂字幕 支持滤镜 水印 gif截图 片头广告 中间广告 多个同时播放 支持基本的拖动 声音 亮度调节 支持边播边缓存 支持视频自带rotation的旋转 9 27 之类 重力旋转与手动旋转的同步支持 支持列表播放 列表全屏动画 视频加载速度 列表小窗口支持拖动 动画效果 调整比例 多分辨率切换 支持切换播放器 进度条小窗口预览 列表切换详情页面无缝播放 rtsp concat mpeg 又一个小商城 litemall Spring Boot后端 Vue管理员前端 微信小程序用户前端 Vue用户移动端基于Spring SpringMVC Mybatis分布式敏捷开发系统架构 提供整套公共微服务服务模块 集中权限管理 单点登录 内容管理 支付中心 用户管理 支持第三方登录 微信平台 存储系统 配置中心 日志分析 任务和通知等 支持服务治理 监控和追踪 努力为中小型企业打造全方位J2EE企业级开发解决方案 项目基于的前后端分离的后台管理系统 项目采用分模块开发方式 权限控制采用 RBAC 支持数据字典与数据权限管理 支持一键生成前后端代码 支持动态路由 史上最简单的Spring Cloud教程源码 CAT 作为服务端项目基础组件 提供了 Java C C Node js Python Go 等多语言客户端 已经在美团点评的基础架构中间件框架 MVC框架 RPC框架 数据库框架 缓存框架等 消息队列 配置系统等 深度集成 为美团点评各业务线提供系统丰富的性能指标 健康状况 实时告警等 spring boot 实践学习案例 是 spring boot 初学者及核心技术巩固的最佳实践 另外写博客 用 OpenWrite Spring Boot基础教程 Spring Boot 2 x版本连载中 帮助 Android App 进行组件化改造的路由框架 提高 Android UI 开发效率的 UI 库时间选择器 省市区三级联动 Luban 鲁班可能是最接近微信朋友圈的图片压缩算法 Gitee 最有价值开源项目 小而全而美的第三方登录开源组件 目前已支持Github Gitee 微博 钉钉 百度 Coding 腾讯云开发者平台 OSChina 支付宝 QQ 微信 淘宝 Google Facebook 抖音 领英 小米 微软 今日头条人人 华为 企业微信 酷家乐 Gitlab 美团 饿了么 推特 飞书 京东 阿里云 喜马拉雅 Amazon Slack和 Line 等第三方平台的授权登录 Login so easy 今日头条屏幕适配方案终极版 一个极低成本的 Android 屏幕适配方案  Banner 2 来了 Android广告图片轮播控件 内部基于ViewPager2实现 Indicator和UI都可以自定义  零代码 热更新 自动化 ORM 库 后端接口和文档零代码 前端 客户端 定制返回 JSON 的数据和结构一个涵盖六个专栏分布式消息队列 分布式事务的仓库 希望胖友小手一抖 右上角来个 Star 感恩 1 24Mybatis通用分页插件OkGo 3 震撼来袭 该库是基于 协议 封装了 OkHttp 的网络请求框架 比 Retrofit 更简单易用 支持 RxJava RxJava2 支持自定义缓存 支持批量断点下载管理和批量上传管理功能含 Flink 入门 概念 原理 实战 性能调优 源码解析等内容 涉及等内容的学习案例 还有 Flink 落地应用的大型项目案例 PVUV 日志存储 百亿数据实时去重 监控告警 分享 欢迎大家支持我的专栏 大数据实时计算引擎 Flink 实战与性能优化 安卓平台上的JavaScript自动化工具  ️一个整合了大量主流开源项目高度可配置化的 Android MVP 快速集成框架 Spring源码阅读大数据入门指南 android 4 4以上沉浸式状态栏和沉浸式导航栏管理 适配横竖屏切换 刘海屏 软键盘弹出等问题 可以修改状态栏字体颜色和导航栏图标颜色 以及不可修改字体颜色手机的适配 适用于一句代码轻松实现 以及对bar的其他设置 详见README 简书请参考 www jianshu com p 2a884e211a62业内为数不多致力于极致体验的超强全自研跨平台 windows android iOS 流媒体内核 通过模块化自由组合 支持实时RTMP推流 RTSP推流 RTMP播放器 RTSP播放器 录像 多路流媒体转发 音视频导播 动态视频合成 音频混音 直播互动 内置轻量级RTSP服务等 比快更快 业界真正靠谱的超低延迟直播SDK 1秒内 低延迟模式下2 4 ms DataX是阿里云DataWorks数据集成的开源版本 mall学习教程 架构 业务 技术要点全方位解析 mall项目 4 k star 是一套电商系统 使用现阶段主流技术实现 涵盖了等技术 采用Docker容器化部署 Android开源弹幕引擎 烈焰弹幕使 ～spring cloud vue oAuth2 全家桶实战 前后端分离模拟商城 完整的购物流程 后端运营平台 可以实现快速搭建企业级微服务项目 支持微信登录等三方登录  一个基于Spring Boot MyBatis的种子项目 用于快速构建中小型API RESTful API项目 强大 可定制 易扩展的 ViewPager 指示器框架 是的最佳替代品 支持角标 更支持在非ViewPager场景下使用 使用hide show 切换Fragment或使用se 27天成为Java大神安卓学习笔记 cim cross IM 适用于开发者的分布式即时通讯系统Android上一个优雅 万能自定义UI 仿iOS 支持垂直 水平方向切换 支持周视图 自定义周起始 性能高效的日历控件 支持热插拔实现的UI定制 支持标记 自定义颜色 农历 自定义月视图各种显示模式等 Canvas绘制 速度快 占用内存低 你真的想不到日历居然还可以如此优雅hsweb haʊs wɛb 是一个基于spring boot 2 x开发 首个使用全响应式编程的企业级后台管理系统基础项目  newbee mall 项目 新蜂商城 是一套电商系统 包括 newbee mall 商城系统及 newbee mall admin 商城后台管理系统 基于 Spring Boot 2 X 及相关技术栈开发 前台商城系统包含首页门户 商品分类 新品上线 首页轮播 商品推荐 商品搜索 商品展示 购物车 订单结算 订单流程 个人订单管理 会员中心 帮助中心等模块 后台管理系统包含数据面板 轮播图管理 商品管理 订单管理 会员管理 分类管理 设置等模块 mall swarm是一套微服务商城系统 采用了等核心技术 同时提供了基于Vue的管理后台方便快速搭建系统 mall swarm在电商业务的基础集成了注册中心 配置中心 监控中心 网关等系统功能 文档齐全 附带全套Spring Cloud教程 阅读是一款可以自定义来源阅读网络内容的工具 为广大网络文学爱好者提供一种方便 快捷舒适的试读体验 Spring Cloud基础教程 持续连载更新中阿里巴巴分布式数据库同步系统 解决中美异地机房  基于谷歌最新AAC架构 MVVM设计模式的一套快速开发库 整合OkRxJava Retrofit Glide等主流模块 满足日常开发需求 使用该框架可以快速开发一个高质量 易维护的Android应用 基于SpringCloud2 1的微服务开发脚手架 整合了等 服务治理方面引入等 让项目开发快速进入业务开发 而不需过多时间花费在架构搭建上 持续更新中基于SOA架构的分布式电商购物商城 前后端分离 前台商城 全家桶 后台管理系统等是 难得一见 的 Jetpack MVVM 最佳实践 在 以简驭繁 的代码中 对 视图控制器 乃至 标准化开发模式 形成正确 深入的理解 V部落 Vue SpringBoot实现的多用户博客管理平台 Android Signature V2 Scheme签名下的新一代渠道包打包神器即时通讯 IM 系统多种编程语言实现 LeetCode 剑指 Offer 第 2 版 程序员面试金典 第 6 版 题解专门为刚开始刷题的同学准备的算法基地 没有最细只有更细 立志用动画将晦涩难懂的算法说的通俗易懂 ansj分词 ict的真正java实现 分词效果速度都超过开源版的ict 中文分词 人名识别 词性标注 用户自定义词典 book 任阅 网络小说阅读器 3D翻页效果 txt pdf epub书籍阅读 Wifi传书 LeetCode刷题记录与面试整理mybatis generator界面工具 让你生成代码更简单更快捷Spring Cloud 学习案例 服务发现 服务治理 链路追踪 服务监控等 XPopup2 版本重磅来袭 2倍以上性能提升 带来可观的动画性能优化和交互细节的提升 功能强大 交互优雅 动画丝滑的通用弹窗 可以替代等组件 自带十几种效果良好的动画 支持完全的UI和动画自定义搜狐视频 sohu tv Redis私有云平台spring boot打造文件文档在线预览项目权限管理系统 预览地址 47 1 4 7 138 login零反射全动态Android插件框架分布式配置管理平台 通用 IM 聊天 UI 组件 已经同时支持 Android iOS RN 手把手教你整合最优雅SSM框架 SpringMVC Spring MyBatis换肤框架 极低的学习成本 极好的用户体验 一行 代码就可以实现换肤 你值得拥有  JVM 底层原理最全知识总结 国内首个Spring Cloud微服务化RBAC的管理平台 核心采用前端采用d2 admin中台框架 记得上边点个star 关注更新tcc transaction是TCC型事务java实现 RecyclerView侧滑菜单 Item拖拽 滑动删除Item 自动加载更多 HeaderView FooterView Item分组黏贴 包含美颜等4 余种实时滤镜相机 可拍照 录像 图片修改springboot 框架与其它组件结合如等安卓选择器类库 包括日期及时间选择器 可用于出生日期 营业时间等 单项选择器 可用于性别 民族 职业 学历 星座等 二三级联动选择器 可用于车牌号 基金定投日期等 城市地址选择器 分省级 地市级及区县级 数字选择器 可用于年龄 身高 体重 温度等 日历选日期择器 可用于酒店及机票预定日期 颜色选择器 文件及目录选择器等 Java工程师面试复习指南 本仓库涵盖大部分Java程序员所需要掌握的核心知识 整合了互联网上的很多优质Java技术文章 力求打造为最完整最实用的Java开发者学习指南 如果对你有帮助 给个star告诉我吧 谢谢  Android MVP 快速开发框架 做国内 示例最全面 注释最详细 使用最简单 代码最严谨 的 Android 开源 UI 框架几行代码快速集成二维码扫描功能MeterSphere 是一站式开源持续测试平台 涵盖测试跟踪 接口测试 性能测试 团队协作等功能 全面兼容 JMeter Postman Swagger 等开源 主流标准 记录各种学习笔记 算法 Java 数据库 并发 下一代Android打包工具 1 个渠道包只需要1 秒钟芋道 mall 商城 基于微服务的思想 构建在 B2C 电商场景下的项目实战 核心技术栈 是 Spring Boot Dubbo 未来 会重构成 Spring Cloud Alibaba Android 万能的等 支持多种Item类型的情况 lanproxy是一个将局域网个人电脑 服务器代理到公网的内网穿透工具 支持tcp流量转发 可支持任何tcp上层协议 访问内网网站 本地支付接口调试 ssh访问 远程桌面 目前市面上提供类似服务的有花生壳 TeamView GoToMyCloud等等 但要使用第三方的公网服务器就必须为第三方付费 并且这些服务都有各种各样的限制 此外 由于数据包会流经第三方 因此对数据安全也是一大隐患 技术交流QQ群 1 6742433 更优雅的驾车体验下载可以很简单 ️ 云阅 一款基于网易云音乐UI 使用玩架构开发的符合Google Material Design的Android客户端开源的 Material Design 豆瓣客户端一款针对系统PopupWindow优化的Popup库 功能强大 支持背景模糊 使用简单 你会爱上他的 PLDroidPlayer 是七牛推出的一款免费的适用于 Android 平台的播放器 SDK 采用全自研的跨平台播放内核 拥有丰富的功能和优异的性能 可高度定制化和二次开发 该项目已停止维护 9 Porn Android 客户端 突破游客每天观看1 次视频的限制 还可以下载视频 ️蓝绿 灰度 路由 限流 熔断 降级 隔离 追踪 流量染色 故障转移一本关于排序算法的 GitBook 在线书籍 十大经典排序算法 多语言实现 多种下拉刷新效果 上拉加载更多 可配置自定义头部广告位完全仿微信的图片选择 并且提供了多种图片加载接口 选择图片后可以旋转 可以裁剪成矩形或圆形 可以配置各种其他的参数SoloPi 自动化测试工具龙果支付系统 roncoo pay 是国内首款开源的互联网支付系统 拥有独立的账户体系 用户体系 支付接入体系 支付交易体系 对账清结算体系 目标是打造一款集成主流支付方式且轻量易用的支付收款系统 满足互联网业务系统打通支付通道实现支付收款和业务资金管理等功能 键盘面板冲突 布局闪动处理方案  咕泡学院实战项目 基于SpringBoot Dubbo构建的电商平台 微服务架构 商城 电商 微服务 高并发 kafka Elasticsearch停车场系统源码 停车场小程序 智能停车 Parking system 功能介绍 ①兼容市面上主流的多家相机 理论上兼容所有硬件 可灵活扩展 ②相机识别后数据自动上传到云端并记录 校验相机唯一id和硬件序列号 防止非法数据录入 ③用户手机查询停车记录详情可自主缴费 支持微信 支付宝 银行接口支付 支持每个停车场指定不同的商户进行收款 支付后出场在免费时间内会自动抬杆 ④支持app上查询附近停车场 导航 可用车位数 停车场费用 优惠券 评分 评论等 可预约车位 ⑤断电断网支持岗亭人员使用app可接管硬件进行停车记录的录入 技术架构 后端开发语言java 框架oauth2 spring 成长路线 但学到不仅仅是Java 业界首个支持渐进式组件化改造的Android组件化开源框架 支持跨进程调用SpringBoot2 从入门到实战 旨在打造在线最佳的 Java 学习笔记 含博客讲解和源码实例 包括 Java SE 和 Java WebJava诊断工具年薪百万互联网架构师课程文档及源码 公开部分 AndroidHttpCapture网络诊断工具 是一款Android手机抓包软件 主要功能包括 手机端抓包 PING DNS TraceRoute诊断 抓包HAR数据上传分享 你也可以看成是Android版的 Fiddler o 这可能是史上功能最全的Java权限认证框架 目前已集成 登录认证 权限认证 分布式Session会话 微服务网关鉴权 单点登录 OAuth2 踢人下线 Redis集成 前后台分离 记住我模式 模拟他人账号 临时身份切换 账号封禁 多账号认证体系 注解式鉴权 路由拦截式鉴权 花式token生成 自动续签 同端互斥登录 会话治理 密码加密 jwt集成 Spring集成 WebFlux集成 Android平台下的富文本解析器 支持Html和Markdown智能图片裁剪框架 自动识别边框 手动调节选区 使用透视变换裁剪并矫正选区 适用于身份证 名片 文档等照片的裁剪 俗名 可垂直跑 可水平跑的跑马灯 学名 可垂直翻 可水平翻的翻页公告 小马哥技术周报 Android Video Player 安卓视频播放器 封装模仿抖音并实现预加载 列表播放 悬浮播放 广告播放 弹幕 重学Java设计模式 是一本互联网真实案例实践书籍 以落地解决方案为核心 从实际业务中抽离出 交易 营销 秒杀 中间件 源码等22个真实场景 来学习设计模式的运用 欢迎关注小傅哥 微信 fustack 公众号 bugstack虫洞栈 博客 bugstack cnmybatis源码中文注释一款开源的GIF在线分享App 乐趣就要和世界分享 MPush开源实时消息推送系统在线云盘 网盘 OneDrive 云存储 私有云 对象存储 h5ai基于Spring Boot 2 x的一站式前后端分离快速开发平台XBoot 微信小程序 Uniapp 前端 Vue iView Admin 后端分布式限流 同步锁 验证码 SnowFlake雪花算法ID 动态权限 数据权限 工作流 代码生成 定时任务 社交账号 短信登录 单点登录 OAuth2开放平台 客服机器人 数据大屏 暗黑模式Guns基于SpringBoot 2 致力于做更简洁的后台管理系统 完美整合项目代码简洁 注释丰富 上手容易 同时Guns包含许多基础模块 用户管理 角色管理 部门管理 字典管理等1 个模块 可以直接作为一个后台管理系统的脚手架  Android 版本更新一个简洁而优雅的Android原生UI框架 解放你的双手 一套完整有效的android组件化方案 支持组件的组件完全隔离 单独调试 集成调试 组件交互 UI跳转 动态加载卸载等功能适用于Java和Android的快速 低内存占用的汉字转拼音库 Codes of my MOOC Course \\u003c我在慕课网上的课程 算法与数据结构 示例代码 包括C 和Java版本 课程的更多更新内容及辅助练习也将逐步添加进这个代码仓  Hope Boot 一款现代化的脚手架项目一个简单漂亮的SSM Spring SpringMVC Mybatis 博客系统根据Gson库使用的要求 将JSONObject格式的String 解析成实体B站 哔哩哔哩 Bilibili 自动签到投币工具 每天轻松获取65经验值 支持每日自动投币 银瓜子兑换硬币 领取大会员福利 大会员月底给自己充电等功能 呐 赶快和我一起成为Lv6吧 IJPay 让支付触手可及 封装了微信支付 QQ支付 支付宝支付 京东支付 银联支付 PayPal 支付等常用的支付方式以及各种常用的接口 不依赖任何第三方 mvc 框架 仅仅作为工具使用简单快速完成支付模块的开发 可轻松嵌入到任何系统里 右上角点下小星星  High quality pure Weex demo 网易严选 App 感受 Weex 开发Android 快速实现新手引导层的库 通过简洁链式调用 一行代码实现引导层的显示通过标签直接生成shape 无需再写shape xml 本库是一款基于RxJava2 Retrofit2实现简单易用的网络请求框架 结合android平台特性的网络封装库 采用api链式调用一点到底 集成cookie管理 多种缓存模式 极简https配置 上传下载进度显示 请求错误自动重试 请求携带token 时间戳 签名sign动态配置 自动登录成功后请求重发功能 3种层次的参数设置默认全局局部 默认标准ApiResult同时可以支持自定义的数据结构 已经能满足现在的大部分网络请求 Android BLE蓝牙通信库 基于Flink实现的商品实时推荐系统 flink统计商品热度 放入redis缓存 分析日志信息 将画像标签和实时记录放入Hbase 在用户发起推荐请求后 根据用户画像重排序热度榜 并结合协同过滤和标签两个推荐模块为新生成的榜单的每一个产品添加关联产品 最后返回新的用户列表 播放器基础库 专注于播放视图组件的高复用性和组件间的低耦合 轻松处理复杂业务 图片选择库 单选 多选 拍照 裁剪 压缩 自定义 包括视频选择和录制 DataX集成可视化页面 选择数据源即可一键生成数据同步任务 支持等数据源 批量创建RDBMS数据同步任务 集成开源调度系统 支持分布式 增量同步数据 实时查看运行日志 监控执行器资源 KILL运行进程 数据源信息加密等  Deprecated android 自定义日历控件 支持左右无限滑动 周月切换 标记日期显示 自定义显示效果跳转到指定日期一个通过动态加载本地皮肤包进行换肤的皮肤框架这是RedSpider社区成员原创与维护的Java多线程系列文章 一站式Apache Kafka集群指标监控与运维管控平台快速开发工具类收集 史上最全的开发工具类 欢迎Follow Fork Star后端技术总结 包括Java基础 JVM 数据库 mysql redis 计算机网络 算法 数据结构 操作系统 设计模式 系统设计 框架原理 最佳阅读地址Android源码设计模式分析项目可能是最好的支付SDK 停止维护 组件化综合案例 包含微信新闻 头条视频 美女图片 百度音乐 干活集中营 玩Android 豆瓣读书电影 知乎日报等等模块 架构模式 组件化阿里VLayout 腾讯X5 腾讯bugly 融合开发中需要的各种小案例 开源OA系统 码云GVP Java开源oa 企业OA办公平台 企业OA 协同办公OA 流程平台OA O2OA OA 支持国产麒麟操作系统和国产数据库 达梦 人大金仓 政务OA 军工信息化OA以Spring Cloud Netflix作为服务治理基础 展示基于tcc思想所实现的分布式事务解决方案一个帮助您完成从缩略视图到原视图无缝过渡转变的神奇框架 系统重构与迁移指南 手把手教你分析 评估现有系统 制定重构策略 探索可行重构方案 搭建测试防护网 进行系统架构重构 服务架构重构 模块重构 代码重构 数据库重构 重构后的架构守护版本检测升级 更新 库小说精品屋是一个多平台 web 安卓app 微信小程序 功能完善的屏幕自适应小说漫画连载系统 包含精品小说专区 轻小说专区和漫画专区 包括小说 漫画分类 小说 漫画搜索 小说 漫画排行 完本小说 漫画 小说 漫画评分 小说 漫画在线阅读 小说 漫画书架 小说 漫画阅读记录 小说下载 小说弹幕 小说 漫画自动采集 更新 纠错 小说内容自动分享到微博 邮件自动推广 链接自动推送到百度搜索引擎等功能 Android 徽章控件 致力于打造一款极致体验的 www wanandroid com 客户端 知识和美是可以并存的哦QAQn ≧ ≦ n 从源码层面 剖析挖掘互联网行业主流技术的底层实现原理 为广大开发者 “提升技术深度” 提供便利 目前开放 Spring 全家桶 Mybatis Netty Dubbo 框架 及 Redis Tomcat 中间件等Redis 一站式管理平台 支持集群的监控 安装 管理 告警以及基本的数据操作该项目不再维护 仅供学习参考专注批量推送的小而美的工具 目前支持 模板消息 公众号 模板消息 小程序 微信客服消息 微信企业号 企业微信消息 阿里云短信 阿里大于模板短信 腾讯云短信 云片网短信 E Mail HTTP请求 钉钉 华为云短信 百度云短信 又拍云短信 七牛云短信Android 平台开源天气 App 采用等开源库来实现 SpringBoot 相关漏洞学习资料 利用方法和技巧合集 黑盒安全评估 check listAndroid 权限请求框架 已适配 Android 11微信SDK JAVA 公众平台 开放平台 商户平台 服务商平台  QMQ是去哪儿网内部广泛使用的消息中间件 自2 12年诞生以来在去哪儿网所有业务场景中广泛的应用 包括跟交易息息相关的订单场景 也包括报价搜索等高吞吐量场景  Java 23种设计模式全归纳linux运维监控工具 支持系统信息 内存 cpu 温度 磁盘空间及IO 硬盘smart 系统负载 网络流量 进程等监控 API接口 大屏展示 拓扑图 端口监控 docker监控 日志文件监控 数据可视化 webSSH工具 堡垒机 跳板机 这可能是全网最好用的ViewPager轮播图 简单 高效 一行代码实现循环轮播 一屏三页任意变 指示器样式任你挑 一种简单有效的android组件化方案 支持组件的代码资源隔离 单独调试 集成调试 组件交互 UI跳转 生命周期等完整功能 一个强大 1 % 兼容 支持 AndroidX 支持 Kotlin并且灵活的组件化框架JPress 一个使用 Java 开发的建站神器 目前已经有 1 w 网站使用 JPress 进行驱动 其中包括多个政府机构 2 上市公司 中科院 红 字会等 分布式事务易用的轻量化网络爬虫 Android系统源码分析重构中一款免费的数据可视化工具 报表与大屏设计 类似于excel操作风格 在线拖拽完成报表设计 功能涵盖 报表设计 图形报表 打印设计 大屏设计等 永久免费 秉承“简单 易用 专业”的产品理念 极大的降低报表开发难度 缩短开发周期 节省成本 解决各类报表难题 Android Activity 滑动返回 支持微信滑动返回样式 横屏滑动返回 全屏滑动返回SpringBoot 基础教程 从入门到上瘾 基于2 M5制作 仿微信视频拍摄UI 基于ffmpeg的视频录制编辑Python 1 天从新手到大师 分享 GitHub 上有趣 入门级的开源项目中英文敏感词 语言检测 中外手机 电话归属地 运营商查询 名字推断性别 手机号抽取 身份证抽取 邮箱抽取 中日文人名库 中文缩写库 拆字词典 词汇情感值 停用词 反动词表 暴恐词表 繁简体转换 英文模拟中文发音 汪峰歌词生成器 职业名称词库 同义词库 反义词库 否定词库 汽车品牌词库 汽车零件词库 连续英文切割 各种中文词向量 公司名字大全 古诗词库 IT词库 财经词库 成语词库 地名词库 历史名人词库 诗词词库 医学词库 饮食词库 法律词库 汽车词库 动物词库 中文聊天语料 中文谣言数据 百度中文问答数据集 句子相似度匹配算法集合 bert资源 文本生成 摘要相关工具 cocoNLP信息抽取 2 21年最新总结 阿里 腾讯 百度 美团 头条等技术面试题目 以及答案 专家出题人分析汇总 AiLearning 机器学习 MachineLearning ML 深度学习 DeepLearning DL 自然语言处理 NLP123 6智能刷票 订票结巴中文分词 动手学深度学习 面向中文读者 能运行 可讨论 中英文版被全球175所大学采用教学 中文分词 词性标注 命名实体识别 依存句法分析 语义依存分析 新词发现 关键词短语提取 自动摘要 文本分类聚类 拼音简繁转换 自然语言处理微信个人号接口 微信机器人及命令行微信 三十行即可自定义个人号机器人 数据结构和算法必知必会的5 个代码实现JumpServer 是全球首款开源的堡垒机 是符合 4A 的专业运维安全审计系统 飞桨 核心框架 深度学习 机器学习高性能单机 分布式训练和跨平台部署 中国程序员容易发音错误的单词微信 跳一跳 Python 辅助 python模拟登陆一些大型网站 还有一些简单的爬虫 希望对你们有所帮助 ️ 如果喜欢记得给个star哦  网络爬虫实战 淘宝 京东 网易云 B站 123 6 抖音 笔趣阁 漫画小说下载 音乐电影下载等Python爬虫代理IP池 proxy pool wtfpython的中文翻译 施工结束 能力有限 欢迎帮我改进翻译提供多款 Shadowrocket 规则 带广告过滤功能 用于 iOS 未越狱设备选择性地自动翻墙  123 6 购票助手 支持集群 多账号 多任务购票以及 Web 页面管理 walle 瓦力 Devops开源项目代码部署平台一些非常有趣的python爬虫例子 对新手比较友好 主要爬取淘宝 天猫 微信 豆瓣 QQ等网站机器学习相关教程1 Chinese Word Vectors 上百种预训练中文词向量 网易云音乐命令行版本一款入门级的人脸 视频 文字检测以及识别的项目  编程随想 整理的 太子党关系网络 专门揭露赵国的权贵微信助手 1 每日定时给好友 女友 发送定制消息 2 机器人自动回复好友 3 群助手功能 例如 查询垃圾分类 天气 日历 电影实时票房 快递物流 PM2 5等 二维码生成器 支持 gif 动态图片二维码 阿布量化交易系统 股票 期权 期货 比特币 机器学习 基于python的开源量化交易 量化投资架构 book 中华新华字典数据库 包括歇后语 成语 词语 汉字  Git AWS Google 镜像 SS SSR VMESS节点行业研究报告的知识储备库中文翻译手写实现李航 统计学习方法 书中全部算法 Python 抖音机器人 论如何在抖音上找到漂亮小姐姐？ 迁移学习python爬虫教程系列 从 到1学习python爬虫 包括浏览器抓包 手机APP抓包 如 fiddler mitmproxy 各种爬虫涉及的模块的使用 如等 以及IP代理 验证码识别 Mysql MongoDB数据库的python使用 多线程多进程爬虫的使用 css 爬虫加密逆向破解 JS爬虫逆向 分布式爬虫 爬虫项目实战实例等Python脚本 模拟登录知乎 爬虫 操作excel 微信公众号 远程开机越来越多的网站具有反爬虫特性 有的用图片隐藏关键数据 有的使用反人类的验证码 建立反反爬虫的代码仓库 通过与不同特性的网站做斗争 无恶意 提高技术 欢迎提交难以采集的网站 因工作原因 项目暂停  人人影视bot 完全对接人人影视全部无删减资源莫烦Python 中文AI教学飞桨 官方模型库 包含多种学术前沿和工业场景验证的深度学习模型 轻量级人脸检测模型 百度云 百度网盘Python客户端 Python进阶 Intermediate Python 中文版 提供同花顺客户端 国金 华泰客户端 雪球的基金 股票自动程序化交易以及自动打新 支持跟踪 joinquant ricequant 模拟交易 和 实盘雪球组合 量化交易组件QUANTAXIS 支持任务调度 分布式部署的 股票 期货 期权 港股 虚拟货币 数据 回测 模拟 交易 可视化 多账户 纯本地量化解决方案INFO SPIDER 是一个集众多数据源于一身的爬虫工具箱 旨在安全快捷的帮助用户拿回自己的数据 工具代码开源 流程透明 支持数据源包括GitHub QQ邮箱 网易邮箱 阿里邮箱 新浪邮箱 Hotmail邮箱 Outlook邮箱 京东 淘宝 支付宝 中国移动 中国联通 中国电信 知乎 哔哩哔哩 网易云音乐 QQ好友 QQ群 生成朋友圈相册 浏览器浏览历史 123 6 博客园 CSDN博客 开源中国博客 简书 中文BERT wwm系列模型 Python入门网络爬虫之精华版中文 iOS Mac 开发博客列表Python网页微信APIpkuseg多领域中文分词工具自己动手做聊天机器人教程基于搜狗微信搜索的微信公众号爬虫接口用深度学习对对联 v2ray xray多用户管理部署程序各种脚本 关于 虾米 xiami com 百度网盘 pan baidu com 115网盘 115 com 网易音乐 music 163 com 百度音乐 music baidu com 36 网盘 云盘 yunpan cn 视频解析 flvxz com bt torrent ↔ magnet ed2k 搜索 tumblr 图片下载 unzip查看被删的微信好友定投改变命运 让时间陪你慢慢变富 onregularinvesting com 机器学习实战 Python3 kNN 决策树 贝叶斯 逻辑回归 SVM 线性回归 树回归Statistical learning methods 统计学习方法 第2版 李航 笔记 代码 notebook 参考文献 Errata lihang stock 股票系统 使用python进行开发 基于深度学习的中文语音识别系统京东抢购助手 包含登录 查询商品库存 价格 添加 清空购物车 抢购商品 下单 查询订单等功能莫烦Python 中文AI教学机器学习算法python实现新浪微博爬虫 用python爬取新浪微博数据的算法以及通用生成对抗网络图像生成的理论与实践研究 青岛大学开源 Online Judge QQ群 49671 125 admin qduoj comWeRoBot 是一个微信公众号开发框架 基于Django的博客系统 中文近义词 聊天机器人 智能问答工具包开源财经数据接口库巡风是一款适用于企业内网的漏洞快速应急 巡航扫描系统  番号大全 解决电脑 手机看电视直播的苦恼 收集各种直播源 电视直播网站知识图谱构建 自动问答 基于kg的自动问答 以疾病为中心的一定规模医药领域知识图谱 并以该知识图谱完成自动问答与分析服务 出处本地电影刮削与整理一体化解决方案自动化运维平台 CMDB CD DevOps 资产管理 任务编排 持续交付 系统监控 运维管理 配置管理 wukong robot 是一个简单 灵活 优雅的中文语音对话机器人 智能音箱项目 还可能是首个支持脑机交互的开源智能音箱项目 获取斗鱼 虎牙 哔哩哔哩 抖音 快手等 55 个直播平台的真实流媒体地址 直播源 和弹幕 直播源可在 PotPlayer flv js 等播放器中播放 宝塔Linux面板 简单好用的服务器运维面板农业知识图谱 AgriKG 农业领域的信息检索 命名实体识别 关系抽取 智能问答 辅助决策CODO是一款为用户提供企业多混合云 一站式DevOps 自动化运维 完全开源的云管理平台 自动化运维平台Web Pentesting Fuzz 字典 一个就够了  计算机网络 自顶向下方法 原书第6版 编程作业 Wireshark实验文档的翻译和解答 中文古诗自动作诗机器人 屌炸天 基于tensorflow1 1 api 正在积极维护升级中 快star 保持更新 PyQt Examples PyQt各种测试和例子 PyQt4 PyQt5海量中文预训练ALBERT模型汉字转拼音 pypinyin 数据结构与算法 leetcode lintcode题解  Pytorch模型训练实用教程 中配套代码实时获取新浪 腾讯 的免费股票行情 集思路的分级基金行情Python爬虫 Flask网站 免费ShadowSocks账号 ssr订阅 json 订阅实战 多种网站 电商数据爬虫 包含 淘宝商品 微信公众号 大众点评 企查查 招聘网站 闲鱼 阿里任务 博客园 微博 百度贴吧 豆瓣电影 包图网 全景网 豆瓣音乐 某省药监局 搜狐新闻 机器学习文本采集 fofa资产采集 汽车之家 国家统计局 百度关键词收录数 蜘蛛泛目录 今日头条 豆瓣影评 携程 小米应用商店 安居客 途家民宿 ️ ️ ️ 微信爬虫展示项目 SQL 审核查询平台团子翻译器 个人兴趣制作的一款基于OCR技术的翻译器自动化运维平台 代码及应用部署CI CD 资产管理CMDB 计划任务管理平台 SQL审核 回滚 任务调度 站内WIKISource Code Security Audit 源代码安全审计 Exphub 漏洞利用脚本库 包括的漏洞利用脚本 最新添加我的自学笔记 终身更新 当前专注System基础 MLSys 使用机器学习算法完成对123 6验证码的自动识别Python 开源项目之 自学编程之路 保姆级教程 AI实验室 宝藏视频 数据结构 学习指南 机器学习实战 深度学习实战 网络爬虫 大厂面经 程序人生 资源分享 中文文本分类基于pytorch 开箱即用 根据网易云音乐的歌单 下载flac无损音乐到本地腾讯优图高精度双分支人脸检测器文本纠错等模型实现 开箱即用 3 天掌握量化交易 持续更新 中文分词 词性标注 命名实体识别 依存句法分析 新词发现 关键词短语提取 自动摘要 文本分类聚类 拼音简繁 自然语言处理中文公开聊天语料库豆瓣读书的爬虫总结梳理自然语言处理工程师 NLP 需要积累的各方面知识 包括面试题 各种基础知识 工程能力等等 提升核心竞争力中文自然语言处理数据集 平时做做实验的材料 欢迎补充提交合并 一个可以自己进行训练的中文聊天机器人 根据自己的语料训练出自己想要的聊天机器人 可以用于智能客服 在线问答 智能聊天等场景 目前包含seq2seq seqGAN版本 tf2 版本 pytorch版本 股票量化框架 支持行情获取以及交易微博爬虫 持续维护  Bilibili 用户爬虫 deepin源移植 Debian Ubuntu上最快的QQ 微信安装方式 新闻网页正文通用抽取器 Beta 版  flag on post 自动更新域名解析到本机IP 支持dnspod 阿里DNS CloudFlare 华为云 DNSCOM 本项目针对字符型图片验证码 使用tensorflow实现卷积神经网络 进行验证码识别 owllook 小说搜索引擎中文语言理解测评基准python中文库 python人工智能大数据自动化接口测试开发 书籍下载及python库汇总china testing github io 2 19新型冠状病毒疫情时间序列数据仓库Python 黑魔法手册单阶段通用目标检测器一个拍照做题程序 输入一张包含数学计算题的图片 输出识别出的数学计算式以及计算结果 video download B站视频下载中文命名实体识别 TensorFlow Python 中文数据结构和算法教程 验证码识别 训练Python爬虫实战 模拟登陆各大网站 包含但不限于 滑块验证 拼多多 美团 百度 bilibili 大众点评 淘宝 如果喜欢请start ️学无止下载器 慕课下载器 Mooc下载 慕课网下载 中国大学下载 爱课程下载 网易云课堂下载 学堂在线下载 超星学习通下载 支持视频 课件同时下载一个高级web目录 文件扫描工具 功能将会强于DirBuster Dirsearch cansina 御剑 搜索所有中文NLP数据集 附常用英文NLP数据集中文实体识别与关系提取2 19新型冠状病毒疫情实时爬虫及github release archive以及项目文件的加速项目安卓应用安全学习抓取大量免费代理 ip 提取有效 ip 使用RoBERTa中文预训练模型 RoBERTa for Chinese 用于训练中英文对话系统的语料库敏感词过滤的几种实现 某1w词敏感词库简单易用的Python爬虫框架 QQ交流群 59751 56 使用Bert ERNIE 进行中文文本分类为 CSAPP 视频课程提供字幕 翻译 PPT Lab PyTorch 官方中文教程包含 6 分钟快速入门教程 强化教程 计算机视觉 自然语言处理 生成对抗网络 强化学习 欢迎 Star Fork 兜哥出品 \\u003c一本开源的NLP入门书籍 图像翻译 条件GAN AI绘画用Resnet1 1 GPT搭建一个玩王者荣耀的AI各种漏洞poc Exp的收集或编写斗地主AIVulmap 是一款 web 漏洞扫描和验证工具 可对 webapps 进行漏洞扫描 并且具备漏洞验证功能提供超過 5 個金融資料 台股為主 每天更新 finmind github io 数据接口 百度 谷歌 头条 微博指数 宏观数据 利率数据 货币汇率 千里马 独角兽公司 新闻联播文字稿 影视票房数据 高校名单 疫情数据 PyOne 一款给力的onedrive文件管理 分享程序 使用开发的用于迅速搭建并使用 WebHook 进行自动化部署和运维 支持 Github GitLab Gogs GitOsc 跟我一起写Makefile重制版 python自动化运维 技术与最佳实践 书中示例及案例源码自然语言处理实验 sougou数据集 TF IDF 文本分类 聚类 词向量 情感识别 关系抽取等微信公众平台 Python 开发包 DEPRECATED Weblogic一键漏洞检测工具 V1 5 更新时间 2 2 73 完备优雅的微信公众号接口 原生支持同步 协程使用 本程序旨在为安全应急响应人员对Linux主机排查时提供便利 实现主机侧Checklist的自动全面化检测 根据检测结果自动数据聚合 进行黑客攻击路径溯源 PyCharm 中文指南 安装 破解 效率 技巧类似按键精灵的鼠标键盘录制和自动化操作 模拟点击和键入GPT2 for Chinese chitchat 用于中文闲聊的GPT2模型 实现了DialoGPT的MMI思想 中华人民共和国国家标准 GB T 226 行政区划代码基于python的量化交易平台中文语音识别基于 OneBot 标准的 Python 异步 QQ 机器人框架Real World Masked Face Dataset 口罩人脸数据集 Vulfocus 是一个漏洞集成平台 将漏洞环境 docker 镜像 放入即可使用 开箱即用 谷歌 百度 必应图片下载 基于方面的情感分析 使用PyTorch实现  深度学习与计算机视觉 配套代码ART环境下自动化脱壳方案利用网络上公开的数据构建一个小型的证券知识图谱 知识库中文资源精选 官方网站 安装教程 入门教程 视频教程 实战项目 学习路径 QQ群 167122861 公众号 磐创AI 微信群二维码 www tensorflownews com Pre Trained Chinese XLNet 中文XLNet预训练模型 新浪微博Python SDK有关burpsuite的插件 非商店 文章以及使用技巧的收集 此项目不再提供burpsuite破解文件 如需要请在博客mrxn net下载Python3编写的CMS漏洞检测框架基于django的工作流引擎 工单 ️ 哔哩云 不支持任意文件的全速上传与下载微信SDK 包括微信支付 微信公众号 微信登陆 微信消息处理等中文自然语言理解堡垒机 云桌面自动化运维 审计 录像 文件管理 sftp上传 实时监控 录像回放 网页版rz sz上传下载 动态口令 django转换中国知网 CAJ 格式文献为 PDF 佛系转换 成功与否 皆是玄学 HanLP作者的新书 自然语言处理入门 详细笔记 业界良心之作 书中不是枯燥无味的公式罗列 而是用白话阐述的通俗易懂的算法模型 从基本概念出发 逐步介绍中文分词 词性标注 命名实体识别 信息抽取 文本聚类 文本分类 句法分析这几个热门问题的算法原理与工程实现  Python3 网络爬虫实战 部分含详细教程 猫眼 腾讯视频 豆瓣 研招网 微博 笔趣阁小说 百度热点 B站 CSDN 网易云阅读 阿里文学 百度股票 今日头条 微信公众号 网易云音乐 拉勾 有道 unsplash 实习僧 汽车之家 英雄联盟盒子 大众点评 链家 LPL赛程 台风 梦幻西游 阴阳师藏宝阁 天气 牛客网 百度文库 睡前故事 知乎 Wish微信公众号文章的爬虫 Python Web开发实战 书中源码一直可用的GoAgent 会定时扫描可用的google gae ip 提供可自动化获取ip运行的版本层剪枝 通道剪枝 知识蒸馏 中文命名实体识别 包括多种模型 HMM CRF BiLSTM BiLSTM CRF的具体实现  The Way to Go 中文译本 中文正式名 Go 入门指南 谢谢 Solutions to LeetCode by Go 1 % test coverage runtime beats 1 % LeetCode 题解一款轻量级 高性能 功能强大的内网穿透代理服务器 支持tcp udp socks5 http等几乎所有流量转发 可用来访问内网网站 本地支付接口调试 ssh访问 远程桌面 内网dns解析 内网socks5代理等等 并带有功能强大的web管理端 Go语言高级编程 开源图书 涵盖CGO Go汇编语言 RPC实现 Protobuf插件实现 Web框架实现 分布式系统等高阶主题 完稿 是一个跨平台的强加密无特征的代理软件 零配置 算法模板 最科学的刷题方式 最快速的刷题路径 你值得拥有 百度网盘不限速客户端 golang qt5 跨平台图形界面是golang实现的高性能http https websocket tcp socks5代理服务器 支持内网穿透 链式代理 通讯加密 夜读 通过 bilibili 在线直播的方式分享 Go 相关的技术话题 每天大家在微信 telegram Slack 上及时沟通交流编程技术话题  支持多家云存储的云盘系统 这里是写博客的地方 Halfrost Field 冰霜之地Lantern官方版本下载 蓝灯 翻墙 代理 科学上网 外网 加速器 梯子 路由基于gin vue搭建的后台管理系统框架 集成jwt鉴权 权限管理 动态路由 分页封装 多点登录拦截 资源权限 上传下载 代码生成器 表单生成器 通用工作流等基础功能 五分钟一套CURD前后端代码 目VUE3版本正在重构 欢迎issue和pr 分布式爬虫管理平台 支持任何语言和框架Golang标准库 对于程序员而言 标准库与语言本身同样重要 它好比一个百宝箱 能为各种常见的任务提供完美的解决方案 以示例驱动的方式讲解Golang的标准库 天用Go动手写 从零实现系列是一个高性能且低损耗的 goroutine 池 有 有 设计模式 Golang实现 研磨设计模式 读书笔记Golang实现的基于beego框架的接口在线文档管理系统高性能开源RTSP流媒体服务器 基于go语言研发 维护和优化 RTSP推模式转发 RTSP拉模式转发 是一个高性能 轻量级 非阻塞的事件驱动 Go 网络框架 基于Gin Vue Element UI的前后端分离权限管理系统脚手架 包含了 多租户的支持 基础用户管理功能 jwt鉴权 代码生成器 RBAC资源控制 表单构建 定时任务等 3分钟构建自己的中后台项目 文档蓝鲸智云配置平台 BlueKing CMDB 今日热榜 一个获取各大热门网站热门头条的聚合网站 使用Go语言编写 多协程异步快速抓取信息 预览 mo fish一条命令离线安装高可用kubernetes 3min装完 7 M 1 年证书 生产环境稳如老狗阿里巴巴开源的一款简单易用 功能强大的混沌实验注入工具  Go语言四十二章经 详细讲述Go语言规范与语法细节及开发中常见的误区 通过研读标准库等经典代码设计模式 启发读者深刻理解Go语言的核心思维 进入Go语言开发的更高阶段  ️一个轻巧的网络混淆代理 基于Golang轻量级TCP并发服务器框架定时任务管理系统KubeOperator 是一个开源的轻量级 Kubernetes 发行版 专注于帮助企业规划 部署和运营生产级别的 K8s 集群 本系统是集工单统计 任务钩子 权限管理 灵活配置流程与模版等等于一身的开源工单系统 当然也可以称之为工作流引擎 致力于减少跨部门之间的沟通 自动任务的执行 提升工作效率与工作质量 减少不必要的工作量与人为出错率 Go实现的Trojan代理 支持多路复用 路由功能 CDN中转 Shadowsocks混淆插件 多平台 无依赖 Go语法树入门 开启自制编程语言和编译器之旅 开源免费图书 Go语言进阶 掌握抽象语法树 Go语言AST 凹语言 一款可全平台运行的浏览器数据导出解密工具 Golang相关 审稿进度8 % Go语法 Go并发思想 Go与web开发 Go微服务设施等Jupiter是斗鱼开源的面向服务治理的Golang微服务框架Elasticsearch 可视化DashBoard 支持Es监控 实时搜索 Index template快捷替换修改 索引列表信息查看 SQL converts to DSL等  从问题切入 串连 Go 语言相关的所有知识 融会贯通 golang design go questionsWeChat SDK for Go 微信SDK 简单 易用 go fastdfs 是一个简单的分布式文件系统 私有云存储 具有无中心 高性能 高可靠 免维护等优点 支持断点续传 分块上传 小文件合并 自动同步 自动修复 Mastering GO 中文译本 玩转 GO 云原生且易用的应用管理平台 Go Web 基础 是一套针对 Google 出品的 Go 语言的视频语音教程 主要面向完成 Go 编程基础 教程后希望进一步了解有关 Go Web 开发的学习者 中文名 悟空 API 网关 是一个基于 Golang开发的微服务网关 能够实现高性能 HTTP API 转发 服务编排 多租户管理 API 访问权限控制等目的 拥有强大的自定义插件系统可以自行扩展 并且提供友好的图形化配置界面 能够快速帮助企业进行 API 服务治理 提高 API 服务的稳定性和安全性  集合多家 API 的新一代图床MIT课程 Distributed Systems 学习和翻译Go语言圣经中文版 只接收PR Issue请提交到golang china gopl zh trojan多用户管理部署程序 支持web页面管理BookStack 基于MinDoc 使用Beego开发的在线文档管理系统 功能类似Gitbook和看云 weixin wechat 微信公众平台 微信企业号 微信商户平台 微信支付 go golang sdk  蓝眼云盘 Eyeblue Cloud Storage 语言高性能编程 Go 语言陷阱 Gotchas Traps 使用 XMind 记录 Linux 操作系统 网络 C Golang 以及数据库的一些设计cqhttp的golang实现 轻量 原生跨平台 mqant是一款基于Golang语言的简洁 高效 高性能的分布式微服务框架基于react node js go开发的微商城 含微信小程序 MM Wiki 一个轻量级的企业知识分享与团队协同软件 可用于快速构建企业 Wiki 和团队知识分享平台 部署方便 使用简单 帮助团队构建一个信息共享 文档管理的协作环境 Go 语言中文网 Golang中文社区 Go语言学习园地 源码基于 Gin 进行模块化设计的 API 框架 封装了常用功能 使用简单 致力于进行快速的业务研发 比如 支持 cors 跨域 jwt 签名验证 zap 日志收集 panic 异常捕获 trace 链路追踪 prometheus 监控指标 swagger 文档生成 viper 配置文件解析 gorm 数据库组件 gormgen 代码生成工具 graphql 查询语言 errno 统一定义错误码 gRPC 的使用 等等 syncd是一款开源的代码部署工具 它具有简单 高效 易用等特点 可以提高团队的工作效率 一款由 YSRC 开源的主机入侵检测系统golang面试题集合这是一个可以识别视频语音自动生成字幕SRT文件的开源 Windows GUI 软件工具 一款内网综合扫描工具 方便一键自动化 全方位漏扫扫描 是一个用于在两个redis之间同步数据的工具 满足用户非常灵活的同步 迁移需求 Overlord是哔哩哔哩基于Go语言编写的memcache和redis cluster的代理及集群管理功能 致力于提供自动化高可用的缓存服务解决方案 Stack RPC 中文示例 教程 资料 源码解读ICMP流量伪装转发工具Freedom是一个基于六边形架构的框架 可以支撑充血的领域模型范式  Go2编程指南 开源图书 重点讲解Go2新特性 以及Go1教程中较少涉及的特性语言高性能分词golang写的IM服务器 服务组件形式  结巴 中文分词的Golang版本xorm是一个简单而强大的Go语言ORM库 通过它可以使数据库操作非常简便 本库是基于原版xorm的定制增强版本 为xorm提供类似ibatis的配置文件及动态SQL支持 支持AcitveRecord操作一个 Go 语言实现的快速 稳定 内嵌的 k v 数据库 高性能表格数据导出器基于Golang的开源社区系统 版本网易云音乐ncm文件格式转换 go 实现的压测工具 ab locust Jmeter压测工具介绍 单台机器1 w连接压测实战 抓包截取项目中的数据库请求并解析成相应的语句  Go专家编程 Go语言快速入门 轻松进阶 \\u003c\\u003c自己动手写docker 源码Go 每日一库kunpeng是一个Golang编写的开源POC框架 库 以动态链接库的形式提供各种语言调用 通过此项目可快速开发漏洞检测类的系统 vue js element框架 golang beego框架 开发的运维发布系统 支持git jenkins版本发布 go ssh BT两种文件传输方式选择 支持部署前准备任务和部署后任务钩子函数 Go 从入门到实战 学习笔记 从零开始学 Go Gin 框架 基本语法包括 26 个Demo Gin 框架包括 Gin 自定义路由配置 Gin 使用 Logrus 进行日志记录 Gin 数据绑定和验证 Gin 自定义错误处理 Go gRPC Hello World 持续更新中 Go 学习之路 Go 开发者博客 Go 微信公众号 Go 学习资料 文档 书籍 视频 微信 WeChat 支付宝 AliPay 的Go版本SDK 极简 易用的聚合支付SDK Go by Example 通过例子学 GolangPPGo Job是一款可视化的 多人多权限的 一任务多机执行的定时任务管理系统 采用golang开发 安装方便 资源消耗少 支持大并发 可同时管理多台服务器上的定时任务 Golang实现的IP代理池是一款用Go语言开发的web应用框架 API特性类似于Tornado并且拥有比Tornado更好的性能  自己动手写Java虚拟机 随书源代码支付宝 AliPay SDK for Go 集成简单 功能完善 持续更新 支持公钥证书和普通公钥进行签名和验签  ARCHIVED Geph 迷霧通帮助你将本地端口暴露在外网 支持TCP UDP 当然也支持HTTP 深入Go并发编程研讨课无状态子域名爆破工具手机号码归属地信息库 手机号归属地查询 phone dat 最后更新 2 21年 6月 golang基于websocket单台机器支持百万连接分布式聊天 IM 系统基于mongodb oplog的集群复制工具 可以满足迁移和同步的需求 进一步实现灾备和多活功能 Gin Gorm开发Golang API快速开发脚手架简单可信赖的任务管理工具Go语言实例教程从入门到进阶 包括基础库使用 设计模式 面试易错点 工具类 对接第三方等授权框架简体中文翻译 自动抓取tg频道 订阅地址 公开互联网上的ss ssr vmess trojan节点信息 聚合去重后提供节点列表轻量级 go 业务框架  哪吒监控 一站式轻监控轻运维系统 支持系统状态 TCP Ping 监控报警 命令批量执行和计划任务 Go 语言官方教程中文版工程师知识管理系统 基于golang go语言 beego框架 每个行业都有自己的知识管理系统 engineercms旨在为土木工程师们打造一款适用的基于web的知识管理系统 它既可以用于管理个人的项目资料 也可以用于管理项目团队资料 它既可以运行于个人电脑 也可以放到服务器上 支持提取码分享文件 onlyoffice实时文档协作 直接在线编辑dwg文件 office文档 在线利用mindoc创作你的书籍 阅览PDF文件 通用的业务流程设置 手机端配套小程序 微信搜索“珠三角设代”或“青少儿书画”即可呼出小程序 边界打点后的自动化渗透工具一个集审核 执行 备份及生成回滚语句于一身的MySQL运维工具汉字转拼音 Go资源精选中文版 含中文图书大全 语言实现的 Redis 服务器和分布式集群 超全golang面试题合集 golang学习指南 golang知识图谱 入门成长路线 一份涵盖大部分golang程序员所需要掌握的核心知识 常用第三方库 mysql mq es redis等 机器学习库 算法库 游戏库 开源框架 自然语言处理nlp库 网络库 视频库 微服务框架 视频教程 音频音乐库 图形图片库 物联网库 地理位置信息 嵌入式脚本库 编译器库 数据库 金融库 电子邮件库 电子书籍 分词 数据结构 设计模式 去html tag标签等 go学习 go面试go语言扩展包 收集一些常用的操作函数 辅助更快的完成开发工作 并减少重复代码百灵快传 基于Go语言的高性能 手机电脑超大文件传输神器 局域网共享文件服务器 LAN large file transfer tool 一个基于云存储的网盘系统 用于自建私人网盘或企业网盘 go分布式服务器 基于内存mmo个人博客微信小程序服务端 SDK for Golang 控制台颜色渲染工具库 支持16色 256色 RGB色彩渲染输出 使用类似于 Print Sprintf 兼容并支持 Windows 环境的色彩渲染基于 IoC 的 Go 后端一站式开发框架 v2ray web manager 是一个v2ray的面板 也是一个集群的解决方案 同时增加了流量控制 账号管理 限速等功能 key admin panel web cluster 集群 proxyServerScan一款使用Golang开发的高并发网络扫描 服务探测工具 是http client领域的瑞士军刀 小巧 强大 犀利 具体用法可看文档 如使用迷惑或者API用得不爽都可提issuesTcpRoute TCP 层的路由器 对于 TCP 连接自动从多个线路 电信 联通 移动 多个域名解析结果中选择最优线路 Bifrost 面向生产环境的 MySQL 同步到Redis MongoDB ClickHouse MySQL等服务的异构中间件应用网关 提供快速 安全的应用交付 身份认证 WAF CC HTTPS以及ACME自动证书 A telegram bot for rss reader 一个支持应用内阅读的 Telegram RSS Bot RESTful API 文档生成工具 支持和 Ruby 等大部分语言 基于gin gorm开发的个人博客项目基于Go语言的国密SM2 SM3 SM4算法库 Golang 设计模式一个阿里云盘列表程序 一款小巧的基于Go构建的开发框架 可以快速构建API服务或者Web网站进行业务开发 遵循SOLID设计原则并发编程实战 第2版 Go 学习 Go 进阶 Go 实用工具类 Go kit Go Micro 微服务实践 Go 推送基于DDD的o2o的业务模型及基础 使用Golang gRPC Thrift实现Sharingan 写轮眼 是一个基于golang的流量录制回放工具 适合项目重构 回归测试等 百度云网盘爬虫基于beego的进销存系统 TeaWeb 可视化的Web代理服务 DEMO teaos cn  白帽子安全开发实战 配套代码抖音推荐 搜索页视频列表视频爬虫方案 基于app 虚拟机或真机 相关技术 golang adb一款甲方资产巡航扫描系统 系统定位是发现资产 进行端口爆破 帮助企业更快发现弱口令问题 主要功能包括 资产探测 端口爆破 定时任务 管理后台识别 报表展示提供微信终端版本 微信命令行版本聊天功能 微信机器人 ️ 互联网最全大厂技术分享PPT 持续更新中 各大技术交流会 活动资料汇总 如 QCon 全球运维技术大会 GDG 全球技术领导力峰会 大前端大会 架构师峰会 敏捷开发DevOps OpenResty Elastic 欢迎 PR Issues日本麻将助手 牌效 防守 记牌 支持雀魂 天凤 开源客服系统GO语言开发GO FLY 免费客服系统一个查询IP地理信息和CDN服务提供商的离线终端工具 是一个用于系统重构 系统迁移和系统分析的瑞士军刀 它可以分析代码中的测试坏味道 模块化分析 行数统计 分析调用与依赖 Git 分析以及自动化重构等 一个直播录制工具Mastering Go 第二版中文版来袭 渗透测试情报收集工具分布式定时任务调度平台高度模块化 遵循 KISS原则的区块链开发框架golang版本的hangout 希望能省些内存 使用了自己写的Kafka lib 虚 不过我们在生产环境已经使用近1年 kafka 版本从 9 1到2 都在使用 目前情况稳定 吞吐量在每天2 亿条以上 Go 语言 Web 应用开发系列教程 从新手到双手残废iris 框架的后台api项目简单好用的DDNS 自动更新域名解析到公网IP 支持阿里云 腾讯云dnspod Cloudflare 华为云  自己动手实现Lua 随书源代码php直播go直播 短视频 直播带货 仿比心 猎游 tt语音聊天 美女约玩 陪玩系统源码开黑 约玩源码 社区开源 云原生的多云和混合云融合平台 Jiajun的编程随想Golang语言社区 腾讯课堂 网易云课堂 字节教育课程PPT及代码基于GF Go Frame 的后台管理系统带你了解一下Golang的市场行情mysql表结构自动同步工具 目前只支持字段 索引的同步 分区等高级功能暂不支持 基于Kubernetes的PaaS平台流媒体NetFlix解锁检测脚本稳定分支2 9 X 版本已更新 由 Golang语言游戏服务器 维护 全球服游戏服务器及区域服框架 目前协议支持websocket KCP TCP及RPC 采用状态同步 帧同步内测 愿景 打造MMO多人竞技游戏框架 功能持续更新中 基于 Golang 类似知乎的私有部署问答应用 包含问答 评论 点赞 管理后台等功能全新的开源漏洞测试框架 实现poc在线编辑 运行 批量测试 使用文档 XAPI MANAGER 专业实用的开源接口管理平台 为程序开发者提供一个灵活 方便 快捷的API管理工具 让API管理变的更加清晰 明朗 如果你觉得xApi对你有用的话 别忘了给我们点个赞哦 qq协议的golang实现 移植于miraigo版本极简工作流引擎全平台Go开源内网渗透扫描器框架 Windows Linux Mac内网渗透 使用它可轻松一键批量探测C段 B段 A段存活主机 高危漏洞检测MS17 1 SmbGhost 远程执行SSH Winrm 密码爆破端口扫描服务识别PortScan指纹识别多网卡主机 端口扫描服务识别PortScan iikira BaiduPCS Go原版基础上集成了分享链接 秒传链接转存功能 e签宝安全团队积累十几年的安全经验 都将对外逐步开放 首开的Ehoney欺骗防御系统 该系统是基于云原生的欺骗防御系统 也是业界唯一开源的对标商业系统的产品 欺骗防御系统通过部署高交互高仿真蜜罐及流量代理转发 再结合自研密签及诱饵 将攻击者攻击引导到蜜罐中达到扰乱引导以及延迟攻击的效果 可以很大程度上保护业务的安全 护网必备良药漂亮的Go语言通用后台管理框架 包含计划任务 MySQL管理 Redis管理 FTP管理 SSH管理 服务器管理 Caddy配置 云存储管理等功能  微信支付 WeChat Pay SDK for Golang用于监控系统的日志采集agent 可无缝对接open falcon阿里巴巴mysql数据库binlog的增量订阅 消费组件 Canal 的 go 客户端 github com alibaba canal 用于比较2个redis数据是否一致 支持单节点 主从 集群版 以及多种proxy 支持同构以及异构对比 redis的版本支持2 x 5 x 使用go micro微服务实现的在线电影院订票系统后端一站式微服务框架 提供API web websocket RPC 任务调度 消息消费服务器红蓝对抗跨平台远控工具Interchain protocol 跨链协议简单易用 足够轻量 性能好的 Golang 库一个go echo vue 开发的快速 简洁 美观 前后端分离的个人博客系统 blog 也可方便二次开发为CMS 内容管理系统 和各种企业门户网站 正在更新权限管理 hauth项目 不是一个前端or后台框架 而是一个集成权限管理 菜单资源管理 域管理 角色管理 用户管理 组织架构管理 操作日志管理等等的快速开发平台． hauth是一个基础产品 在这个基础产品上 根据业务需求 快速的开发应用服务．账号 admin 密码 123456通用的数据验证与过滤库 使用简单 内置大部分常用验证 过滤器 支持自定义验证器 自定义消息 字段翻译 CTF AWD Attack with Defense 线下赛平台 AWD platform 欢迎 Star 蓝鲸智云容器管理平台 BlueKing Container Service 程序员如何优雅的挣零花钱 2 版 升级为小书了 一个 PHP 微信 SDKAV 电影管理系统 avmoo javbus javlibrary 爬虫 线上 AV 影片图书馆 AV 磁力链接数据库ThinkPHP Framework 十年匠心的高性能PHP框架 最全的前端资源汇总仓库 包括前端学习 开发资源 求职面试等 多语言多货币多入口的开源电商 B2C 商城 支持移动端vue app html5 微信小程序微店 微信小程序商城等可能是我用过的最优雅的 Alipay 和 WeChat 的支付 SDK 扩展包了 基于词库的中文转拼音优质解决方案 我用爬虫一天时间“偷了”知乎一百万用户 只为证明PHP是世界上最好的语言 所使用的程序微信 SDK for Laravel 基于 overtrue wechat开源在线教育点播系统  一款满足你的多种发送需求的短信发送组件 基于 Laravel 的后台系统构建工具 Laravel Admin 使用很少的代码快速构建一个功能完善的高颜值后台系统 内置丰富的后台常用组件 开箱即用 让开发者告别冗杂的HTML代码一个想帮你总结所有类型的上传漏洞的靶场优雅的渐进式PHP采集框架 Laravel 电商实战教程的项目代码Payment是php版本的支付聚合第三方sdk 集成了微信支付 支付宝支付 招商一网通支付 提供统一的调用接口 方便快速接入各种支付 查询 退款 转账能力 服务端接入支付功能 方便 快捷 SPF Swoole PHP Framework 世界第一款基于Swoole扩展的PHP框架 开发者是Swoole创始人 A Wonderful WordPress Theme 樱花庄的白猫博客主题图床 此项目已弃用 基于 ThinkPHP 基础开发平台 登录账号密码都是 admin PanDownload网页复刻版一个开源的网址导航网站项目 您可以拿来制作自己的网址导航 使用PHP Swoole实现的网页即时聊天工具 独角数卡 发卡 开源式站长自动化售货解决方案 高效 稳定 快速 卡密商城系统 高效安全的在线卡密商城 ️命令行模式开发框架ShopXO免费开源商城系统 国内领先企业级B2C免费开源电商系统 包含PC h5 微信小程序 支付宝小程序 百度小程序 头条 抖音小程序 QQ小程序 APP 多商户 遵循MIT开源协议发布 基于 ThinkPHP5 1框架研发Wizard是一款开源的文档管理工具 支持Markdown Swagger Table类型的文档 Swoole MySQL Proxy 一个基于 MySQL 协议 Swoole 开发的MySQL数据库连接池 学习资源整合Freenom域名自动续期一个好玩的Web安全 漏洞测试平台一个基于Yii2高级框架的快速开发应用引擎蓝天采集器是一款免费的数据采集发布爬虫软件 采用php mysql开发 可部署在云服务器 几乎能采集所有类型的网页 无缝对接各类CMS建站程序 免登录实时发布数据 全自动无需人工干预 是网页大数据采集软件中完全跨平台的云端爬虫系统免费开源的中文搜索引擎 采用 C C 编写 基于 xapian 和 scws 提供 PHP 的开发接口和丰富文档WDScanner平台目前实现了如下功能 分布式web漏洞扫描 客户管理 漏洞定期扫描 子域名枚举 端口扫描 网站爬虫 暗链检测 坏链检测 网站指纹搜集 专项漏洞检测 代理搜集及部署等功能  ️兰空图床图标工场 移动应用图标生成工具 一键生成所有尺寸的应用图标和启动图 Argon 一个轻盈 简洁的 WordPress 主题Typecho Fans插件作品目录PHP代码审计分段讲解一个结构清晰的 易于维护的 现代的PHP Markdown解析器百度贴吧云签到 在服务器上配置好就无需进行任何操作便可以实现贴吧的全自动签到 配合插件使用还可实现云灌水 点赞 封禁 删帖 审查等功能 注意 Gitee 原Git osc 仓库将不再维护 目前唯一指定的仓库为 Github 本项目没有官方交流群 如需交流可以直接使用Github的Discussions 没有商业版本 目前贴吧云签到由社区共同维护 不会停止更新 PR 通常在一天内处理 微信调试 API调试和AJAX的调试的工具 能将日志通过WebSocket输出到Chrome浏览器的console中 結巴 中文分詞 做最好的 PHP 中文分詞 中文斷詞組件EleTeam开源项目 电商全套解决方案之PHP版 Shop for PHP Yii2 一个类似京东 天猫 淘宝的商城 有对应的APP支持 由EleTeam团队维护 RhaPHP是微信第三方管理平台 微信公众号管理系统 支持多公众号管理 CRM会员管理 小程序开发 APP接口开发 几乎集合微信功能 简洁 快速上手 快速开发微信各种各样应用 简洁 好用 快速 项目开发快几倍 群 656868 一刻社区后端 API 源码 新 微信服务号 微信小程序 微信支付 支付宝支付苹果cms v1 maccms v1 麦克cms 开源cms 内容管理系统 视频分享程序 分集剧情程序 网址导航程序 文章程序 漫画程序 图片程序一个PHP文件搞定支付宝支付系列 包括电脑网站支付 手机网站支付 现金红包 消费红包 扫码支付 JSAPI支付 单笔转账到支付宝账户 交易结算 分账 分润 网页授权获取用户信息等restful api风格接口 APP接口 APP接口权限 oauth2 接口版本管理 接口鉴权基于企业微信的开源SCRM应用开发框架 引擎 也是一套通用的企业私域流量管理系统 API接口大全不断更新中 欢迎Fork和Star 1 一言 古诗句版 api 2 必应每日一图api 3 在线ip查询 4 m3u8视频在线解析api 5 随机生成二次元图片api 6 快递查询api 支持国内百家快递 7 flv视频在线解析api 8 抖音视频无水印解析api 9 一句话随机图片api 1 QQ用户信息获取api 11 哔哩哔哩封面图获取api 12 千图网58pic无水印解析下载api 13 喜马拉雅主播FM数据采集api 14 网易云音乐api 15 CCTV央视网视频解析api 16 微信运动刷步数api 17 皮皮搞笑 基于swoole的定时器程序 支持秒级处理群 656868 ️ Saber PHP异步协程HTTP客户端微信支付单文件版 一个PHP文件搞定微信支付系列 包括原生支付 扫码支付 H5支付 公众号支付 现金红包 企业付款到零钱等 新增V3版 一个还不错的图床工具 支持Mac Win Linux服务器 支持压缩后上传 添加图片或文字水印 多文件同时上传 同时上传到多个云 右击任意文件上传 快捷键上传剪贴板截图 Web版上传 支持作为Mweb Typora发布图片接口 作为PicGo ShareX uPic等的自定义图床 支持在服务器上部署作为图床接口 支持上传任意格式文件 可能是我用过的最优雅的 Alipay 和 WeChat 的 laravel 支付扩展包了上传大文件的Laravel扩展包开发内功修炼Laravel核心代码学习南京邮电大学开源 Online Judge QQ群 6681 8264 免费IP地址数据库 已支持IPV4 IPV6 结构化输出为国家 省 市 县 运营商 中文数据库 方便实用 laravel5 5和vue js结合的前后端分离项目模板 后端使用了laravel的LTS版本 5 5 前端使用了流行的vue element template项目 作为程序的起点 可以直接以此为基础来进行业务扩展 模板内容包括基础的用户管理和权限管理 日志管理 集成第三方登录 整合laravel echo server 实现了websocket 做到了消息的实时推送 并在此基础上 实现了聊天室和客服功能 权限管理包括后端Token认证和前端vue js的动态权限 解决了前后端完整分离的情况下 vue js的认证与权限相关的痛点 已在本人的多个项目中集成使用  Web安全之机器学习入门 网易云音乐升级APIPHP 集成支付 SDK 集成了支付宝 微信支付的支付接口和其它相关接口的操作 支持 php fpm 和 Swoole 所有框架通用 宇润PHP全家桶技术支持群 17916227MDClub 社区系统后端代码imi 是基于 Swoole 的 PHP 协程开发框架 它支持 Http2 WebSocket TCP UDP MQTT 等主流协议的服务开发 特别适合互联网微服务 即时通讯聊天im 物联网等场景 QQ群 17916227WordPress 版 WebStack 导航主题 nav iowen cnLive2D 看板娘插件 www fghrsh net post 123 html 上使用的后端 API简单搜索 一个简单的前端界面 用惯了各种导航首页 满屏幕尽是各种不厌其烦的广告和资讯 尝试自己写个自己的主页 国内各大CTF赛题及writeup整理收集自网络各处的 webshell 样本 用于测试 webshell 扫描器检测率  PHP微信SDK 微信平台 微信支付 码小六 GitHub 代码泄露监控系统PHP表单生成器 快速生成现代化的form表单 支持前后端分离 内置复选框 单选框 输入框 下拉选择框 省市区三级联动 时间选择 日期选择 颜色选择 文件 图片上传等17种常用组件 悟空CRM 基于TP5 vue ElementUI的前后端分离CRM系统V免签PHP版 完全开源免费的个人免签约解决方案Composer 全量镜像发布于2 17年3月 曾不间断运行2年多 这个开源有助于理解 Composer 镜像的工作原理一个多彩 轻松上手 体验完善 具有强大自定义功能的WordPress主题 基于Sakura主题全球免费代理IP库 高可用IP 精心筛选优质IP 2s必达LaraCMS 是在学习 laravel web 开发实战进阶 实战构架 API 服务器 过程中产生的一个业余作品 试图通过简单的方式 快速构建一套基本的企业站同时保留很灵活的扩展能力和优雅的代码方式 当然这些都得益Laravel的优秀设计 同时LaraCMS 也是一个学习Laravel 不错的参考示例  已停止维护 HookPHP基于C扩展搭建内置AI编程的架构系统 支持微服务部署 热插拔业务组件 集成业务模型 权限模型 UI组件库 多模板 多平台 多域名 多终端 多语言 含常驻内存 前后分离 API平台 LUA QQ群 67911638 中华人民共和国居民身份证 中华人民共和国港澳居民居住证以及中华人民共和国台湾居民居住证号码验证工具 PHP 版 最简单的91porn爬虫php版本Fend 是一款短小精悍 可在 FPM Swoole 服务容器平滑切换的高性能PHP框架 no evil 实现过滤敏感词汇 基于确定有穷自动机 DFA 算法 支持composer安装扩展Z BlogPHP博客程序IYUU自动辅种工具 目前能对国内大部分的PT站点自动辅种 支持下载器集群 支持多盘位 支持多下载目录 支持远程连接等 果酱小店 基于 Laravel swoole 小程序的开源电商系统 优雅与性能兼顾 這是一份純靠北工程師的專案 請好好愛護它 謝謝 EC ecjia 到家是一款可开展O2O业务的移动电商系统 它包含 移动端APP 采用原生模式开发 覆盖使用iOS 及Android系统的移 动终端 后台系统 针对平台日常运营维护的平台后台 针对入驻店铺管理的商家后台 独立并行 移动端H5 能够灵活部署于微信及其他APP 网页等 Material Design 指南的中文翻译 一个纯php分词 thinkphp5 1 layui 实现的带rbac的基础管理后台 方便快速开发法使用百度pcs上传脚本目前最全的前端开发面试题及答案樱花内网穿透网站源代码 2 2 重制版MeepoPS是Meepo PHP Socket的缩写 旨在提供稳定的Socket服务 可以轻松构建在线实时聊天 即时游戏 视频流媒体播放等 基础目录 聚合所有其他目录 包含文档和例子基于 Vue js 的简洁一般强大的 WordPress 单栏博客主题阿里云打造Laravel最好的OSS Storage扩展 网上在线商城 综合网上购物平台swoolefy是一个基于swoole实现的轻量级 高性能 协程级 开放性的API应用服务框架基于redis实现高可用 易拓展 接入方便 生产环境稳定运行的延迟队列 一款基于WordPress开发的高颜值的自适应主题 支持白天与黑夜模式 无刷新加载等 阿里云 OSS 官方 SDK 的 Composer 封装 支持任何 PHP 项目 包括 Laravel Symfony TinyLara 等等 此插件将你的WordPress接入本土生态体系之中 使之更适合国内应用环境PHP的服务化框架 适用于Api Server Rpc Server 帮助原生PHP项目转向微服务化 出色的性能与支持高并发的协程相结合基于ThinkPHP V6 开发的面向API的后台管理系统  PHP Swoole 开发的在线同步点歌台 支持自由点歌 切歌 调整排序 删除指定音乐以及基础权限分级信呼 免费开源的办公OA系统 包括APP pc上客户端 REIM即时通信 服务端等 让每个企业单位都有自己的办公系统 来客电商 微信小程序商城 APP商城 公众号商城 PC商城系统 支付宝小程序商城 抖音小程序商城 百度小程序电商系统 前后端代码全部开源 注重界面美感与用户体验 打造独特电商系统生态圈哔哩哔哩 Bilibili B 站主站助手 直播助手 直播抽奖 挂机升级 贴心小棉袄脚本 Lv6 离你仅有一步之遥 PHP 版 Personal 一个运用php与swoole实现的统计监控系统短视频去水印 抖音 皮皮虾 火山 微视 微博 绿洲 最右 轻视频 快手 全民小视频 巴塞电影 陌陌 Before避风 开眼 Vue Vlog 小咖秀 皮皮搞笑 全民K歌 西瓜视频  中国农历 阴历 与阳历 公历 转换与查询工具AoiAWD 专为比赛设计 便携性好 低权限运行的EDR系统 项目管理系统后端接口ThinkPHP 队列支持Typecho Theme Aria 书写自己的篇章PHP 中文工具包 支持汉字转拼音 拼音分词 简繁互转 数字 金额大写 QQ群 17916227Yii2 community 请访问淘客5合一SDK 支持淘宝联盟 京东联盟 多多进宝 唯品会 苏宁基于 thinkphp 开发的的 blogMojito Admin 基于 Laravel Vue Element 构建的后台管理系统一个经典的XSS渗透管理平台一款基于 RageFrame2 的免费开源的基础销售功能的商城基于Laravel 5 4 的开发的博客系统 代号 myPersimmon证件照片排版在线生成器 在一张6寸的照片上排版多张证件照清华大学计算机学科推荐学术会议和期刊列表WordPress响应式免费主题 Art Blog唯品秀博客 weipxiu com 备用域名weipxiu cn 开源给小伙伴免费使用 如使用过程有任何问题 在线技术支持QQ 欢迎打扰 原创不易 如喜欢 请多多打赏 演示 EwoMail是基于Linux的企业邮箱服务器 集成了众多优秀稳定的组件 是一个快速部署 简单高效 多语言 安全稳定的邮件解决方案 笔记本新版简单强大的无数据库的图床2 版 演示地址  Bilibili B 站自动领瓜子 直播助手 直播挂机脚本 主站助手 PHP 版微信群二维码活码工具 生成微信群活码 随时可以切换二维码 短视频的PHP拓展包 集成各大短视频的去水印功能 抖音 快手 微视主流短视频 PHP去水印一个PHPer的升级之路酷瓜云课堂 在线教育 网课系统 网校系统 知识付费系统 不加密不阉割 1 %全功能开源 可免费商用 框架主要使用ThinkPHP6 layui 拥有完善的权限的管理模块以及敏捷的开发方式 让你开发起来更加的舒服 laravel5 5搭建的后台管理 和 api服务 的小程序商城基于ThinkPHP5 AdminLTE的后台管理系统魔改版本 为 OLAINDEX 添加多网盘挂载及一些小修复海豚PHP 基于ThinkPHP5 1 41LTS的快速开发框架挂载Teambition文件 可直链分享 支持网盘 需申请 和项目文件 无需邀请码 准确率99 9%的ip地址定位库laravel ant design vue 权限后台PHP 第三方登录授权 SDK 集成了QQ 微信 微博 Github等常用接口 支持 php fpm 和 Swoole 所有框架通用 QQ群 17916227抖音去水印PHP版接口一个分布式统计监控系统 包含PHP客户端 服务端整合多接口的IP查询工具 基于阿里云OSS的WordPress远程附件支持插件 后会有期 开箱即用的Laravel后台扩展 前后端分离 后端控制前端组件 无需编写vue即可创建一个的项目 丰富的表单 表格组件 强大的自定义组件功能 yii2 swoole 让yii2运行在swoole上胖鼠采集 WordPress优秀开源采集插件CatchAdmin是一款基于thinkphp6 和 element admin 开发的后台管理系统 基于 ServiceProvider 系统模块完全接耦 随时卸载安装模块 提供了完整的权限和数据权限等功能 大量内置的开发工具提升你的开发体验 官网地址 微信公众平台php版开发包微信小程序 校园小情书后台源码 好玩的表白墙 告白墙 功能全面的PHP命令行应用库 提供控制台参数解析 命令运行 颜色风格输出 用户信息交互 特殊格式信息显示基于 chinese poetry 数据整理的一份 mysql 格式数据帮助 thinkphp 5 开发者快速 轻松的构建Api hyperf admin 是基于 hyperf vue 的配置化后台开发工具 微信支付php 写的视频下载工具 现已支持 Youku Miaopai 腾讯 XVideos Pornhub 91porn 微博酷燃 bilibili 今日头条 芒果TVCorePress 主题 一款高性能 高颜值的WordPress主题快链电商 直播电商 分销商城 微信小程序商城 APP商城 公众号商城 PC商城系统 支付宝小程序商城 抖音小程序商城 百度小程序电商系统 前后端代码全部开源 Laravel vue开发 成熟商用项目 shop mall 商城 电商 利用 PHP cURL 转发 Disqus API 请求可能是最优雅 简易的淘宝客SDKUniAdmin是一套渐进式模块化开源后台 采用前后端分离技术 数据交互采用json格式 功能低耦合高内聚 核心模块支持系统设置 权限管理 用户管理 菜单管理 API管理等功能 后期上线模块商城将打造类似composer npm的开放式插件市场 同时我们将打造一套兼容性的API标准 从ThinkPHP5 1 Vue2开始 逐步吸引爱好者共同加入 以覆盖等多语言框架 PHP 多接口获取快递物流信息包LightCMS 是一个基于 Laravel 开发的轻量级 CMS 系统 也可以作为一个通用的后台管理框架使用单点登录系统快乐二级域名分发系统Typecho Theme Story 爱上你我的故事 一个轻量化的留言板 记事本 社交系统 博客 人类的本质是 咕咕咕？微信域名拦截检测 QQ域名拦截检测 t xzkxb com 查询有缓存 如需实时查询请自行部署 高性能分布式并发锁 行为限流Emlog是一款基于PHP和MySQL的功能强大的博客及CMS建站系统 追求快速 稳定 简单 舒适的建站体验Hyperf admin 基于Hyperf Element UI 通用管理后台企业仓库管理系统HisiPHP V2版是基于ThinkPHP5 1和Layui开发的后台框架 承诺永久免费开源 您可用于学习和商用 但须保留版权信息正常显示 如果HisiPHP对您有帮助 您可以点击右上角 Star 支持一下哦 谢谢 使用PHP开发的简约导航 书签管理系统 软擎是基于 Php 7 2 和 Swoole 4 4 的高性能 简单易用的开发框架 支持同时在 Swoole Server 和 php fpm 两种模式下运行 内置了服务 集成了大量成熟的组件 可以用于构建高性能的Web系统 API 中间件 基础服务等等 个人发卡源码 发卡系统 二次元发卡系统 二次元发卡源码 发卡程序 动漫发卡 PHP发卡源码聊天应用 php实现的dht爬虫搭建的webim客服系统 即时通讯一些实用的python脚本同城拼车微信小程序后端代码 一个响应式干净和简洁优雅的 Typecho 主题php仓库进销存深度学习5 问 以问答形式对常用的概率知识 线性代数 机器学习 深度学习 计算机视觉等热点问题进行阐述 以帮助自己及有需要的读者 全书分为18个章节 5 余万字 由于水平有限 书中不妥之处恳请广大读者批评指正 未完待续 如有意合作 联系scutjy2 15 163 com 版权所有 违权必究 Tan 2 18 6题解 记录自己的leetcode解题之路 最全中华古诗词数据库 唐宋两朝近一万四千古诗人 接近5 5万首唐诗加26万宋诗 两宋时期1564位词人 21 5 首词 uni app 是使用 Vue 语法开发小程序 H5 App的统一框架采用自身模块规范编写的前端 UI 框架 遵循原生 HTML CSS JS 的书写形式 极低门槛 拿来即用 我是依扬 木易杨 公众号 高级前端进阶 作者 每天搞定一道前端大厂面试题 祝大家天天进步 一年后会看到不一样的自己 YApi 是一个可本地部署的 打通前后端及QA的 可视化的接口管理平台小程序组件化开发框架网易云音乐 Node js API service基于 Vue js 的小程序开发框架 从底层支持 Vue js 语法和构建工具体系  ECMAScript 6入门 是一本开源的 JavaScript 语言教程 全面介绍 ECMAScript 6 新增的语法特性  谷粒 Chrome插件英雄榜 为优秀的Chrome插件写一本中文说明书 让Chrome插件英雄们造福人类公众号 加1 同步更新前端面试每日 3 1 以面试题来驱动学习 提倡每日学习与思考 每天进步一点 每天早上5点纯手工发布面试题 死磕自己 愉悦大家 4 道前端面试题全面覆盖小程序 软技能 本文原文由知名 Hacker Eric S Raymond 所撰寫 教你如何正確的提出技術問題並獲得你滿意的答案 千古前端图文教程 超详细的前端入门到进阶学习笔记 从零开始学前端 做一名精致优雅的前端工程师 公众号 千古壹号 作者  book Node js 包教不包会 by alsotang收集所有区块链 BlockChain 技术开发相关资料 包括Fabric和Ethereum开发资料轻量 可靠的小程序 UI 组件库微信小程序商城 微信小程序微店一个可以观看国内主流视频平台所有视频的客户端可伸缩布局方案基于 node js Mongodb 构建的后台系统 js 源码解析磁力链接聚合搜索中华人民共和国行政区划 省级 省份直辖市自治区 地级 城市 县级 区县 乡级 乡镇街道 村级 村委会居委会 中国省市区镇村二级三级四级五级联动地址数据 Web接口管理工具 开源免费 接口自动化 MOCK数据自动生成 自动化测试 企业级管理 阿里妈妈MUX团队出品 阿里巴巴都在用 1 公司的选择 RAP2已发布请移步至github com thx rap2 delosKuboard 是基于 Kubernetes 的微服务管理界面 同时提供 Kubernetes 免费中文教程 入门教程 最新版本的 Kubernetes v1 2 安装手册 k8s install 在线答疑 持续更新 ApacheCN 数据结构与算法译文集 chick 是使用 Node js 和 MongoDB 开发的社区系统一个非常适合IT团队的在线API文档 技术文档工具 Chinese sticker pack More joy 表情包的博物馆 Github最有毒的仓库 中国表情包大集合 聚欢乐 高颜值的第三方网易云播放器 支持 Windows macOS Linux vue2 vue router vuex 入门项目网易云音乐第三方 Flutter实战 电子书 一套代码运行多端 一端所见即多端所见 计算机速成课 Crash Course 字幕组 全4 集 2 18 5 1 精校完成 一个 react redux 的完整项目 和 个人总结中文独立博客列表CSS Inspiration 在这里找到写 CSS 的灵感 rich text 富文本编辑器 汉字拼音 hàn zì pīn yīn Chrome插件开发全攻略 配套完整Demo 欢迎clone体验微信调试 各种WebView样式调试 手机浏览器的页面真机调试 便捷的远程调试手机页面 抓包工具 支持 HTTPS 无需USB连接设备  master分支 渲染器 微信小程序组件 API 云开发示例简悦 SimpRead 让你瞬间进入沉浸式阅读的扩展让H5制作像搭积木一样简单 轻松搭建H5页面 H5网站 PC端网站 LowCode平台  一套组件化 可复用 易扩展的微信小程序 UI 组件库这是一个数据可视化项目 能够将历史数据排名转化为动态柱状图图表微信小程序图表charts组件 Charts for WeChat small app类似易企秀的H5制作 建站工具 可视化搭建系统 一个在你编程时疯狂称赞你的 VSCode 扩展插件全家桶后台管理框架解锁网易云音乐客户端变灰歌曲美观易用的React富文本编辑器 基于draft js开发一个致力于微信小程序和 Web 端同构的解决方案從零開始學 ReactJS ReactJS 1 1 是一本希望讓初學者一看就懂的 React 中文入門教學書 由淺入深學習 ReactJS 生態系源码解读 系列文章 完 我就是来分享脚本玩玩的开发者边车 github打不开 github加速 git clone加速 git release下载加速 stackoverflow加速vue源码逐行注释分析 4 多m的vue源码程序流程图思维导图 diff部分待后续更新 微信小程序解决方案 1KB javascript 覆盖状态管理 跨页通讯 插件开发和云数据库开发给老司机用的一个番号推荐系统 FeHelper Web前端助手记录成长的过程哔哩哔哩 bilibili com 辅助工具 可以替换播放器 推送通知并进行一些快捷操作提供了百度坐标 BD 9 国测局坐标 火星坐标 GCJ 2 和WGS84坐标系之间的转换F2etest是一个面向前端 测试 产品等岗位的多浏览器兼容性测试整体解决方案  ️ 阿里飞猪 很易用的中后台 表单 表格 图表 解决方案CRMEB Min 前后端分离版自带客服系统 是CRMEB品牌全新推出的一款轻量级 高性能 前后端分离的开源电商系统 完善的后台权限管理 会员管理 订单管理 产品管理 客服管理 CMS管理 多端管理 页面DIY 数据统计 系统配置 组合数据管理 日志管理 数据库管理 一键开通短信 产品采集 物流查询等接口  React技术揭秘 一本自顶向下的React源码分析书微信小程序 基于wepy 商城 微店 微信小程序 欢迎学习交流大屏数据可视化Pytorch 中文文档经典的网页对话框组件 强大的动态表单生成器 简洁 易用 灵活的微信小程序组件库 一款 Material Design 风格的 Hexo 主题签到一个帮助你自动申请京东价格保护的chrome拓展后台admin前端模板 基于 layui 编写的最简洁 易用的后台框架模板 只需提供一个接口就直接初始化整个框架 无需复杂操作 小程序生成图片库 轻松通过 json 方式绘制一张可以发到朋友圈的图片安卓应用层抓包通杀脚本一个轻量的工具集合婚礼大屏互动 微信请柬一站式解决方案mili 是一个开源的社区系统 界面优雅 功能丰富 丝般顺滑的触摸运动方案做最好的接口管理平台Mpx 一款具有优秀开发体验和深度性能优化的增强型跨端小程序框架省市区县乡镇三级或四级城市数据 带拼音标注 坐标 行政区域边界范围 2 21年 7月 3日最新采集 提供csv格式文件 支持在线转成多级联动js代码 通用json格式 提供软件转成shp geojson sql 导入数据库 带浏览器里面运行的js采集源码 综合了中华人民共和国民政部 国家统计局 高德地图 腾讯地图行政区划数据在vscode中用于生成文件头部注释和函数注释的插件 经过多版迭代后 插件 支持所有主流语言 功能强大 灵活方便 文档齐全 食用简单 觉得插件不错的话 点击右上角给个Star ️呀  JAVClub 让你的大姐姐不再走丢️你想要的最全 Android 进阶路线知识图谱 干货资料收集  开发者推荐阅读的书籍 2 2 淘宝 京东 支付宝双十一 双11全民养猫 全民营业自动化脚本 全额奖励 防检测 一款高性能敏感词 非法词 脏字 检测过滤组件 附带繁体简体互换 支持全角半角互换 汉字转拼音 模糊搜索等功能 前端博客 关注基础知识和性能优化 vue cli4配置vue config js持续更新PT 助手 Plus 为 Google Chrome 和 Firefox 浏览器插件 Web Extensions 主要用于辅助下载 PT 站的种子 基于vue2 koa2的 H5制作工具 让不会写代码的人也能轻松快速上手制作H5页面 类似易企秀 百度H5等H5制作 建站工具最全最新中国省 市 地区json及sql数据首个 Taro 多端统一实例 网易严选 小程序 H5 React Native By 趣店 FED地理信息可视化库企业级 Node js 应用性能监控与线上故障定位解决方案 Node js区块链开发 注 新版代码已开源 请star支持哦 基于Auto js的蚂蚁森林能量自动收取脚本 结巴 中文分词的Node js版本 译 面向机器学习的特征工程webfunny是一款轻量级的前端监控系统 webfunny也是一款前端性能监控系统 无埋点监控前端日志 实时分析前端健康状态一个实现汉字与拼音互转的小巧web工具库 演示地址  前端进阶 优质博文 一个 Chrome 插件 将 Google CDN 替换为国内的  Vue ElementUI构建的CMS开发框架超完整的React Native项目 功能丰富 适合学习和日常使用 GSYGithubApp系列的优势 我们目前已经拥有四个版本 功能齐全 项目框架内技术涉及面广 完成度高 配套文章 适合全面学习 对比参考 开源Github客户端App 更好的体验 更丰富的功能 旨在更好的日常管理和维护个人Github 提供更好更方便的驾车体验Σ 同款Weex版本同款Flutter版本 https github com CarGu 群 宇宙最强的前端面试指南 lucifer ren fe interview 微慕小程序开源版 WordPress版微信小程序函数式编程指北中文版Node js面试题 侧重后端应用与对Node核心的理解jQuery源码解析小白入坑vue三部曲 关于 vue 在工作的使用问题总结 请看博客 sunseekers github io 一直保持更新基于 Vue 的 PWA 解决方案 帮助开发者快速搭建 PWA 应用 解决接入 PWA 的各种问题ThinkCMF是一款支持Swoole的开源内容管理框架 基于ThinkPHP开发 同时支持PHP FPM和Swoole双模式 让WEB开发更快 微信小程序图片裁剪工具前端知识月刊Next Terminal是一个轻量级堡垒机系统 易安装 易使用 支持RDP SSH VNC Telnet Kubernetes协议 微信小程序 日历组件 基于 ueditor的更现代化的富文本编辑器 支持HTTPS基于JavaScript React Vue2的流程图组件 采用Spring MyBatis Shiro框架 开发的一套权限系统 极低门槛 拿来即用 设计之初 就非常注重安全性 为企业系统保驾护航 让一切都变得如此简单 QQ群 32478 2 4 145799952  一个前端的博客  春松客服 多渠道智能客服系统 开源客服系统 机器人客服一个工作流平台小程序 小游戏以及 Web 通用 Canvas 渲染引擎 在线工具秘籍 为在线工具写一本优质说明书 让在线工具造福人类 前端内参 有关于JavaScript 编程范式 设计模式 软件开发的艺术等大前端范畴内的知识分享 旨在帮助前端工程师们夯实技术基础以通过一线互联网企业技术面试  ️ vCards 中国黄页 优化 iOS Android 来电 信息界面体验各平台的分流规则 复写规则及自动化脚本 基于vue2 的实时聊天项目 图片剪裁上传组件 等笔记快速分享 GoogleDrive OneDrive 每日时报 以前端技术体系为主要分享课题 根据 文章 工具 新闻 视频几大板块作为主要分类 一款高效 高性能的帧动画生成工具 停止维护 一个在线音乐播放器 仅 UI 无功能 小程序富文本组件 支持渲染和编辑 html 支持在微信 QQ 百度 支付宝 头条和 uni app 平台使用基于 electron vue 开发的音乐播放器 界面模仿QQ音乐 技术栈欢迎starweui 是在weui和zepto基础上开发的增强UI组件 目前分为表单 基础 组件 js插件四大类 共计百余项功能 是最全的weui样式同步和更新大佬脚本库 更新懒人配置腾讯云即时通信 IM 服务 国内下载镜像 基于 Vue 的小程序开发框架React 16 8打造精美音乐WebAppWK系列开发框架 V1至V5 Java开源企业级开发框架 单应用 微服务 分布式  ️中国 省市区 三级联动 地址选择器 微信小程序2d动画库 分布式 Redis缓存 Shiro权限管理 Spring Session单点登录 Quartz分布式集群调度 Restful服务 QQ 微信登录 App token登录 微信 支付宝支付 日期转换 数据类型转换 序列化 汉字转拼音 身份证号码验证 数字转人民币 发送短信 发送邮件 加密解密 图片处理 excel导入导出 FTP SFTP fastDFS上传下载 二维码 XML读写 高精度计算 系统配置工具类等等 EduSoho 网络课堂是由杭州阔知网络科技有限公司研发的开源网校系统 EduSoho 包含了在线教学 招生和管理等完整功能 让教育机构可以零门槛建立网校 成功转型在线教育 EduSoho 也可作为企业内训平台 帮助企业实现人才培养  自用的一些乱七八糟 油猴脚本 为刚刚学习php语言以及web网站开发整理的一套资源 有视频 实战代码 学习路径等 会持续更新 This is a goindex theme 一个goindex的扩展主题 NumPy官方中文文档 完整版  搭建移动端开发 基于适配方案 axios封装 构建手机端模板脚手架 后台管理 脚手架接口 从简单开始 PhalApi简称π框架 一个轻量级PHP开源接口框架 专注于接口服务开发 前端特效存档Chrome浏览器 抢购 秒杀插件 秒杀助手 定时自动点击云存储管理客户端 支持七牛云 腾讯云 青云 阿里云 又拍云 亚马逊S3 京东云 仿文件夹管理 图片预览 拖拽上传 文件夹上传 同步 批量导出URL等功能font carrier是一个功能强大的字体操作库 使用它你可以随心所欲的操作字体 让你可以在svg的维度改造字体的展现形状 CRN是Ctrip React Native简称 由携程无线平台研发团队基于React Native框架优化 定制成稳定性和性能更佳 也更适合业务场景的跨平台开发框架  油猴脚本页面浮窗广告完全过滤净化 国服最强最全最新CSDN脚本微信小程序即时通讯模板 使用WebSocket通信小程序反编译 支持分包 “想学吗”个人知识管理与自媒体营销工具 超多经典 Canvas 实例 动态离子背景 炫彩小球 贪吃蛇 坦克大战 是男人就下1 层 心形文字等  Vue UEditor v model双向绑定 HQChart H5 微信小程序 沪深 港股 数字货币 期货 美股 K线图 kline 走势图 缩放 拖拽 十字光标 画图工具 截图 筹码图 分析家语法 通达信语法 麦语法 第3方数据替换接口基于koa2的标准前后端分离框架 一款企业信息化开发基础平台 拟集成OA 办公自动化 CMS 内容管理系统 等企业系统的通用业务功能 JeePlatform项目是一款以SpringBoot为核心框架 集ORM框架Mybatis Web层框架SpringMVC和多种开源组件框架而成的一款通用基础平台 代码已经捐赠给开源中国社区基于inception的自动化SQL操作平台 支持SQL执行 LDAP认证 发邮件 OSC SQL查询 SQL优化建议 权限管理等功能 支持docker镜像是一款专门面向个人 团队和小型组织的私有网盘系统 轻量 开源 完善 无论是在家庭 学校还是在办公室 您都能立刻开始使用它 了解更多请访问官方网站 Node js API 中文文档dubbo服务管理以及监控系统拯救B站的弹幕体验  对抗假消息系列项目之一 截屏 实锤？相信你就输了 ”突破性“更新 支持修改任何网站  ️一个简洁 优雅且高效的 Hugo 主题Vue js 示例项目 简易留言板 本项目拥有完善的文档说明与注释 让您快速上手 Vue js 开发? Vue Validator? Vuex?最佳实践基于 Node js Koa2 实战开发的一套完整的博客项目网站 用 React 编写的基于Taro Dva构建的适配不同端 微信 百度 支付宝小程序 H5 React Native 等 的时装衣橱信息泄漏监控系统 伪装115浏览器干爆前端 一网打尽前端面试 学习路径 优秀好文等各类内容 帮助大家一年内拿到期望的 offer 前端性能监控系统 消息队列 高可用 集群等相关架构SpringBoot v2项目是努力打造springboot框架的极致细腻的脚手架 包括一套漂亮的前台 无其他杂七杂八的功能 原生纯净 中文文本标注工具 最全最新中国 省 市 区县 乡镇街道 json csv sql数据 一款轻巧的渐进式微信小程序框架 全网 1 w 阅读量的进阶前端技术博客仓库 Vue 源码解析 React 深度实践 TypeScript 进阶艺术 工程化 性能优化实践 完整开源 Java快速开发平台 基于Spring SpringMVC Mybatis架构 MStore提供更多好用的插件与模板 文章 商城 微信 论坛 会员 评论 支付 积分 工作流 任务调度等 同时提供上百套免费模板任意选择 价值源自分享 铭飞系统不仅一套简单好用的开源系统 更是一整套优质的开源生态内容体系 铭飞的使命就是降低开发成本提高开发效率 提供全方位的企业级开发解决方案 每月28定期更新版本WeHalo 简约风 的微信小程序版博客 基于 vue2 vuex 构建一个具有 45 个页面的大型单页面应用基于Vue3 Element Plus 的后台管理系统解决方案基于 vue element ui 的后台管理系统鲜亮的高饱和色彩 专注视觉的小程序组件库 ️ 跨平台桌面端视频资源播放器 简洁无广告 免费高颜值 后台管理主线版本基于三者并行开发维护 同时支持电脑 手机 平板 切换分支查看不同的vue版本 element plus版本已发布 vue3 vue3 vue vue3 x vue js 程序无国界 但程序员有国界 中国国家尊严不容挑衅 如果您在特殊时期 mall admin web是一个电商后台管理系统的前端项目 基于Vue Element实现 主要包括商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等功能 一款完善的安全评估工具 支持常见 web 安全问题扫描和自定义 poc 使用之前务必先阅读文档Vue数据可视化组件库 类似阿里DataV 大屏数据展示 提供SVG的边框及装饰 图表 水位图 飞线图等组件 简单易用 长期更新 React版已发布  UI表单设计及代码生成器基于Vue的可视化表单设计器 让表单开发简单而高效 基于vue的高扩展在线网页制作平台 可自定义组件 可添加脚本 可数据统计 vue后台管理框架 精致的下拉刷新和上拉加载 js框架 支持vue 完美运行于移动端和主流PC浏览器 基于vue2 vuex element ui后台管理系统  Vue js高仿饿了么外卖App课程源码 coding imooc com class 74 html京东风格移动端 Vue2 Vue3 组件库eladmin前端源码 项目基于的前后端分离后台管理系统 权限控制采用 RBAC 菜单动态路由资源采集站在线播放uView UI 是uni app生态最优秀的UI框架 全面的组件和便捷的工具会让您信手拈来 如鱼得水Vue2 全家桶仿 微信App 项目 支持多人在线聊天和机器人聊天前端vue 后端koa 全栈式开发bilibili首页 A magical vue admin 记得star互联网大厂内推及大厂面经整理 并且每天一道面试题推送 每天五分钟 半年大厂中 Vue3 全家桶 Vant 搭建大型单页面商城项目 新蜂商城 Vue3 版本 技术栈为基于Vue开发的XMall商城前台页面 PC端 Vue全家桶 Vant 搭建大型单页面电商项目 ddbuy 7 orange cn前后端分离权限管理系统 精力有限 停止维护 用 Vue js 开发的跨三端应用Prototyping Tool For Vue Devs 适用于Vue的原型工具实战商城  基于Vue2 高仿微信App的单页应用electron跨平台音乐播放器 可搜网易云 QQ音乐 虾米音乐 支持QQ 微博 Github登录 云歌单 支持一键导入音乐平台歌单ThorUI组件库 轻量 简洁的移动端组件库 组件文档地址 thorui cn doc 最近更新时间 2 21 5 28uni app框架演示示例 Electron Vue 仿网易云音乐windows客户端 基于 Vue2 Vue CLI3 的高仿网易云 mac 客户端播放器 PC Online Music PlayerGitHub 泄露监控系统pear 梨子 轻量级的在线项目 任务协作系统 远程办公协作自选基金助手是一款Chrome扩展 用来快速获取关注基金的实时数据 查看自选基金的实时估值情况支持 markdown 渲染的博客前台展示Vue 音乐搜索 播放 Demo 一个基于 vue2 vue3 的 大转盘 九宫格 抽奖插件奖品 文字 图片 颜色 按钮均可配置 支持同步 异步抽奖 概率前 后端可控 自动根据 dpr 调整清晰度适配移动端 基于 Vue 的在线音乐播放器 PC Online music player美团饿了吗外卖红包外卖优惠券 先领红包再下单 外卖红包优惠券 cps分成 别人领红包下单 你拿佣金 讨论如何构建一套可靠的大型分布式系统用 vue 写小程序 基于 mpvue 框架重写 weui 基于Vue框架构建的github数据可视化平台使用GitHub API 搭建一个可动态发布文章的博客可视化拖拽组件库 DEMO基于开源组件 Inception SQLAdvisor SOAR 的SQL审核 SQL优化的Web平台显示当前网站的所有可用Tampermonkey脚本 专注Web与算法无缝滚动component精通以太坊 中文版 网页模拟桌面基于的多模块前后端分离的博客项目网易云音乐 QQ音乐 咪咕音乐 第三方 web端 可播放 vip 下架歌曲 基于权限管理的后台管理系统基于 Node js 的开源个人博客系统 采用 Nuxt Vue TypeScript 技术栈  一款简洁高效的VuePress知识管理 博客 blog 主题基于uni app的ui框架基于Vue Vuex iView的电子商城网站 Vue2 全家桶 Vant 搭建大型单页面商城项目 新蜂商城前后端分离版本 前端Vue项目源码酷狗 ️ 极客猿梦导航 独立开发者的导航站  Vue SpringBoot MyBatis 音乐网站基于 RageFrame2 的一款免费开源的基础商城销售功能的开源微商城 基于vue2 的网易云音乐播放器 api来自于NeteaseCloudMusicApi v2 为最新版本编写的一套后台管理系统全栈开发王者荣耀手机端官网和管理后台基于 Vue3 x TypeScript 的在线演示文稿应用 实现PPT幻灯片的在线编辑 演示 基于vue2 生态的后台管理系统模板开发的后台管理系统 Vchat 从头到脚 撸一个社交聊天系统 vue node mongodb h5编辑器类似maka 易企秀 账号 密码 admin996 公司展示 讨论Vue Spring boot前后端分离项目 wh web wh server的升级版 基于element ui的数据驱动表单组件基于 GitHub API 开发的图床神器 图片外链使用 jsDelivr 进行 CDN 加速 免下载 免安装 打开网站即可直接使用 免费 稳定 高效  ️ Vue初 中级项目 CnodeJS社区重构预览 DEMO 基于vue2全家桶实现的 仿移动端QQ基于Vue Echarts 构建的数据可视化平台 酷炫大屏展示模板和组件库 持续更新各行各业实用模板和炫酷小组件 基于Spring Boot的在线考试系统 预览地址 129 211 88 191 账户分别是admin teacher student 密码是admin123 6pan 6盘小白羊 第二版 vue3 antd typescript on bone and knife 基于Vue 全家桶 2 x 制作的美团外卖APP 本项目是一款基于 Avue 的表单设计器 拖拽式操作让你快速构建一个表单 一刻社区前端源码基于Vue iView Admin开发的XBoot前后端分离开放平台前端 权限可控制至按钮显示 动态路由权限菜单 多语言 简洁美观 前后端分离 ️一个开源的社区程序 临时测试站 https t myrpg cnecharts地图geoJson行政边界数据的实时获取与应用 省市区县多级联动下钻 真正意义的下钻至县级 附最新geoJson文件下载 Vue的Nuxt js服务端渲染框架 NodeJS为后端的全栈项目 Docker一键部署 面向小白的完美博客系统vue瀑布流组件 vue waterfall easy 2 x Ego 移动端购物商城 vue vuex ruoter webpack  Vue js Node js Mongodb 前后端分离的个人博客头像加口罩小程序 基于uniapp使用vue快速实现 广告月收入4k  基于vue3 的管理端模板教你如何打造舒适 高效 时尚的前端开发环境基于 Flask 和 Vue js 前后端分离的微型博客项目 支持多用户 Markdown文章 喜欢 收藏文章 粉丝关注 用户评论 点赞 动态通知 站内私信 黑名单 邮件支持 管理后台 权限管理 RQ任务队列 Elasticsearch全文搜索 Linux VPS部署 Docker容器部署等基于 vue 和 heyui 组件库的中后端系统 admin heyui topVue 轻量级后台管理系统基础模板uni app项目插件功能集合We川大小程序 scuplus 使用wepy开发的完善的校园综合小程序 4 页面 前后端开源 包括成绩 课表 失物招领 图书馆 新闻资讯等等常见校园场景功能一个全随机的刷装备小游戏一个vue全家桶入门Demo 是一個可以幫助您 Vue js 的項目測試及偵錯的工具 也同時支持 Vuex及 Vue Router  微信公众号管理系统 包含公众号菜单管理 自动回复 素材管理 模板消息 粉丝管理 ️等功能 前后端都开源免费  基于vue 的管理后台 配合Blog Core与Blog Vue等多个项目使用海风小店 开源商城 微信小程序商城管理后台 后台管理 VUE IT之家第三方小程序版客户端 使用 mpvue 开发 兼容 web vue 可以拖拽排序的树形表格 现代 Web 开发语法基础与工程实践 涵盖 Web 开发基础 前端工程化 应用架构 性能与体验优化 混合开发 React 实践 Vue 实践 WebAssembly 等多方面  数据大屏可视化编辑器一个适用于摄影从业者 爱好者 设计师等创意行业从业者的图像工具箱 武汉大学图书馆助手 桌面端基于form generator 仿钉钉审批流程创建 表单创建 流程节点可视化配置 必填条件及校验 一个完整electron桌面记账程序 技术栈主要使用electron vue vuetify 开机自动启动 自动更新 托盘最小化 闪烁等常用功能 Nsis制作漂亮的安装包 程序猿的婚礼邀请函 一个基于vue和element ui的树形穿梭框及邮件通讯录版本见示例见 基于Gin Vue Element UI的前后端分离权限管理系统的前端模块通用书籍阅读APP BookChat 的 uni app 实现版本 支持多端分发 编译生成Android和iOS 手机APP以及各平台的小程序基于Vue3的Material design风格移动端组件库进阶资深前端开发在线考试系统 springboot vue前后端分离的一个项目  ️ 无后端的仿 YouTube Live Chat 风格的简易 Bilibili 弹幕姬vue后端管理系统界面 基于ui组件iviewBilibili直播弹幕库 for Mac Windows LinuxVue高仿网易云音乐 基本实现网易云所有音乐 MV相关功能 现已更新到第二版 仅用于学习 下面有详细教程 武汉大学图书馆助手 移动端Zeus基于Golang Gin casbin 致力于做企业统一权限 账号中心管理系统 包含账号管理 数据权限 功能权限 应用管理 多数据库适配 可docker 一键运行 社区活跃 版本迭代快 加群免费技术支持 Vue高仿网易云音乐 Vue入门实践 在线预览 暂时停止基于 Vue 和 ElementUI 构建的一个企业级后台管理系统 ️ 跨平台移动端视频资源播放器 简洁免费 ZY Player 移动端 APP 基于 Uni app 开发 Vue实战项目基于参考小米商城 实现的电商项目 h5制作 移动端专题活动页面可视化编辑仿钉钉审批流程设置动态表单页面设计 自动生成页面微前端项目实战vue项目 基于vue3 qiankun2 进阶版 github com wl ui wl mfe基于 d2 admin的RBAC权限管理解决方案VueNode 是一套基于的前后端分离项目 基于仿京东淘宝的 移动端H5电商平台  巨树 基于ztree封装的Vue树形组件 轻松实现海量数据的高性能渲染 微信红包封面领取 用户观看视频广告或者邀请用户可获取微信红包序列号基于 Vue 的可视化布局编辑器插件kbone ui 是一套能同时支持 小程序 kbone 和 vue 框架开发的多端 UI 库 PS 新版 kbone ui 已出炉并迁移到 kbone 主仓库 此仓库仅做旧版维护之用  一个vue的个人博客项目 配合 net core api教程 打造前后端分离Tumo Blog For Vue js 前后端分离bpmn js流程设计器组件 基于vue elementui美化属性面板 满足9 %以上的业务需求专门为 Weex 前端开发者打造的一套高质量UI框架 想用vue把我现在的个人网站重新写一下 新的风格 新的技术 什么都是新的 本项目是一个在线聊天系统 最大程度的还原了Mac客户端QQ vue cli3 后台管理模板 heart 基于vue2和vuex的复杂单页面应用 2 页面53个API 仿实验楼 基于 Vue Koa 的 WebDesktop 视窗系统 Jeebase是一款前后端分离的开源开发框架 基于开发 一套SpringBoot后台 两套前端页面 可以自由选择基于ElementUI或者AntDesign的前端界面 二期会整合react前端框架 Ant Design React 在实际应用中已经使用这套框架开发了CMS网站系统 社区论坛系统 微信小程序 微信服务号等 后面会逐步整理开源 本项目主要目的在于整合主流技术框架 寻找应用最佳项目实践方案 实现可直接使用的快速开发框架 使用 vue cli3 搭建的vue vuex router element 开发模版 集成常用组件 功能模块JEECG BOOT APP 移动解决方案 采用uniapp框架 一份代码多终端适配 同时支持APP 小程序 H5 实现了与JeecgBoot平台完美对接的移动解决方案 目前已经实现登录 用户信息 通讯录 公告 移动首页 九宫格等基础功能 明日方舟工具箱 支持中台美日韩服vue的验证码插件这里有一些标准组件库可能没有的功能组件 已有组件 放大镜 签到 图片标签 滑动验证 倒计时 水印 拖拽 大家来找茬 基于Vue2 Nodejs MySQL的博客 有后台管理系统 支持 登陆 注册 留言 评论 回复 点赞长江证券郑州大学超市管理系统天安门坦克桌面行驶工况茶马古道金融文本情感自动黄河银行营销通许章润\",\"html_url\":\"https://github.com/cirosantilli/china-dictatorship\",\"language\":\"HTML\",\"stargazers_count\":2715,\"forks_count\":278,\"open_issues_count\":808,\"updated_at\":\"2025-11-19T13:11:16Z\",\"created_at\":\"2015-04-02T20:51:50Z\",\"topics\":[\"996\",\"censorship\",\"censorship-circumvention\",\"china\",\"china-dictatorship\",\"chinese-communist-party\",\"covid-19\",\"covid-19-china\",\"dictator\",\"dictatorship\",\"falun-gong\",\"gfw\",\"great-firewall\",\"human-rights\",\"shadowsocks\",\"socks5\",\"tiananmen\",\"totalitarian\",\"xi-jinping\",\"xinjiang\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":99764044,\"name\":\"AI-Chip\",\"full_name\":\"basicmi/AI-Chip\",\"description\":\"A list of ICs and IPs for AI, Machine Learning and Deep Learning.\",\"html_url\":\"https://github.com/basicmi/AI-Chip\",\"language\":\"PHP\",\"stargazers_count\":1694,\"forks_count\":278,\"open_issues_count\":24,\"updated_at\":\"2025-11-15T11:29:11Z\",\"created_at\":\"2017-08-09T04:10:54Z\",\"topics\":[\"ai-chips\",\"chip\",\"deep-learning\",\"machine-learning\",\"processor\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":297003752,\"name\":\".github\",\"full_name\":\"gege-circle/.github\",\"description\":\"这里是GitHub的草场，也是戈戈圈爱好者的交流地，主要讨论动漫、游戏、科技、人文、生活等所有话题，欢迎各位小伙伴们在此讨论趣事。This is GitHub grassland, and the community place for Gege circle lovers, mainly discusses anime, games, technology, lifing and other topics. You are welcome to share interest things here.  　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　　缺氧修改器 捏脸 反光板 MySQL数据库 玉米杂草数据集 销售系统开发 疫情期间网民情绪识别比赛 996icu 预测结果导出 赖伟林刺杀小说家 购物商场 英语词汇量小程序 联级选择器Bitcoin区块链 技术面试必备基础知识 Leetcode 计算机操作系统 计算机网络 系统设计 Java学习 面试指南 一份涵盖大部分 Java 程序员所需要掌握的核心知识 准备 Java 面试 首选 JavaGuide Python 1 天从新手到大师刷算法全靠套路 认准 labuladong 就够了 免费的计算机编程类中文书籍 欢迎投稿用动画的形式呈现解LeetCode题目的思路 互联网 Java 工程师进阶知识完全扫盲 涵盖高并发 分布式 高可用 微服务 海量数据处理等领域知识后端架构师技术图谱mall项目是一套电商系统 包括前台商城系统及后台管理系统 基于SpringBoot MyBatis实现 采用Docker容器化部署 前台商城系统包含首页门户 商品推荐 商品搜索 商品展示 购物车 订单流程 会员中心 客户服务 帮助中心等模块 后台管理系统包含商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等模块 微信小程序开发资源汇总 最全中华古诗词数据库 唐宋两朝近一万四千古诗人 接近5 5万首唐诗加26万宋诗 两宋时期1564位词人 21 5 首词 uni app 是使用 Vue 语法开发小程序 H5 App的统一框架2 21年最新总结 阿里 腾讯 百度 美团 头条等技术面试题目 以及答案 专家出题人分析汇总 科学上网 自由上网 翻墙 软件 方法 一键翻墙浏览器 免费账号 节点分享 vps一键搭建脚本 教程AiLearning 机器学习 MachineLearning ML 深度学习 DeepLearning DL 自然语言处理 NLP123 6智能刷票 订票开放式跨端跨框架解决方案 支持使用 React Vue Nerv 等框架来开发微信 京东 百度 支付宝 字节跳动 QQ 小程序 H5 React Native 等应用 taro zone 掘金翻译计划 可能是世界最大最好的英译中技术社区 最懂读者和译者的翻译平台 no evil 程序员找工作黑名单 换工作和当技术合伙人需谨慎啊 更新有赞 算法面试 算法知识 针对小白的算法训练 还包括 1 阿里 字节 滴滴 百篇大厂面经汇总 2 千本开源电子书 3 百张思维导图 右侧来个 star 吧 English version supported 955 不加班的公司名单 工作 955 work–life balance 工作与生活的平衡 诊断利器Arthas The Way to Go 中文译本 中文正式名 Go 入门指南 Java面试 Java学习指南 一份涵盖大部分Java程序员所需要掌握的核心知识 教程 技术栈示例代码 快速简单上手教程 2 17年买房经历总结出来的买房购房知识分享给大家 希望对大家有所帮助 买房不易 且买且珍惜http下载工具 基于http代理 支持多连接分块下载 动手学深度学习 面向中文读者 能运行 可讨论 中英文版被全球175所大学采用教学 阿里云计算平台团队出品 为监控而生的数据库连接池程序员简历模板系列 包括PHP程序员简历模板 iOS程序员简历模板 Android程序员简历模板 Web前端程序员简历模板 Java程序员简历模板 C C 程序员简历模板 NodeJS程序员简历模板 架构师简历模板以及通用程序员简历模板采用自身模块规范编写的前端 UI 框架 遵循原生 HTML CSS JS 的书写形式 极低门槛 拿来即用 贵校课程资料民间整理 企业级低代码平台 前后端分离架构强大的代码生成器让前后端代码一键生成 无需写任何代码 引领新的开发模式OnlineCoding 代码生成 手工MERGE 帮助Java项目解决7 %重复工作 让开发更关注业务 既能快速提高效率 帮助公司节省成本 同时又不失灵活性 我是依扬 木易杨 公众号 高级前端进阶 作者 每天搞定一道前端大厂面试题 祝大家天天进步 一年后会看到不一样的自己 冴羽写博客的地方 预计写四个系列 JavaScript深入系列 JavaScript专题系列 ES6系列 React系列 中文分词 词性标注 命名实体识别 依存句法分析 语义依存分析 新词发现 关键词短语提取 自动摘要 文本分类聚类 拼音简繁转换 自然语言处理flutter 开发者帮助 APP 包含 flutter 常用 14 组件的demo 演示与中文文档 下拉刷新 上拉加载 二级刷新 淘宝二楼智能下拉刷新框架 支持越界回弹 越界拖动 具有极强的扩展性 集成了几十种炫酷的Header和 Footer 该项目已成功集成 actuator 监控 admin 可视化监控 logback 日志 aopLog 通过AOP记录web请求日志 统一异常处理 json级别和页面级别 freemarker 模板引擎 thymeleaf 模板引擎 Beetl 模板引擎 Enjoy 模板引擎 JdbcTemplate 通用JDBC操作数据库 JPA 强大的ORM框架 mybatis 强大的ORM框架 通用Mapper 快速操作Mybatis PageHelper 通用的Mybatis分页插件 mybatis plus 快速操作Mybatis BeetlSQL 强大的ORM框架 u Python资源大全中文版 包括 Web框架 网络爬虫 模板引擎 数据库 数据可视化 图片处理等 由 开源前哨 和 Python开发者 微信公号团队维护更新 吴恩达老师的机器学习课程个人笔记To Be Top Javaer Java工程师成神之路循序渐进 学习博客Spring系列源码 mrbird cc谢谢可能是让你受益匪浅的英语进阶指南镜像网易云音乐 Node js API service快速 简单避免OOM的java处理Excel工具基于 Vue js 的小程序开发框架 从底层支持 Vue js 语法和构建工具体系 中文版 Apple 官方 Swift 教程本项目曾冲到全球第一 干货集锦见本页面最底部 另完整精致的纸质版 编程之法 面试和算法心得 已在京东 当当上销售好耶 是女装Security Guide for Developers 实用性开发人员安全须知 阿里巴巴 MySQL binlog 增量订阅 消费组件 ECMAScript 6入门 是一本开源的 JavaScript 语言教程 全面介绍 ECMAScript 6 新增的语法特性 C C 技术面试基础知识总结 包括语言 程序库 数据结构 算法 系统 网络 链接装载库等知识及面试经验 招聘 内推等信息 一款优秀的开源博客发布应用 Solutions to LeetCode by Go 1 % test coverage runtime beats 1 % LeetCode 题解分布式任务调度平台XXL JOB 谷粒 Chrome插件英雄榜 为优秀的Chrome插件写一本中文说明书 让Chrome插件英雄们造福人类公众号 加1 同步更新互联网公司技术架构 微信 淘宝 微博 腾讯 阿里 美团点评 百度 Google Facebook Amazon eBay的架构 欢迎PR补充IntelliJ IDEA 简体中文专题教程程序员技能图谱前端面试每日 3 1 以面试题来驱动学习 提倡每日学习与思考 每天进步一点 每天早上5点纯手工发布面试题 死磕自己 愉悦大家 4 道前端面试题全面覆盖小程序 软技能 华为鸿蒙操作系统 互联网首份程序员考公指南 由3位已经进入体制内的前大厂程序员联合献上 Mac微信功能拓展 微信插件 微信小助手 A plugin for Mac WeChat 机器学习 西瓜书 公式推导解析 在线阅读地址一款轻量级 高性能 功能强大的内网穿透代理服务器 支持tcp udp socks5 http等几乎所有流量转发 可用来访问内网网站 本地支付接口调试 ssh访问 远程桌面 内网dns解析 内网socks5代理等等 并带有功能强大的web管理端一款面向泛前端产品研发全生命周期的效率平台 文言文編程語言清华大学计算机系课程攻略面向云原生微服务的高可用流控防护组件 On Java 8 中文版 本文原文由知名 Hacker Eric S Raymond 所撰寫 教你如何正確的提出技術問題並獲得你滿意的答案 React Native指南汇集了各类react native学习资源 开源App和组件1 Days Of ML Code中文版千古前端图文教程 超详细的前端入门到进阶学习笔记 从零开始学前端 做一名精致优雅的前端工程师 公众号 千古壹号 作者 基于 React 的渐进式研发框架 ice work视频播放器支持弹幕 外挂字幕 支持滤镜 水印 gif截图 片头广告 中间广告 多个同时播放 支持基本的拖动 声音 亮度调节 支持边播边缓存 支持视频自带rotation的旋转 9 27 之类 重力旋转与手动旋转的同步支持 支持列表播放 列表全屏动画 视频加载速度 列表小窗口支持拖动 动画效果 调整比例 多分辨率切换 支持切换播放器 进度条小窗口预览 列表切换详情页面无缝播放 rtsp concat mpeg JumpServer 是全球首款开源的堡垒机 是符合 4A 的专业运维安全审计系统 Linux命令大全搜索工具 内容包含Linux命令手册 详解 学习 搜集 git io linux book Node js 包教不包会 by alsotang又一个小商城 litemall Spring Boot后端 Vue管理员前端 微信小程序用户前端 Vue用户移动端微信 跳一跳 Python 辅助Java资源大全中文版 包括开发库 开发工具 网站 博客 微信 微博等 由伯乐在线持续更新 python模拟登陆一些大型网站 还有一些简单的爬虫 希望对你们有所帮助 ️ 如果喜欢记得给个star哦 C 那些事 网络爬虫实战 淘宝 京东 网易云 B站 123 6 抖音 笔趣阁 漫画小说下载 音乐电影下载等deeplearning ai 吴恩达老师的深度学习课程笔记及资源 Spring Boot基础教程 Spring Boot 2 x版本连载中 帮助 Android App 进行组件化改造的路由框架 最接近原生APP体验的高性能框架基于Vue3 Element Plus 的后台管理系统解决方案程序员如何优雅的挣零花钱 2 版 升级为小书了 从Java基础 JavaWeb基础到常用的框架再到面试题都有完整的教程 几乎涵盖了Java后端必备的知识点spring boot 实践学习案例 是 spring boot 初学者及核心技术巩固的最佳实践 另外写博客 用 OpenWrite 最好用的 V2Ray 一键安装脚本 管理脚本中国程序员容易发音错误的单词 统计学习方法 的代码实现关于Python的面试题本项目将 动手学深度学习 Dive into Deep Learning 原书中的MXNet实现改为PyTorch实现 提高 Android UI 开发效率的 UI 库前端精读周刊 帮你理解最前沿 实用的技术 的奇技淫巧时间选择器 省市区三级联动 Python爬虫代理IP池 proxy pool LeetCode 刷题攻略 2 道经典题目刷题顺序 共6 w字的详细图解 视频难点剖析 5 余张思维导图 从此算法学习不再迷茫 来看看 你会发现相见恨晚 一个基于 electron 的音乐软件Flutter 超完整的开源项目 功能丰富 适合学习和日常使用 GSYGithubApp系列的优势 我们目前已经拥有四个版本 功能齐全 项目框架内技术涉及面广 完成度高 持续维护 配套文章 适合全面学习 对比参考 跨平台的开源Github客户端App 更好的体验 更丰富的功能 旨在更好的日常管理和维护个人Github 提供更好更方便的驾车体验Σ 同款Weex版本同款React Native版本 https g 这是一个用于显示当前网速 CPU及内存利用率的桌面悬浮窗软件 并支持任务栏显示 支持更换皮肤 是一个跨平台的强加密无特征的代理软件 零配置 V2rayU 基于v2ray核心的mac版客户端 用于科学上网 使用swift编写 支持vmess shadowsocks socks5等服务协议 支持订阅 支持二维码 剪贴板导入 手动配置 二维码分享等算法模板 最科学的刷题方式 最快速的刷题路径 你值得拥有 经典编程书籍大全 涵盖 计算机系统与网络 系统架构 算法与数据结构 前端开发 后端开发 移动开发 数据库 测试 项目与团队 程序员职业修炼 求职面试等wangEditor 轻量级web富文本框前端跨框架跨平台框架 每个 JavaScript 工程师都应懂的33个概念 leonardomso一个可以观看国内主流视频平台所有视频的客户端Android开发人员不得不收集的工具类集合 支付宝支付 微信支付 统一下单 微信分享 Zip4j压缩 支持分卷压缩与加密 一键集成UCrop选择圆形头像 一键集成二维码和条形码的扫描与生成 常用Dialog WebView的封装可播放视频 仿斗鱼滑动验证码 Toast封装 震动 GPS Location定位 图片缩放 Exif 图片添加地理位置信息 经纬度 蛛网等级 颜色选择器 ArcGis VTPK 编译运行一下说不定会找到惊喜 123 6 购票助手 支持集群 多账号 多任务购票以及 Web 页面管理 Android广告图片轮播控件 内部基于ViewPager2实现 Indicator和UI都可以自定义 零代码 热更新 自动化 ORM 库 后端接口和文档零代码 前端 客户端 定制返回 JSON 的数据和结构 Linux Windows macOS 跨平台 V2Ray 客户端 支持使用 C Qt 开发 可拓展插件式设计 walle 瓦力 Devops开源项目代码部署平台基于 node js Mongodb 构建的后台系统 js 源码解析一个涵盖六个专栏分布式消息队列 分布式事务的仓库 希望胖友小手一抖 右上角来个 Star 感恩 1 24基于 vue element ui 的后台管理系统磁力链接聚合搜索中华人民共和国行政区划 省级 省份直辖市自治区 地级 城市 县级 区县 乡级 乡镇街道 村级 村委会居委会 中国省市区镇村二级三级四级五级联动地址数据 iOS开发常用三方库 插件 知名博客等等LeetCode题解 151道题完整版／中文文案排版指北最良心的 Python 教程 业内为数不多致力于极致体验的超强全自研跨平台 windows android iOS 流媒体内核 通过模块化自由组合 支持实时RTMP推流 RTSP推流 RTMP播放器 RTSP播放器 录像 多路流媒体转发 音视频导播 动态视频合成 音频混音 直播互动 内置轻量级RTSP服务等 比快更快 业界真正靠谱的超低延迟直播SDK 1秒内 低延迟模式下2 4 ms 一个 PHP 微信 SDK ️ 跨平台桌面端视频资源播放器 简洁无广告 免费高颜值 后台管理主线版本基于三者并行开发维护 同时支持电脑 手机 平板 切换分支查看不同的vue版本 element plus版本已发布 vue3 vue3 vue vue3 x vue js 程序无国界 但程序员有国界 中国国家尊严不容挑衅 如果您在特殊时期 此项目是机器学习 Machine Learning 深度学习 Deep Learning NLP面试中常考到的知识点和代码实现 也是作为一个算法工程师必会的理论基础知识 夜读 通过 bilibili 在线直播的方式分享 Go 相关的技术话题 每天大家在微信 telegram Slack 上及时沟通交流编程技术话题 GitHubDaily 分享内容定期整理与分类 欢迎推荐 自荐项目 让更多人知道你的项目 支持多家云存储的云盘系统机器学习相关教程DataX是阿里云DataWorks数据集成的开源版本 这里是写博客的地方 Halfrost Field 冰霜之地mall学习教程 架构 业务 技术要点全方位解析 mall项目 4 k star 是一套电商系统 使用现阶段主流技术实现 涵盖了等技术 采用Docker容器化部署 chick 是使用 Node js 和 MongoDB 开发的社区系统一个非常适合IT团队的在线API文档 技术文档工具汇总各大互联网公司容易考察的高频leetcode题 1 Chinese Word Vectors 上百种预训练中文词向量 Android开源弹幕引擎 烈焰弹幕使 ～深度学习框架PyTorch 入门与实战 网易云音乐命令行版本 对开发人员有用的定律 理论 原则和模式TeachYourselfCS 的中文翻译高颜值的第三方网易云播放器 支持 Windows macOS Linux spring cloud vue oAuth2 全家桶实战 前后端分离模拟商城 完整的购物流程 后端运营平台 可以实现快速搭建企业级微服务项目 支持微信登录等三方登录 Chinese sticker pack More joy 表情包的博物馆 Github最有毒的仓库 中国表情包大集合 聚欢乐 Lantern官方版本下载 蓝灯 翻墙 代理 科学上网 外网 加速器 梯子 路由一款入门级的人脸 视频 文字检测以及识别的项目 vue2 vue router vuex 入门项目PanDownload的个人维护版本 一个基于Spring Boot MyBatis的种子项目 用于快速构建中小型API RESTful API项目 iOS interview questions iOS面试题集锦 附答案 学习qq群或 Telegram 群交流为互联网IT人打造的中文版awesome go强大 可定制 易扩展的 ViewPager 指示器框架 是的最佳替代品 支持角标 更支持在非ViewPager场景下使用 使用hide show 切换Fragment或使用se Kubernetes中文指南 云原生应用架构实践手册For macOS 百度网盘 破解SVIP 下载速度限制 架构师技术图谱 助你早日成为架构师mall admin web是一个电商后台管理系统的前端项目 基于Vue Element实现 主要包括商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等功能 网易云音乐第三方 编程随想 整理的 太子党关系网络 专门揭露赵国的权贵基于gin vue搭建的后台管理系统框架 集成jwt鉴权 权限管理 动态路由 分页封装 多点登录拦截 资源权限 上传下载 代码生成器 表单生成器 通用工作流等基础功能 五分钟一套CURD前后端代码 目VUE3版本正在重构 欢迎issue和pr 27天成为Java大神一个基于浏览器端 JS 实现的在线代理编程电子书 电子书 编程书籍 包括人工智能 大数据类 并发编程 数据库类 数据挖掘 新面试题 架构设计 算法系列 计算机类 设计模式 软件测试 重构优化 等更多分类ADB Usage Complete ADB 用法大全二维码生成器 支持 gif 动态图片二维码 Vim 从入门到精通阿布量化交易系统 股票 期权 期货 比特币 机器学习 基于python的开源量化交易 量化投资架构一个简洁优雅的hexo主题 Wiki of OI ICPC for everyone 某大型游戏线上攻略 内含炫酷算术魔法 Google 开源项目风格指南 中文版 Git AWS Google 镜像 SS SSR VMESS节点行业研究报告的知识储备库 cim cross IM 适用于开发者的分布式即时通讯系统微信小程序开源项目库汇总每天更新 全网热门 BT Tracker 列表 天用Go动手写 从零实现系列强大的哔哩哔哩增强脚本 下载视频 音乐 封面 弹幕 简化直播间 评论区 首页 自定义顶栏 删除广告 夜间模式 触屏设备支持Evil Huawei 华为作过的恶Android上一个优雅 万能自定义UI 仿iOS 支持垂直 水平方向切换 支持周视图 自定义周起始 性能高效的日历控件 支持热插拔实现的UI定制 支持标记 自定义颜色 农历 自定义月视图各种显示模式等 Canvas绘制 速度快 占用内存低 你真的想不到日历居然还可以如此优雅已不再维护科学上网插件的离线安装包储存在这里ThinkPHP Framework 十年匠心的高性能PHP框架 Java 程序员眼中的 Linux 一个支持多选 选原图和视频的图片选择器 同时有预览 裁剪功能 支持hsweb haʊs wɛb 是一个基于spring boot 2 x开发 首个使用全响应式编程的企业级后台管理系统基础项目 自动学习wxParse 微信小程序富文本解析自定义组件 支持HTML及markdown解析 newbee mall 项目 新蜂商城 是一套电商系统 包括 newbee mall 商城系统及 newbee mall admin 商城后台管理系统 基于 Spring Boot 2 X 及相关技术栈开发 前台商城系统包含首页门户 商品分类 新品上线 首页轮播 商品推荐 商品搜索 商品展示 购物车 订单结算 订单流程 个人订单管理 会员中心 帮助中心等模块 后台管理系统包含数据面板 轮播图管理 商品管理 订单管理 会员管理 分类管理 设置等模块 最全的前端资源汇总仓库 包括前端学习 开发资源 求职面试等 中文翻译手写实现李航 统计学习方法 书中全部算法 Python 抖音机器人 论如何在抖音上找到漂亮小姐姐？ ️A static blog writing client 一个静态博客写作客户端 超级速查表 编程语言 框架和开发工具的速查表 单个文件包含一切你需要知道的东西 迁移学习前端低代码框架 通过 JSON 配置就能生成各种页面 技术面试最后反问面试官的话Machine Learning Yearning 中文版 机器学习训练秘籍 Andrew Ng 著越来越多的网站具有反爬虫特性 有的用图片隐藏关键数据 有的使用反人类的验证码 建立反反爬虫的代码仓库 通过与不同特性的网站做斗争 无恶意 提高技术 欢迎提交难以采集的网站 因工作原因 项目暂停 本项目收藏这些年来看过或者听过的一些不错的常用的上千本书籍 没准你想找的书就在这里呢 包含了互联网行业大多数书籍和面试经验题目等等 有人工智能系列 常用深度学习框架TensorFlow pytorch keras NLP 机器学习 深度学习等等 大数据系列 Spark Hadoop Scala kafka等 程序员必修系列 C C java 数据结构 linux 设计模式 数据库等等 人人影视bot 完全对接人人影视全部无删减资源Spring Cloud基础教程 持续连载更新中一个用于在 macOS 上平滑你的鼠标滚动效果或单独设置滚动方向的小工具 让你的滚轮爽如触控板阿里妈妈前端团队出品的开源接口管理工具RAP第二代超轻量级中文ocr 支持竖排文字识别 支持ncnn mnn tnn推理总模型仅4 7M 微信全平台 SDK Senparc Weixin for C 支持 NET Framework 及 NET Core NET 6 已支持微信公众号 小程序 小游戏 企业号 企业微信 开放平台 微信支付 JSSDK 微信周边等全平台 WeChat SDK for C 中文独立博客列表高效率 QQ 机器人支持库支持定制任何播放器SDK和控制层 OpenPower工作组收集汇总的医院开放数据Xray 基于 Nginx 的 VLESS XTLS 一键安装脚本 FlutterDemo合集 今天你fu了吗莫烦Python 中文AI教学中国特色 TabBar 一行代码实现 Lottie 动画TabBar 支持中间带 号的TabBar样式 自带红点角标 支持动态刷新 Flutter豆瓣客户端 Awesome Flutter Project 全网最1 %还原豆瓣客户端 首页 书影音 小组 市集及个人中心 一个不拉 img xuvip top douyademo mp4 基于SpringCloud2 1的微服务开发脚手架 整合了等 服务治理方面引入等 让项目开发快速进入业务开发 而不需过多时间花费在架构搭建上 持续更新中基于 Vue2 和 ECharts 封装的图表组件 SSR 去广告ACL规则 SS完整GFWList规则 Clash规则碎片 Telegram频道订阅地址和我一步步部署 kubernetes 集群搜集 整理 维护实用规则 中文自然语言处理相关资料基于SOA架构的分布式电商购物商城 前后端分离 前台商城 全家桶 后台管理系统等What happens when 的中文翻译 原仓库QMUI iOS 致力于提高项目 UI 开发效率的解决方案新型冠状病毒防疫信息收集平台告别枯燥 致力于打造 Python 实用小例子在线制作 sorry 为所欲为 的gifNodejs学习笔记以及经验总结 公众号 程序猿小卡 李宏毅 机器学习 笔记 在线阅读地址 Vue js 源码分析V部落 Vue SpringBoot实现的多用户博客管理平台 Android Signature V2 Scheme签名下的新一代渠道包打包神器Autoscroll Banner 无限循环图片 文字轮播器 多种编程语言实现 LeetCode 剑指 Offer 第 2 版 程序员面试金典 第 6 版 题解一套高质量的微信小程序 UI 组件库飞桨 官方模型库 包含多种学术前沿和工业场景验证的深度学习模型 中文 Python 笔记专门为刚开始刷题的同学准备的算法基地 没有最细只有更细 立志用动画将晦涩难懂的算法说的通俗易懂 版入门实例代码 实战教程 是一个高性能且低损耗的 goroutine 池 CVPR 2 21 论文和开源项目合集有 有 Python进阶 Intermediate Python 中文版 机器人视觉 移动机器人 VS SLAM ORB SLAM2 深度学习目标检测 yolov3 行为检测 opencv PCL 机器学习 无人驾驶后台管理系统解决方案创建在线课程 学术简历或初创网站 Chrome插件开发全攻略 配套完整Demo 欢迎clone体验QUANTAXIS 支持任务调度 分布式部署的 股票 期货 期权 港股 虚拟货币 数据 回测 模拟 交易 可视化 多账户 纯本地量化解决方案微信调试 各种WebView样式调试 手机浏览器的页面真机调试 便捷的远程调试手机页面 抓包工具 支持 HTTPS 无需USB连接设备 rich text 富文本编辑器 汉字拼音 hàn zì pīn yīn面向开发人员梳理的代码安全指南以撸代码的形式学习Python提供同花顺客户端 国金 华泰客户端 雪球的基金 股票自动程序化交易以及自动打新 支持跟踪 joinquant ricequant 模拟交易 和 实盘雪球组合 量化交易组件搜狐视频 sohu tv Redis私有云平台spring boot打造文件文档在线预览项目计算机基础 计算机网络 操作系统 数据库 Git 面试问题全面总结 包含详细的follow up question以及答案 全部采用 问题 追问 答案 的形式 即拿即用 直击互联网大厂面试 可用于模拟面试 面试前复习 短期内快速备战面试 首款微信 macOS 客户端撤回拦截与多开windows kernel exploits Windows平台提权漏洞集合权限管理系统 预览地址 47 1 4 7 138 loginpkuseg多领域中文分词工具一款完善的安全评估工具 支持常见 web 安全问题扫描和自定义 poc 使用之前务必先阅读文档零反射全动态Android插件框架Python入门网络爬虫之精华版分布式配置管理平台 中文 iOS Mac 开发博客列表周志华 机器学习 又称西瓜书是一本较为全面的书籍 书中详细介绍了机器学习领域不同类型的算法 例如 监督学习 无监督学习 半监督学习 强化学习 集成降维 特征选择等 记录了本人在学习过程中的理解思路与扩展知识点 希望对新人阅读西瓜书有所帮助 国内首个Spring Cloud微服务化RBAC的管理平台 核心采用前端采用d2 admin中台框架 记得上边点个star 关注更新Apache ECharts incubating 的微信小程序版本C 资源大全中文版 标准库 Web应用框架 人工智能 数据库 图片处理 机器学习 日志 代码分析等 由 开源前哨 和 CPP开发者 微信公号团队维护更新 stackoverflow上Java相关回答整理翻译 基于Google Flutter的WanAndroid客户端 支持Android和iOS 包括BLoC RxDart 国际化 主题色 启动页 引导页 本代码库是作者小傅哥多年从事一线互联网 Java 开发的学习历程技术汇总 旨在为大家提供一个清晰详细的学习教程 侧重点更倾向编写Java核心内容 如果本仓库能为您提供帮助 请给予支持 关注 点赞 分享 C 资源大全中文版 包括了 构建系统 编译器 数据库 加密 初中高的教程 指南 书籍 库等 NET m3u8 downloader 开源的命令行m3u8 HLS dash下载器 支持普通AES 128 CBC解密 多线程 自定义请求头等 支持简体中文 繁体中文和英文 English Supported 国内低代码平台从业者交流tcc transaction是TCC型事务java实现设计模式 Golang实现 研磨设计模式 读书笔记Vue数据可视化组件库 类似阿里DataV 大屏数据展示 提供SVG的边框及装饰 图表 水位图 飞线图等组件 简单易用 长期更新 React版已发布 自己动手做聊天机器人教程 RecyclerView侧滑菜单 Item拖拽 滑动删除Item 自动加载更多 HeaderView FooterView Item分组黏贴 腾讯物联网终端操作系统一个小巧 轻量的浏览器内核 用来取代wke和libcef包含美颜等4 余种实时滤镜相机 可拍照 录像 图片修改springboot 框架与其它组件结合如等用深度学习对对联 技术面试必备基础知识 Leetcode 计算机操作系统 计算机网络 系统设计 Java学习 面试指南 一份涵盖大部分 Java 程序员所需要掌握的核心知识 准备 Java 面试 首选 JavaGuide 用动画的形式呈现解LeetCode题目的思路 互联网 Java 工程师进阶知识完全扫盲 涵盖高并发 分布式 高可用 微服务 海量数据处理等领域知识mall项目是一套电商系统 包括前台商城系统及后台管理系统 基于SpringBoot MyBatis实现 采用Docker容器化部署 前台商城系统包含首页门户 商品推荐 商品搜索 商品展示 购物车 订单流程 会员中心 客户服务 帮助中心等模块 后台管理系统包含商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等模块 GitHub中文排行榜 帮助你发现高分优秀中文项目 更高效地吸收国人的优秀经验成果 榜单每周更新一次 敬请关注 算法面试 算法知识 针对小白的算法训练 还包括 1 阿里 字节 滴滴 百篇大厂面经汇总 2 千本开源电子书 3 百张思维导图 右侧来个 star 吧 English version supported 诊断利器Arthas教程 技术栈示例代码 快速简单上手教程 http下载工具 基于http代理 支持多连接分块下载阿里云计算平台团队出品 为监控而生的数据库连接池 企业级低代码平台 前后端分离架构强大的代码生成器让前后端代码一键生成 无需写任何代码 引领新的开发模式OnlineCoding 代码生成 手工MERGE 帮助Java项目解决7 %重复工作 让开发更关注业务 既能快速提高效率 帮助公司节省成本 同时又不失灵活性 下拉刷新 上拉加载 二级刷新 淘宝二楼智能下拉刷新框架 支持越界回弹 越界拖动 具有极强的扩展性 集成了几十种炫酷的Header和 Footer 该项目已成功集成 actuator 监控 admin 可视化监控 logback 日志 aopLog 通过AOP记录web请求日志 统一异常处理 json级别和页面级别 freemarker 模板引擎 thymeleaf 模板引擎 Beetl 模板引擎 Enjoy 模板引擎 JdbcTemplate 通用JDBC操作数据库 JPA 强大的ORM框架 mybatis 强大的ORM框架 通用Mapper 快速操作Mybatis PageHelper 通用的Mybatis分页插件 mybatis plus 快速操作Mybatis BeetlSQL 强大的ORM框架 u 微人事是一个前后端分离的人力资源管理系统 项目采用SpringBoot Vue开发 秒杀系统设计与实现 互联网工程师进阶与分析 To Be Top Javaer Java工程师成神之路循序渐进 学习博客Spring系列源码 mrbird cc快速 简单避免OOM的java处理Excel工具阿里巴巴 MySQL binlog 增量订阅 消费组件 一款优秀的开源博客发布应用 分布式任务调度平台XXL JOB 一款面向泛前端产品研发全生命周期的效率平台 面向云原生微服务的高可用流控防护组件 视频播放器支持弹幕 外挂字幕 支持滤镜 水印 gif截图 片头广告 中间广告 多个同时播放 支持基本的拖动 声音 亮度调节 支持边播边缓存 支持视频自带rotation的旋转 9 27 之类 重力旋转与手动旋转的同步支持 支持列表播放 列表全屏动画 视频加载速度 列表小窗口支持拖动 动画效果 调整比例 多分辨率切换 支持切换播放器 进度条小窗口预览 列表切换详情页面无缝播放 rtsp concat mpeg 又一个小商城 litemall Spring Boot后端 Vue管理员前端 微信小程序用户前端 Vue用户移动端基于Spring SpringMVC Mybatis分布式敏捷开发系统架构 提供整套公共微服务服务模块 集中权限管理 单点登录 内容管理 支付中心 用户管理 支持第三方登录 微信平台 存储系统 配置中心 日志分析 任务和通知等 支持服务治理 监控和追踪 努力为中小型企业打造全方位J2EE企业级开发解决方案 项目基于的前后端分离的后台管理系统 项目采用分模块开发方式 权限控制采用 RBAC 支持数据字典与数据权限管理 支持一键生成前后端代码 支持动态路由 史上最简单的Spring Cloud教程源码 CAT 作为服务端项目基础组件 提供了 Java C C Node js Python Go 等多语言客户端 已经在美团点评的基础架构中间件框架 MVC框架 RPC框架 数据库框架 缓存框架等 消息队列 配置系统等 深度集成 为美团点评各业务线提供系统丰富的性能指标 健康状况 实时告警等 spring boot 实践学习案例 是 spring boot 初学者及核心技术巩固的最佳实践 另外写博客 用 OpenWrite Spring Boot基础教程 Spring Boot 2 x版本连载中 帮助 Android App 进行组件化改造的路由框架 提高 Android UI 开发效率的 UI 库时间选择器 省市区三级联动 Luban 鲁班可能是最接近微信朋友圈的图片压缩算法 Gitee 最有价值开源项目 小而全而美的第三方登录开源组件 目前已支持Github Gitee 微博 钉钉 百度 Coding 腾讯云开发者平台 OSChina 支付宝 QQ 微信 淘宝 Google Facebook 抖音 领英 小米 微软 今日头条人人 华为 企业微信 酷家乐 Gitlab 美团 饿了么 推特 飞书 京东 阿里云 喜马拉雅 Amazon Slack和 Line 等第三方平台的授权登录 Login so easy 今日头条屏幕适配方案终极版 一个极低成本的 Android 屏幕适配方案 Banner 2 来了 Android广告图片轮播控件 内部基于ViewPager2实现 Indicator和UI都可以自定义 零代码 热更新 自动化 ORM 库 后端接口和文档零代码 前端 客户端 定制返回 JSON 的数据和结构一个涵盖六个专栏分布式消息队列 分布式事务的仓库 希望胖友小手一抖 右上角来个 Star 感恩 1 24Mybatis通用分页插件OkGo 3 震撼来袭 该库是基于 协议 封装了 OkHttp 的网络请求框架 比 Retrofit 更简单易用 支持 RxJava RxJava2 支持自定义缓存 支持批量断点下载管理和批量上传管理功能含 Flink 入门 概念 原理 实战 性能调优 源码解析等内容 涉及等内容的学习案例 还有 Flink 落地应用的大型项目案例 PVUV 日志存储 百亿数据实时去重 监控告警 分享 欢迎大家支持我的专栏 大数据实时计算引擎 Flink 实战与性能优化 安卓平台上的JavaScript自动化工具 ️一个整合了大量主流开源项目高度可配置化的 Android MVP 快速集成框架 Spring源码阅读大数据入门指南 android 4 4以上沉浸式状态栏和沉浸式导航栏管理 适配横竖屏切换 刘海屏 软键盘弹出等问题 可以修改状态栏字体颜色和导航栏图标颜色 以及不可修改字体颜色手机的适配 适用于一句代码轻松实现 以及对bar的其他设置 详见README 简书请参考 www jianshu com p 2a884e211a62业内为数不多致力于极致体验的超强全自研跨平台 windows android iOS 流媒体内核 通过模块化自由组合 支持实时RTMP推流 RTSP推流 RTMP播放器 RTSP播放器 录像 多路流媒体转发 音视频导播 动态视频合成 音频混音 直播互动 内置轻量级RTSP服务等 比快更快 业界真正靠谱的超低延迟直播SDK 1秒内 低延迟模式下2 4 ms DataX是阿里云DataWorks数据集成的开源版本 mall学习教程 架构 业务 技术要点全方位解析 mall项目 4 k star 是一套电商系统 使用现阶段主流技术实现 涵盖了等技术 采用Docker容器化部署 Android开源弹幕引擎 烈焰弹幕使 ～spring cloud vue oAuth2 全家桶实战 前后端分离模拟商城 完整的购物流程 后端运营平台 可以实现快速搭建企业级微服务项目 支持微信登录等三方登录 一个基于Spring Boot MyBatis的种子项目 用于快速构建中小型API RESTful API项目 强大 可定制 易扩展的 ViewPager 指示器框架 是的最佳替代品 支持角标 更支持在非ViewPager场景下使用 使用hide show 切换Fragment或使用se 27天成为Java大神安卓学习笔记 cim cross IM 适用于开发者的分布式即时通讯系统Android上一个优雅 万能自定义UI 仿iOS 支持垂直 水平方向切换 支持周视图 自定义周起始 性能高效的日历控件 支持热插拔实现的UI定制 支持标记 自定义颜色 农历 自定义月视图各种显示模式等 Canvas绘制 速度快 占用内存低 你真的想不到日历居然还可以如此优雅hsweb haʊs wɛb 是一个基于spring boot 2 x开发 首个使用全响应式编程的企业级后台管理系统基础项目 newbee mall 项目 新蜂商城 是一套电商系统 包括 newbee mall 商城系统及 newbee mall admin 商城后台管理系统 基于 Spring Boot 2 X 及相关技术栈开发 前台商城系统包含首页门户 商品分类 新品上线 首页轮播 商品推荐 商品搜索 商品展示 购物车 订单结算 订单流程 个人订单管理 会员中心 帮助中心等模块 后台管理系统包含数据面板 轮播图管理 商品管理 订单管理 会员管理 分类管理 设置等模块 mall swarm是一套微服务商城系统 采用了等核心技术 同时提供了基于Vue的管理后台方便快速搭建系统 mall swarm在电商业务的基础集成了注册中心 配置中心 监控中心 网关等系统功能 文档齐全 附带全套Spring Cloud教程 阅读是一款可以自定义来源阅读网络内容的工具 为广大网络文学爱好者提供一种方便 快捷舒适的试读体验 Spring Cloud基础教程 持续连载更新中阿里巴巴分布式数据库同步系统 解决中美异地机房 基于谷歌最新AAC架构 MVVM设计模式的一套快速开发库 整合OkRxJava Retrofit Glide等主流模块 满足日常开发需求 使用该框架可以快速开发一个高质量 易维护的Android应用 基于SpringCloud2 1的微服务开发脚手架 整合了等 服务治理方面引入等 让项目开发快速进入业务开发 而不需过多时间花费在架构搭建上 持续更新中基于SOA架构的分布式电商购物商城 前后端分离 前台商城 全家桶 后台管理系统等是 难得一见 的 Jetpack MVVM 最佳实践 在 以简驭繁 的代码中 对 视图控制器 乃至 标准化开发模式 形成正确 深入的理解 V部落 Vue SpringBoot实现的多用户博客管理平台 Android Signature V2 Scheme签名下的新一代渠道包打包神器即时通讯 IM 系统多种编程语言实现 LeetCode 剑指 Offer 第 2 版 程序员面试金典 第 6 版 题解专门为刚开始刷题的同学准备的算法基地 没有最细只有更细 立志用动画将晦涩难懂的算法说的通俗易懂 ansj分词 ict的真正java实现 分词效果速度都超过开源版的ict 中文分词 人名识别 词性标注 用户自定义词典 book 任阅 网络小说阅读器 3D翻页效果 txt pdf epub书籍阅读 Wifi传书 LeetCode刷题记录与面试整理mybatis generator界面工具 让你生成代码更简单更快捷Spring Cloud 学习案例 服务发现 服务治理 链路追踪 服务监控等 XPopup2 版本重磅来袭 2倍以上性能提升 带来可观的动画性能优化和交互细节的提升 功能强大 交互优雅 动画丝滑的通用弹窗 可以替代等组件 自带十几种效果良好的动画 支持完全的UI和动画自定义搜狐视频 sohu tv Redis私有云平台spring boot打造文件文档在线预览项目权限管理系统 预览地址 47 1 4 7 138 login零反射全动态Android插件框架分布式配置管理平台 通用 IM 聊天 UI 组件 已经同时支持 Android iOS RN 手把手教你整合最优雅SSM框架 SpringMVC Spring MyBatis换肤框架 极低的学习成本 极好的用户体验 一行 代码就可以实现换肤 你值得拥有 JVM 底层原理最全知识总结 国内首个Spring Cloud微服务化RBAC的管理平台 核心采用前端采用d2 admin中台框架 记得上边点个star 关注更新tcc transaction是TCC型事务java实现 RecyclerView侧滑菜单 Item拖拽 滑动删除Item 自动加载更多 HeaderView FooterView Item分组黏贴 包含美颜等4 余种实时滤镜相机 可拍照 录像 图片修改springboot 框架与其它组件结合如等安卓选择器类库 包括日期及时间选择器 可用于出生日期 营业时间等 单项选择器 可用于性别 民族 职业 学历 星座等 二三级联动选择器 可用于车牌号 基金定投日期等 城市地址选择器 分省级 地市级及区县级 数字选择器 可用于年龄 身高 体重 温度等 日历选日期择器 可用于酒店及机票预定日期 颜色选择器 文件及目录选择器等 Java工程师面试复习指南 本仓库涵盖大部分Java程序员所需要掌握的核心知识 整合了互联网上的很多优质Java技术文章 力求打造为最完整最实用的Java开发者学习指南 如果对你有帮助 给个star告诉我吧 谢谢 Android MVP 快速开发框架 做国内 示例最全面 注释最详细 使用最简单 代码最严谨 的 Android 开源 UI 框架几行代码快速集成二维码扫描功能MeterSphere 是一站式开源持续测试平台 涵盖测试跟踪 接口测试 性能测试 团队协作等功能 全面兼容 JMeter Postman Swagger 等开源 主流标准 记录各种学习笔记 算法 Java 数据库 并发 下一代Android打包工具 1 个渠道包只需要1 秒钟芋道 mall 商城 基于微服务的思想 构建在 B2C 电商场景下的项目实战 核心技术栈 是 Spring Boot Dubbo 未来 会重构成 Spring Cloud Alibaba Android 万能的等 支持多种Item类型的情况 lanproxy是一个将局域网个人电脑 服务器代理到公网的内网穿透工具 支持tcp流量转发 可支持任何tcp上层协议 访问内网网站 本地支付接口调试 ssh访问 远程桌面 目前市面上提供类似服务的有花生壳 TeamView GoToMyCloud等等 但要使用第三方的公网服务器就必须为第三方付费 并且这些服务都有各种各样的限制 此外 由于数据包会流经第三方 因此对数据安全也是一大隐患 技术交流QQ群 1 6742433 更优雅的驾车体验下载可以很简单 ️ 云阅 一款基于网易云音乐UI 使用玩架构开发的符合Google Material Design的Android客户端开源的 Material Design 豆瓣客户端一款针对系统PopupWindow优化的Popup库 功能强大 支持背景模糊 使用简单 你会爱上他的 PLDroidPlayer 是七牛推出的一款免费的适用于 Android 平台的播放器 SDK 采用全自研的跨平台播放内核 拥有丰富的功能和优异的性能 可高度定制化和二次开发 该项目已停止维护 9 Porn Android 客户端 突破游客每天观看1 次视频的限制 还可以下载视频 ️蓝绿 灰度 路由 限流 熔断 降级 隔离 追踪 流量染色 故障转移一本关于排序算法的 GitBook 在线书籍 十大经典排序算法 多语言实现 多种下拉刷新效果 上拉加载更多 可配置自定义头部广告位完全仿微信的图片选择 并且提供了多种图片加载接口 选择图片后可以旋转 可以裁剪成矩形或圆形 可以配置各种其他的参数SoloPi 自动化测试工具龙果支付系统 roncoo pay 是国内首款开源的互联网支付系统 拥有独立的账户体系 用户体系 支付接入体系 支付交易体系 对账清结算体系 目标是打造一款集成主流支付方式且轻量易用的支付收款系统 满足互联网业务系统打通支付通道实现支付收款和业务资金管理等功能 键盘面板冲突 布局闪动处理方案 咕泡学院实战项目 基于SpringBoot Dubbo构建的电商平台 微服务架构 商城 电商 微服务 高并发 kafka Elasticsearch停车场系统源码 停车场小程序 智能停车 Parking system 功能介绍 ①兼容市面上主流的多家相机 理论上兼容所有硬件 可灵活扩展 ②相机识别后数据自动上传到云端并记录 校验相机唯一id和硬件序列号 防止非法数据录入 ③用户手机查询停车记录详情可自主缴费 支持微信 支付宝 银行接口支付 支持每个停车场指定不同的商户进行收款 支付后出场在免费时间内会自动抬杆 ④支持app上查询附近停车场 导航 可用车位数 停车场费用 优惠券 评分 评论等 可预约车位 ⑤断电断网支持岗亭人员使用app可接管硬件进行停车记录的录入 技术架构 后端开发语言java 框架oauth2 spring 成长路线 但学到不仅仅是Java 业界首个支持渐进式组件化改造的Android组件化开源框架 支持跨进程调用SpringBoot2 从入门到实战 旨在打造在线最佳的 Java 学习笔记 含博客讲解和源码实例 包括 Java SE 和 Java WebJava诊断工具年薪百万互联网架构师课程文档及源码 公开部分 AndroidHttpCapture网络诊断工具 是一款Android手机抓包软件 主要功能包括 手机端抓包 PING DNS TraceRoute诊断 抓包HAR数据上传分享 你也可以看成是Android版的 Fiddler o 这可能是史上功能最全的Java权限认证框架 目前已集成 登录认证 权限认证 分布式Session会话 微服务网关鉴权 单点登录 OAuth2 踢人下线 Redis集成 前后台分离 记住我模式 模拟他人账号 临时身份切换 账号封禁 多账号认证体系 注解式鉴权 路由拦截式鉴权 花式token生成 自动续签 同端互斥登录 会话治理 密码加密 jwt集成 Spring集成 WebFlux集成 Android平台下的富文本解析器 支持Html和Markdown智能图片裁剪框架 自动识别边框 手动调节选区 使用透视变换裁剪并矫正选区 适用于身份证 名片 文档等照片的裁剪 俗名 可垂直跑 可水平跑的跑马灯 学名 可垂直翻 可水平翻的翻页公告 小马哥技术周报 Android Video Player 安卓视频播放器 封装模仿抖音并实现预加载 列表播放 悬浮播放 广告播放 弹幕 重学Java设计模式 是一本互联网真实案例实践书籍 以落地解决方案为核心 从实际业务中抽离出 交易 营销 秒杀 中间件 源码等22个真实场景 来学习设计模式的运用 欢迎关注小傅哥 微信 fustack 公众号 bugstack虫洞栈 博客 bugstack cnmybatis源码中文注释一款开源的GIF在线分享App 乐趣就要和世界分享 MPush开源实时消息推送系统在线云盘 网盘 OneDrive 云存储 私有云 对象存储 h5ai基于Spring Boot 2 x的一站式前后端分离快速开发平台XBoot 微信小程序 Uniapp 前端 Vue iView Admin 后端分布式限流 同步锁 验证码 SnowFlake雪花算法ID 动态权限 数据权限 工作流 代码生成 定时任务 社交账号 短信登录 单点登录 OAuth2开放平台 客服机器人 数据大屏 暗黑模式Guns基于SpringBoot 2 致力于做更简洁的后台管理系统 完美整合项目代码简洁 注释丰富 上手容易 同时Guns包含许多基础模块 用户管理 角色管理 部门管理 字典管理等1 个模块 可以直接作为一个后台管理系统的脚手架 Android 版本更新一个简洁而优雅的Android原生UI框架 解放你的双手 一套完整有效的android组件化方案 支持组件的组件完全隔离 单独调试 集成调试 组件交互 UI跳转 动态加载卸载等功能适用于Java和Android的快速 低内存占用的汉字转拼音库 Codes of my MOOC Course \\u003c我在慕课网上的课程 算法与数据结构 示例代码 包括C 和Java版本 课程的更多更新内容及辅助练习也将逐步添加进这个代码仓 Hope Boot 一款现代化的脚手架项目一个简单漂亮的SSM Spring SpringMVC Mybatis 博客系统根据Gson库使用的要求 将JSONObject格式的String 解析成实体B站 哔哩哔哩 Bilibili 自动签到投币工具 每天轻松获取65经验值 支持每日自动投币 银瓜子兑换硬币 领取大会员福利 大会员月底给自己充电等功能 呐 赶快和我一起成为Lv6吧 IJPay 让支付触手可及 封装了微信支付 QQ支付 支付宝支付 京东支付 银联支付 PayPal 支付等常用的支付方式以及各种常用的接口 不依赖任何第三方 mvc 框架 仅仅作为工具使用简单快速完成支付模块的开发 可轻松嵌入到任何系统里 右上角点下小星星 High quality pure Weex demo 网易严选 App 感受 Weex 开发Android 快速实现新手引导层的库 通过简洁链式调用 一行代码实现引导层的显示通过标签直接生成shape 无需再写shape xml 本库是一款基于RxJava2 Retrofit2实现简单易用的网络请求框架 结合android平台特性的网络封装库 采用api链式调用一点到底 集成cookie管理 多种缓存模式 极简https配置 上传下载进度显示 请求错误自动重试 请求携带token 时间戳 签名sign动态配置 自动登录成功后请求重发功能 3种层次的参数设置默认全局局部 默认标准ApiResult同时可以支持自定义的数据结构 已经能满足现在的大部分网络请求 Android BLE蓝牙通信库 基于Flink实现的商品实时推荐系统 flink统计商品热度 放入redis缓存 分析日志信息 将画像标签和实时记录放入Hbase 在用户发起推荐请求后 根据用户画像重排序热度榜 并结合协同过滤和标签两个推荐模块为新生成的榜单的每一个产品添加关联产品 最后返回新的用户列表 播放器基础库 专注于播放视图组件的高复用性和组件间的低耦合 轻松处理复杂业务 图片选择库 单选 多选 拍照 裁剪 压缩 自定义 包括视频选择和录制 DataX集成可视化页面 选择数据源即可一键生成数据同步任务 支持等数据源 批量创建RDBMS数据同步任务 集成开源调度系统 支持分布式 增量同步数据 实时查看运行日志 监控执行器资源 KILL运行进程 数据源信息加密等 Deprecated android 自定义日历控件 支持左右无限滑动 周月切换 标记日期显示 自定义显示效果跳转到指定日期一个通过动态加载本地皮肤包进行换肤的皮肤框架这是RedSpider社区成员原创与维护的Java多线程系列文章 一站式Apache Kafka集群指标监控与运维管控平台快速开发工具类收集 史上最全的开发工具类 欢迎Follow Fork Star后端技术总结 包括Java基础 JVM 数据库 mysql redis 计算机网络 算法 数据结构 操作系统 设计模式 系统设计 框架原理 最佳阅读地址Android源码设计模式分析项目可能是最好的支付SDK 停止维护 组件化综合案例 包含微信新闻 头条视频 美女图片 百度音乐 干活集中营 玩Android 豆瓣读书电影 知乎日报等等模块 架构模式 组件化阿里VLayout 腾讯X5 腾讯bugly 融合开发中需要的各种小案例 开源OA系统 码云GVP Java开源oa 企业OA办公平台 企业OA 协同办公OA 流程平台OA O2OA OA 支持国产麒麟操作系统和国产数据库 达梦 人大金仓 政务OA 军工信息化OA以Spring Cloud Netflix作为服务治理基础 展示基于tcc思想所实现的分布式事务解决方案一个帮助您完成从缩略视图到原视图无缝过渡转变的神奇框架 系统重构与迁移指南 手把手教你分析 评估现有系统 制定重构策略 探索可行重构方案 搭建测试防护网 进行系统架构重构 服务架构重构 模块重构 代码重构 数据库重构 重构后的架构守护版本检测升级 更新 库小说精品屋是一个多平台 web 安卓app 微信小程序 功能完善的屏幕自适应小说漫画连载系统 包含精品小说专区 轻小说专区和漫画专区 包括小说 漫画分类 小说 漫画搜索 小说 漫画排行 完本小说 漫画 小说 漫画评分 小说 漫画在线阅读 小说 漫画书架 小说 漫画阅读记录 小说下载 小说弹幕 小说 漫画自动采集 更新 纠错 小说内容自动分享到微博 邮件自动推广 链接自动推送到百度搜索引擎等功能 Android 徽章控件 致力于打造一款极致体验的 www wanandroid com 客户端 知识和美是可以并存的哦QAQn ≧ ≦ n 从源码层面 剖析挖掘互联网行业主流技术的底层实现原理 为广大开发者 “提升技术深度” 提供便利 目前开放 Spring 全家桶 Mybatis Netty Dubbo 框架 及 Redis Tomcat 中间件等Redis 一站式管理平台 支持集群的监控 安装 管理 告警以及基本的数据操作该项目不再维护 仅供学习参考专注批量推送的小而美的工具 目前支持 模板消息 公众号 模板消息 小程序 微信客服消息 微信企业号 企业微信消息 阿里云短信 阿里大于模板短信 腾讯云短信 云片网短信 E Mail HTTP请求 钉钉 华为云短信 百度云短信 又拍云短信 七牛云短信Android 平台开源天气 App 采用等开源库来实现 SpringBoot 相关漏洞学习资料 利用方法和技巧合集 黑盒安全评估 check listAndroid 权限请求框架 已适配 Android 11微信SDK JAVA 公众平台 开放平台 商户平台 服务商平台 QMQ是去哪儿网内部广泛使用的消息中间件 自2 12年诞生以来在去哪儿网所有业务场景中广泛的应用 包括跟交易息息相关的订单场景 也包括报价搜索等高吞吐量场景 Java 23种设计模式全归纳linux运维监控工具 支持系统信息 内存 cpu 温度 磁盘空间及IO 硬盘smart 系统负载 网络流量 进程等监控 API接口 大屏展示 拓扑图 端口监控 docker监控 日志文件监控 数据可视化 webSSH工具 堡垒机 跳板机 这可能是全网最好用的ViewPager轮播图 简单 高效 一行代码实现循环轮播 一屏三页任意变 指示器样式任你挑 一种简单有效的android组件化方案 支持组件的代码资源隔离 单独调试 集成调试 组件交互 UI跳转 生命周期等完整功能 一个强大 1 % 兼容 支持 AndroidX 支持 Kotlin并且灵活的组件化框架JPress 一个使用 Java 开发的建站神器 目前已经有 1 w 网站使用 JPress 进行驱动 其中包括多个政府机构 2 上市公司 中科院 红 字会等 分布式事务易用的轻量化网络爬虫 Android系统源码分析重构中一款免费的数据可视化工具 报表与大屏设计 类似于excel操作风格 在线拖拽完成报表设计 功能涵盖 报表设计 图形报表 打印设计 大屏设计等 永久免费 秉承“简单 易用 专业”的产品理念 极大的降低报表开发难度 缩短开发周期 节省成本 解决各类报表难题 Android Activity 滑动返回 支持微信滑动返回样式 横屏滑动返回 全屏滑动返回SpringBoot 基础教程 从入门到上瘾 基于2 M5制作 仿微信视频拍摄UI 基于ffmpeg的视频录制编辑Python 1 天从新手到大师 分享 GitHub 上有趣 入门级的开源项目中英文敏感词 语言检测 中外手机 电话归属地 运营商查询 名字推断性别 手机号抽取 身份证抽取 邮箱抽取 中日文人名库 中文缩写库 拆字词典 词汇情感值 停用词 反动词表 暴恐词表 繁简体转换 英文模拟中文发音 汪峰歌词生成器 职业名称词库 同义词库 反义词库 否定词库 汽车品牌词库 汽车零件词库 连续英文切割 各种中文词向量 公司名字大全 古诗词库 IT词库 财经词库 成语词库 地名词库 历史名人词库 诗词词库 医学词库 饮食词库 法律词库 汽车词库 动物词库 中文聊天语料 中文谣言数据 百度中文问答数据集 句子相似度匹配算法集合 bert资源 文本生成 摘要相关工具 cocoNLP信息抽取 2 21年最新总结 阿里 腾讯 百度 美团 头条等技术面试题目 以及答案 专家出题人分析汇总 AiLearning 机器学习 MachineLearning ML 深度学习 DeepLearning DL 自然语言处理 NLP123 6智能刷票 订票结巴中文分词 动手学深度学习 面向中文读者 能运行 可讨论 中英文版被全球175所大学采用教学 中文分词 词性标注 命名实体识别 依存句法分析 语义依存分析 新词发现 关键词短语提取 自动摘要 文本分类聚类 拼音简繁转换 自然语言处理微信个人号接口 微信机器人及命令行微信 三十行即可自定义个人号机器人 数据结构和算法必知必会的5 个代码实现JumpServer 是全球首款开源的堡垒机 是符合 4A 的专业运维安全审计系统 飞桨 核心框架 深度学习 机器学习高性能单机 分布式训练和跨平台部署 中国程序员容易发音错误的单词微信 跳一跳 Python 辅助 python模拟登陆一些大型网站 还有一些简单的爬虫 希望对你们有所帮助 ️ 如果喜欢记得给个star哦 网络爬虫实战 淘宝 京东 网易云 B站 123 6 抖音 笔趣阁 漫画小说下载 音乐电影下载等Python爬虫代理IP池 proxy pool wtfpython的中文翻译 施工结束 能力有限 欢迎帮我改进翻译提供多款 Shadowrocket 规则 带广告过滤功能 用于 iOS 未越狱设备选择性地自动翻墙 123 6 购票助手 支持集群 多账号 多任务购票以及 Web 页面管理 walle 瓦力 Devops开源项目代码部署平台一些非常有趣的python爬虫例子 对新手比较友好 主要爬取淘宝 天猫 微信 豆瓣 QQ等网站机器学习相关教程1 Chinese Word Vectors 上百种预训练中文词向量 网易云音乐命令行版本一款入门级的人脸 视频 文字检测以及识别的项目 编程随想 整理的 太子党关系网络 专门揭露赵国的权贵微信助手 1 每日定时给好友 女友 发送定制消息 2 机器人自动回复好友 3 群助手功能 例如 查询垃圾分类 天气 日历 电影实时票房 快递物流 PM2 5等 二维码生成器 支持 gif 动态图片二维码 阿布量化交易系统 股票 期权 期货 比特币 机器学习 基于python的开源量化交易 量化投资架构 book 中华新华字典数据库 包括歇后语 成语 词语 汉字 Git AWS Google 镜像 SS SSR VMESS节点行业研究报告的知识储备库中文翻译手写实现李航 统计学习方法 书中全部算法 Python 抖音机器人 论如何在抖音上找到漂亮小姐姐？ 迁移学习python爬虫教程系列 从 到1学习python爬虫 包括浏览器抓包 手机APP抓包 如 fiddler mitmproxy 各种爬虫涉及的模块的使用 如等 以及IP代理 验证码识别 Mysql MongoDB数据库的python使用 多线程多进程爬虫的使用 css 爬虫加密逆向破解 JS爬虫逆向 分布式爬虫 爬虫项目实战实例等Python脚本 模拟登录知乎 爬虫 操作excel 微信公众号 远程开机越来越多的网站具有反爬虫特性 有的用图片隐藏关键数据 有的使用反人类的验证码 建立反反爬虫的代码仓库 通过与不同特性的网站做斗争 无恶意 提高技术 欢迎提交难以采集的网站 因工作原因 项目暂停 人人影视bot 完全对接人人影视全部无删减资源莫烦Python 中文AI教学飞桨 官方模型库 包含多种学术前沿和工业场景验证的深度学习模型 轻量级人脸检测模型 百度云 百度网盘Python客户端 Python进阶 Intermediate Python 中文版 提供同花顺客户端 国金 华泰客户端 雪球的基金 股票自动程序化交易以及自动打新 支持跟踪 joinquant ricequant 模拟交易 和 实盘雪球组合 量化交易组件QUANTAXIS 支持任务调度 分布式部署的 股票 期货 期权 港股 虚拟货币 数据 回测 模拟 交易 可视化 多账户 纯本地量化解决方案INFO SPIDER 是一个集众多数据源于一身的爬虫工具箱 旨在安全快捷的帮助用户拿回自己的数据 工具代码开源 流程透明 支持数据源包括GitHub QQ邮箱 网易邮箱 阿里邮箱 新浪邮箱 Hotmail邮箱 Outlook邮箱 京东 淘宝 支付宝 中国移动 中国联通 中国电信 知乎 哔哩哔哩 网易云音乐 QQ好友 QQ群 生成朋友圈相册 浏览器浏览历史 123 6 博客园 CSDN博客 开源中国博客 简书 中文BERT wwm系列模型 Python入门网络爬虫之精华版中文 iOS Mac 开发博客列表Python网页微信APIpkuseg多领域中文分词工具自己动手做聊天机器人教程基于搜狗微信搜索的微信公众号爬虫接口用深度学习对对联 v2ray xray多用户管理部署程序各种脚本 关于 虾米 xiami com 百度网盘 pan baidu com 115网盘 115 com 网易音乐 music 163 com 百度音乐 music baidu com 36 网盘 云盘 yunpan cn 视频解析 flvxz com bt torrent ￼ magnet ed2k 搜索 tumblr 图片下载 unzip查看被删的微信好友定投改变命运 让时间陪你慢慢变富 onregularinvesting com 机器学习实战 Python3 kNN 决策树 贝叶斯 逻辑回归 SVM 线性回归 树回归Statistical learning methods 统计学习方法 第2版 李航 笔记 代码 notebook 参考文献 Errata lihang stock 股票系统 使用python进行开发 基于深度学习的中文语音识别系统京东抢购助手 包含登录 查询商品库存 价格 添加 清空购物车 抢购商品 下单 查询订单等功能莫烦Python 中文AI教学机器学习算法python实现新浪微博爬虫 用python爬取新浪微博数据的算法以及通用生成对抗网络图像生成的理论与实践研究 青岛大学开源 Online Judge QQ群 49671 125 admin qduoj comWeRoBot 是一个微信公众号开发框架 基于Django的博客系统 中文近义词 聊天机器人 智能问答工具包开源财经数据接口库巡风是一款适用于企业内网的漏洞快速应急 巡航扫描系统 番号大全 解决电脑 手机看电视直播的苦恼 收集各种直播源 电视直播网站知识图谱构建 自动问答 基于kg的自动问答 以疾病为中心的一定规模医药领域知识图谱 并以该知识图谱完成自动问答与分析服务 出处本地电影刮削与整理一体化解决方案自动化运维平台 CMDB CD DevOps 资产管理 任务编排 持续交付 系统监控 运维管理 配置管理 wukong robot 是一个简单 灵活 优雅的中文语音对话机器人 智能音箱项目 还可能是首个支持脑机交互的开源智能音箱项目 获取斗鱼 虎牙 哔哩哔哩 抖音 快手等 55 个直播平台的真实流媒体地址 直播源 和弹幕 直播源可在 PotPlayer flv js 等播放器中播放 宝塔Linux面板 简单好用的服务器运维面板农业知识图谱 AgriKG 农业领域的信息检索 命名实体识别 关系抽取 智能问答 辅助决策CODO是一款为用户提供企业多混合云 一站式DevOps 自动化运维 完全开源的云管理平台 自动化运维平台Web Pentesting Fuzz 字典 一个就够了 计算机网络 自顶向下方法 原书第6版 编程作业 Wireshark实验文档的翻译和解答 中文古诗自动作诗机器人 屌炸天 基于tensorflow1 1 api 正在积极维护升级中 快star 保持更新 PyQt Examples PyQt各种测试和例子 PyQt4 PyQt5海量中文预训练ALBERT模型汉字转拼音 pypinyin 数据结构与算法 leetcode lintcode题解 Pytorch模型训练实用教程 中配套代码实时获取新浪 腾讯 的免费股票行情 集思路的分级基金行情Python爬虫 Flask网站 免费ShadowSocks账号 ssr订阅 json 订阅实战 多种网站 电商数据爬虫 包含 淘宝商品 微信公众号 大众点评 企查查 招聘网站 闲鱼 阿里任务 博客园 微博 百度贴吧 豆瓣电影 包图网 全景网 豆瓣音乐 某省药监局 搜狐新闻 机器学习文本采集 fofa资产采集 汽车之家 国家统计局 百度关键词收录数 蜘蛛泛目录 今日头条 豆瓣影评 携程 小米应用商店 安居客 途家民宿 ️ ️ ️ 微信爬虫展示项目 SQL 审核查询平台团子翻译器 个人兴趣制作的一款基于OCR技术的翻译器自动化运维平台 代码及应用部署CI CD 资产管理CMDB 计划任务管理平台 SQL审核 回滚 任务调度 站内WIKISource Code Security Audit 源代码安全审计 Exphub 漏洞利用脚本库 包括的漏洞利用脚本 最新添加我的自学笔记 终身更新 当前专注System基础 MLSys 使用机器学习算法完成对123 6验证码的自动识别Python 开源项目之 自学编程之路 保姆级教程 AI实验室 宝藏视频 数据结构 学习指南 机器学习实战 深度学习实战 网络爬虫 大厂面经 程序人生 资源分享 中文文本分类基于pytorch 开箱即用 根据网易云音乐的歌单 下载flac无损音乐到本地腾讯优图高精度双分支人脸检测器文本纠错等模型实现 开箱即用 3 天掌握量化交易 持续更新 中文分词 词性标注 命名实体识别 依存句法分析 新词发现 关键词短语提取 自动摘要 文本分类聚类 拼音简繁 自然语言处理中文公开聊天语料库豆瓣读书的爬虫总结梳理自然语言处理工程师 NLP 需要积累的各方面知识 包括面试题 各种基础知识 工程能力等等 提升核心竞争力中文自然语言处理数据集 平时做做实验的材料 欢迎补充提交合并 一个可以自己进行训练的中文聊天机器人 根据自己的语料训练出自己想要的聊天机器人 可以用于智能客服 在线问答 智能聊天等场景 目前包含seq2seq seqGAN版本 tf2 版本 pytorch版本 股票量化框架 支持行情获取以及交易微博爬虫 持续维护 Bilibili 用户爬虫 deepin源移植 Debian Ubuntu上最快的QQ 微信安装方式 新闻网页正文通用抽取器 Beta 版 flag on post 自动更新域名解析到本机IP 支持dnspod 阿里DNS CloudFlare 华为云 DNSCOM 本项目针对字符型图片验证码 使用tensorflow实现卷积神经网络 进行验证码识别 owllook 小说搜索引擎中文语言理解测评基准python中文库 python人工智能大数据自动化接口测试开发 书籍下载及python库汇总china testing github io 2 19新型冠状病毒疫情时间序列数据仓库Python 黑魔法手册单阶段通用目标检测器一个拍照做题程序 输入一张包含数学计算题的图片 输出识别出的数学计算式以及计算结果 video download B站视频下载中文命名实体识别 TensorFlow Python 中文数据结构和算法教程 验证码识别 训练Python爬虫实战 模拟登陆各大网站 包含但不限于 滑块验证 拼多多 美团 百度 bilibili 大众点评 淘宝 如果喜欢请start ️学无止下载器 慕课下载器 Mooc下载 慕课网下载 中国大学下载 爱课程下载 网易云课堂下载 学堂在线下载 超星学习通下载 支持视频 课件同时下载一个高级web目录 文件扫描工具 功能将会强于DirBuster Dirsearch cansina 御剑 搜索所有中文NLP数据集 附常用英文NLP数据集中文实体识别与关系提取2 19新型冠状病毒疫情实时爬虫及github release archive以及项目文件的加速项目安卓应用安全学习抓取大量免费代理 ip 提取有效 ip 使用RoBERTa中文预训练模型 RoBERTa for Chinese 用于训练中英文对话系统的语料库敏感词过滤的几种实现 某1w词敏感词库简单易用的Python爬虫框架 QQ交流群 59751 56 使用Bert ERNIE 进行中文文本分类为 CSAPP 视频课程提供字幕 翻译 PPT Lab PyTorch 官方中文教程包含 6 分钟快速入门教程 强化教程 计算机视觉 自然语言处理 生成对抗网络 强化学习 欢迎 Star Fork 兜哥出品 \\u003c一本开源的NLP入门书籍 图像翻译 条件GAN AI绘画用Resnet1 1 GPT搭建一个玩王者荣耀的AI各种漏洞poc Exp的收集或编写斗地主AIVulmap 是一款 web 漏洞扫描和验证工具 可对 webapps 进行漏洞扫描 并且具备漏洞验证功能提供超過 5 個金融資料 台股為主 每天更新 finmind github io 数据接口 百度 谷歌 头条 微博指数 宏观数据 利率数据 货币汇率 千里马 独角兽公司 新闻联播文字稿 影视票房数据 高校名单 疫情数据 PyOne 一款给力的onedrive文件管理 分享程序 使用开发的用于迅速搭建并使用 WebHook 进行自动化部署和运维 支持 Github GitLab Gogs GitOsc 跟我一起写Makefile重制版 python自动化运维 技术与最佳实践 书中示例及案例源码自然语言处理实验 sougou数据集 TF IDF 文本分类 聚类 词向量 情感识别 关系抽取等微信公众平台 Python 开发包 DEPRECATED Weblogic一键漏洞检测工具 V1 5 更新时间 2 2 73 完备优雅的微信公众号接口 原生支持同步 协程使用 本程序旨在为安全应急响应人员对Linux主机排查时提供便利 实现主机侧Checklist的自动全面化检测 根据检测结果自动数据聚合 进行黑客攻击路径溯源 PyCharm 中文指南 安装 破解 效率 技巧类似按键精灵的鼠标键盘录制和自动化操作 模拟点击和键入GPT2 for Chinese chitchat 用于中文闲聊的GPT2模型 实现了DialoGPT的MMI思想 中华人民共和国国家标准 GB T 226 行政区划代码基于python的量化交易平台中文语音识别基于 OneBot 标准的 Python 异步 QQ 机器人框架Real World Masked Face Dataset 口罩人脸数据集 Vulfocus 是一个漏洞集成平台 将漏洞环境 docker 镜像 放入即可使用 开箱即用 谷歌 百度 必应图片下载 基于方面的情感分析 使用PyTorch实现 深度学习与计算机视觉 配套代码ART环境下自动化脱壳方案利用网络上公开的数据构建一个小型的证券知识图谱 知识库中文资源精选 官方网站 安装教程 入门教程 视频教程 实战项目 学习路径 QQ群 167122861 公众号 磐创AI 微信群二维码 www tensorflownews com Pre Trained Chinese XLNet 中文XLNet预训练模型 新浪微博Python SDK有关burpsuite的插件 非商店 文章以及使用技巧的收集 此项目不再提供burpsuite破解文件 如需要请在博客mrxn net下载Python3编写的CMS漏洞检测框架基于django的工作流引擎 工单 ️ 哔哩云 不支持任意文件的全速上传与下载微信SDK 包括微信支付 微信公众号 微信登陆 微信消息处理等中文自然语言理解堡垒机 云桌面自动化运维 审计 录像 文件管理 sftp上传 实时监控 录像回放 网页版rz sz上传下载 动态口令 django转换中国知网 CAJ 格式文献为 PDF 佛系转换 成功与否 皆是玄学 HanLP作者的新书 自然语言处理入门 详细笔记 业界良心之作 书中不是枯燥无味的公式罗列 而是用白话阐述的通俗易懂的算法模型 从基本概念出发 逐步介绍中文分词 词性标注 命名实体识别 信息抽取 文本聚类 文本分类 句法分析这几个热门问题的算法原理与工程实现 Python3 网络爬虫实战 部分含详细教程 猫眼 腾讯视频 豆瓣 研招网 微博 笔趣阁小说 百度热点 B站 CSDN 网易云阅读 阿里文学 百度股票 今日头条 微信公众号 网易云音乐 拉勾 有道 unsplash 实习僧 汽车之家 英雄联盟盒子 大众点评 链家 LPL赛程 台风 梦幻西游 阴阳师藏宝阁 天气 牛客网 百度文库 睡前故事 知乎 Wish微信公众号文章的爬虫 Python Web开发实战 书中源码一直可用的GoAgent 会定时扫描可用的google gae ip 提供可自动化获取ip运行的版本层剪枝 通道剪枝 知识蒸馏 中文命名实体识别 包括多种模型 HMM CRF BiLSTM BiLSTM CRF的具体实现 The Way to Go 中文译本 中文正式名 Go 入门指南 谢谢 Solutions to LeetCode by Go 1 % test coverage runtime beats 1 % LeetCode 题解一款轻量级 高性能 功能强大的内网穿透代理服务器 支持tcp udp socks5 http等几乎所有流量转发 可用来访问内网网站 本地支付接口调试 ssh访问 远程桌面 内网dns解析 内网socks5代理等等 并带有功能强大的web管理端 Go语言高级编程 开源图书 涵盖CGO Go汇编语言 RPC实现 Protobuf插件实现 Web框架实现 分布式系统等高阶主题 完稿 是一个跨平台的强加密无特征的代理软件 零配置 算法模板 最科学的刷题方式 最快速的刷题路径 你值得拥有 百度网盘不限速客户端 golang qt5 跨平台图形界面是golang实现的高性能http https websocket tcp socks5代理服务器 支持内网穿透 链式代理 通讯加密 夜读 通过 bilibili 在线直播的方式分享 Go 相关的技术话题 每天大家在微信 telegram Slack 上及时沟通交流编程技术话题 支持多家云存储的云盘系统 这里是写博客的地方 Halfrost Field 冰霜之地Lantern官方版本下载 蓝灯 翻墙 代理 科学上网 外网 加速器 梯子 路由基于gin vue搭建的后台管理系统框架 集成jwt鉴权 权限管理 动态路由 分页封装 多点登录拦截 资源权限 上传下载 代码生成器 表单生成器 通用工作流等基础功能 五分钟一套CURD前后端代码 目VUE3版本正在重构 欢迎issue和pr 分布式爬虫管理平台 支持任何语言和框架Golang标准库 对于程序员而言 标准库与语言本身同样重要 它好比一个百宝箱 能为各种常见的任务提供完美的解决方案 以示例驱动的方式讲解Golang的标准库 天用Go动手写 从零实现系列是一个高性能且低损耗的 goroutine 池 有 有 设计模式 Golang实现 研磨设计模式 读书笔记Golang实现的基于beego框架的接口在线文档管理系统高性能开源RTSP流媒体服务器 基于go语言研发 维护和优化 RTSP推模式转发 RTSP拉模式转发 是一个高性能 轻量级 非阻塞的事件驱动 Go 网络框架 基于Gin Vue Element UI的前后端分离权限管理系统脚手架 包含了 多租户的支持 基础用户管理功能 jwt鉴权 代码生成器 RBAC资源控制 表单构建 定时任务等 3分钟构建自己的中后台项目 文档蓝鲸智云配置平台 BlueKing CMDB 今日热榜 一个获取各大热门网站热门头条的聚合网站 使用Go语言编写 多协程异步快速抓取信息 预览 mo fish一条命令离线安装高可用kubernetes 3min装完 7 M 1 年证书 生产环境稳如老狗阿里巴巴开源的一款简单易用 功能强大的混沌实验注入工具 Go语言四十二章经 详细讲述Go语言规范与语法细节及开发中常见的误区 通过研读标准库等经典代码设计模式 启发读者深刻理解Go语言的核心思维 进入Go语言开发的更高阶段 ️一个轻巧的网络混淆代理 基于Golang轻量级TCP并发服务器框架定时任务管理系统KubeOperator 是一个开源的轻量级 Kubernetes 发行版 专注于帮助企业规划 部署和运营生产级别的 K8s 集群 本系统是集工单统计 任务钩子 权限管理 灵活配置流程与模版等等于一身的开源工单系统 当然也可以称之为工作流引擎 致力于减少跨部门之间的沟通 自动任务的执行 提升工作效率与工作质量 减少不必要的工作量与人为出错率 Go实现的Trojan代理 支持多路复用 路由功能 CDN中转 Shadowsocks混淆插件 多平台 无依赖 Go语法树入门 开启自制编程语言和编译器之旅 开源免费图书 Go语言进阶 掌握抽象语法树 Go语言AST 凹语言 一款可全平台运行的浏览器数据导出解密工具 Golang相关 审稿进度8 % Go语法 Go并发思想 Go与web开发 Go微服务设施等Jupiter是斗鱼开源的面向服务治理的Golang微服务框架Elasticsearch 可视化DashBoard 支持Es监控 实时搜索 Index template快捷替换修改 索引列表信息查看 SQL converts to DSL等 从问题切入 串连 Go 语言相关的所有知识 融会贯通 golang design go questionsWeChat SDK for Go 微信SDK 简单 易用 go fastdfs 是一个简单的分布式文件系统 私有云存储 具有无中心 高性能 高可靠 免维护等优点 支持断点续传 分块上传 小文件合并 自动同步 自动修复 Mastering GO 中文译本 玩转 GO 云原生且易用的应用管理平台 Go Web 基础 是一套针对 Google 出品的 Go 语言的视频语音教程 主要面向完成 Go 编程基础 教程后希望进一步了解有关 Go Web 开发的学习者 中文名 悟空 API 网关 是一个基于 Golang开发的微服务网关 能够实现高性能 HTTP API 转发 服务编排 多租户管理 API 访问权限控制等目的 拥有强大的自定义插件系统可以自行扩展 并且提供友好的图形化配置界面 能够快速帮助企业进行 API 服务治理 提高 API 服务的稳定性和安全性 集合多家 API 的新一代图床MIT课程 Distributed Systems 学习和翻译Go语言圣经中文版 只接收PR Issue请提交到golang china gopl zh trojan多用户管理部署程序 支持web页面管理BookStack 基于MinDoc 使用Beego开发的在线文档管理系统 功能类似Gitbook和看云 weixin wechat 微信公众平台 微信企业号 微信商户平台 微信支付 go golang sdk 蓝眼云盘 Eyeblue Cloud Storage 语言高性能编程 Go 语言陷阱 Gotchas Traps 使用 XMind 记录 Linux 操作系统 网络 C Golang 以及数据库的一些设计cqhttp的golang实现 轻量 原生跨平台 mqant是一款基于Golang语言的简洁 高效 高性能的分布式微服务框架基于react node js go开发的微商城 含微信小程序 MM Wiki 一个轻量级的企业知识分享与团队协同软件 可用于快速构建企业 Wiki 和团队知识分享平台 部署方便 使用简单 帮助团队构建一个信息共享 文档管理的协作环境 Go 语言中文网 Golang中文社区 Go语言学习园地 源码基于 Gin 进行模块化设计的 API 框架 封装了常用功能 使用简单 致力于进行快速的业务研发 比如 支持 cors 跨域 jwt 签名验证 zap 日志收集 panic 异常捕获 trace 链路追踪 prometheus 监控指标 swagger 文档生成 viper 配置文件解析 gorm 数据库组件 gormgen 代码生成工具 graphql 查询语言 errno 统一定义错误码 gRPC 的使用 等等 syncd是一款开源的代码部署工具 它具有简单 高效 易用等特点 可以提高团队的工作效率 一款由 YSRC 开源的主机入侵检测系统golang面试题集合这是一个可以识别视频语音自动生成字幕SRT文件的开源 Windows GUI 软件工具 一款内网综合扫描工具 方便一键自动化 全方位漏扫扫描 是一个用于在两个redis之间同步数据的工具 满足用户非常灵活的同步 迁移需求 Overlord是哔哩哔哩基于Go语言编写的memcache和redis cluster的代理及集群管理功能 致力于提供自动化高可用的缓存服务解决方案 Stack RPC 中文示例 教程 资料 源码解读ICMP流量伪装转发工具Freedom是一个基于六边形架构的框架 可以支撑充血的领域模型范式 Go2编程指南 开源图书 重点讲解Go2新特性 以及Go1教程中较少涉及的特性语言高性能分词golang写的IM服务器 服务组件形式 结巴 中文分词的Golang版本xorm是一个简单而强大的Go语言ORM库 通过它可以使数据库操作非常简便 本库是基于原版xorm的定制增强版本 为xorm提供类似ibatis的配置文件及动态SQL支持 支持AcitveRecord操作一个 Go 语言实现的快速 稳定 内嵌的 k v 数据库 高性能表格数据导出器基于Golang的开源社区系统 版本网易云音乐ncm文件格式转换 go 实现的压测工具 ab locust Jmeter压测工具介绍 单台机器1 w连接压测实战 抓包截取项目中的数据库请求并解析成相应的语句 Go专家编程 Go语言快速入门 轻松进阶 \\u003c\\u003c自己动手写docker 源码Go 每日一库kunpeng是一个Golang编写的开源POC框架 库 以动态链接库的形式提供各种语言调用 通过此项目可快速开发漏洞检测类的系统 vue js element框架 golang beego框架 开发的运维发布系统 支持git jenkins版本发布 go ssh BT两种文件传输方式选择 支持部署前准备任务和部署后任务钩子函数 Go 从入门到实战 学习笔记 从零开始学 Go Gin 框架 基本语法包括 26 个Demo Gin 框架包括 Gin 自定义路由配置 Gin 使用 Logrus 进行日志记录 Gin 数据绑定和验证 Gin 自定义错误处理 Go gRPC Hello World 持续更新中 Go 学习之路 Go 开发者博客 Go 微信公众号 Go 学习资料 文档 书籍 视频 微信 WeChat 支付宝 AliPay 的Go版本SDK 极简 易用的聚合支付SDK Go by Example 通过例子学 GolangPPGo Job是一款可视化的 多人多权限的 一任务多机执行的定时任务管理系统 采用golang开发 安装方便 资源消耗少 支持大并发 可同时管理多台服务器上的定时任务 Golang实现的IP代理池是一款用Go语言开发的web应用框架 API特性类似于Tornado并且拥有比Tornado更好的性能 自己动手写Java虚拟机 随书源代码支付宝 AliPay SDK for Go 集成简单 功能完善 持续更新 支持公钥证书和普通公钥进行签名和验签 ARCHIVED Geph 迷霧通帮助你将本地端口暴露在外网 支持TCP UDP 当然也支持HTTP 深入Go并发编程研讨课无状态子域名爆破工具手机号码归属地信息库 手机号归属地查询 phone dat 最后更新 2 21年 6月 golang基于websocket单台机器支持百万连接分布式聊天 IM 系统基于mongodb oplog的集群复制工具 可以满足迁移和同步的需求 进一步实现灾备和多活功能 Gin Gorm开发Golang API快速开发脚手架简单可信赖的任务管理工具Go语言实例教程从入门到进阶 包括基础库使用 设计模式 面试易错点 工具类 对接第三方等授权框架简体中文翻译 自动抓取tg频道 订阅地址 公开互联网上的ss ssr vmess trojan节点信息 聚合去重后提供节点列表轻量级 go 业务框架 哪吒监控 一站式轻监控轻运维系统 支持系统状态 TCP Ping 监控报警 命令批量执行和计划任务 Go 语言官方教程中文版工程师知识管理系统 基于golang go语言 beego框架 每个行业都有自己的知识管理系统 engineercms旨在为土木工程师们打造一款适用的基于web的知识管理系统 它既可以用于管理个人的项目资料 也可以用于管理项目团队资料 它既可以运行于个人电脑 也可以放到服务器上 支持提取码分享文件 onlyoffice实时文档协作 直接在线编辑dwg文件 office文档 在线利用mindoc创作你的书籍 阅览PDF文件 通用的业务流程设置 手机端配套小程序 微信搜索“珠三角设代”或“青少儿书画”即可呼出小程序 边界打点后的自动化渗透工具一个集审核 执行 备份及生成回滚语句于一身的MySQL运维工具汉字转拼音 Go资源精选中文版 含中文图书大全 语言实现的 Redis 服务器和分布式集群 超全golang面试题合集 golang学习指南 golang知识图谱 入门成长路线 一份涵盖大部分golang程序员所需要掌握的核心知识 常用第三方库 mysql mq es redis等 机器学习库 算法库 游戏库 开源框架 自然语言处理nlp库 网络库 视频库 微服务框架 视频教程 音频音乐库 图形图片库 物联网库 地理位置信息 嵌入式脚本库 编译器库 数据库 金融库 电子邮件库 电子书籍 分词 数据结构 设计模式 去html tag标签等 go学习 go面试go语言扩展包 收集一些常用的操作函数 辅助更快的完成开发工作 并减少重复代码百灵快传 基于Go语言的高性能 手机电脑超大文件传输神器 局域网共享文件服务器 LAN large file transfer tool 一个基于云存储的网盘系统 用于自建私人网盘或企业网盘 go分布式服务器 基于内存mmo个人博客微信小程序服务端 SDK for Golang 控制台颜色渲染工具库 支持16色 256色 RGB色彩渲染输出 使用类似于 Print Sprintf 兼容并支持 Windows 环境的色彩渲染基于 IoC 的 Go 后端一站式开发框架 v2ray web manager 是一个v2ray的面板 也是一个集群的解决方案 同时增加了流量控制 账号管理 限速等功能 key admin panel web cluster 集群 proxyServerScan一款使用Golang开发的高并发网络扫描 服务探测工具 是http client领域的瑞士军刀 小巧 强大 犀利 具体用法可看文档 如使用迷惑或者API用得不爽都可提issuesTcpRoute TCP 层的路由器 对于 TCP 连接自动从多个线路 电信 联通 移动 多个域名解析结果中选择最优线路 Bifrost 面向生产环境的 MySQL 同步到Redis MongoDB ClickHouse MySQL等服务的异构中间件应用网关 提供快速 安全的应用交付 身份认证 WAF CC HTTPS以及ACME自动证书 A telegram bot for rss reader 一个支持应用内阅读的 Telegram RSS Bot RESTful API 文档生成工具 支持和 Ruby 等大部分语言 基于gin gorm开发的个人博客项目基于Go语言的国密SM2 SM3 SM4算法库 Golang 设计模式一个阿里云盘列表程序 一款小巧的基于Go构建的开发框架 可以快速构建API服务或者Web网站进行业务开发 遵循SOLID设计原则并发编程实战 第2版 Go 学习 Go 进阶 Go 实用工具类 Go kit Go Micro 微服务实践 Go 推送基于DDD的o2o的业务模型及基础 使用Golang gRPC Thrift实现Sharingan 写轮眼 是一个基于golang的流量录制回放工具 适合项目重构 回归测试等 百度云网盘爬虫基于beego的进销存系统 TeaWeb 可视化的Web代理服务 DEMO teaos cn 白帽子安全开发实战 配套代码抖音推荐 搜索页视频列表视频爬虫方案 基于app 虚拟机或真机 相关技术 golang adb一款甲方资产巡航扫描系统 系统定位是发现资产 进行端口爆破 帮助企业更快发现弱口令问题 主要功能包括 资产探测 端口爆破 定时任务 管理后台识别 报表展示提供微信终端版本 微信命令行版本聊天功能 微信机器人 ️ 互联网最全大厂技术分享PPT 持续更新中 各大技术交流会 活动资料汇总 如 QCon 全球运维技术大会 GDG 全球技术领导力峰会 大前端大会 架构师峰会 敏捷开发DevOps OpenResty Elastic 欢迎 PR Issues日本麻将助手 牌效 防守 记牌 支持雀魂 天凤 开源客服系统GO语言开发GO FLY 免费客服系统一个查询IP地理信息和CDN服务提供商的离线终端工具 是一个用于系统重构 系统迁移和系统分析的瑞士军刀 它可以分析代码中的测试坏味道 模块化分析 行数统计 分析调用与依赖 Git 分析以及自动化重构等 一个直播录制工具Mastering Go 第二版中文版来袭 渗透测试情报收集工具分布式定时任务调度平台高度模块化 遵循 KISS原则的区块链开发框架golang版本的hangout 希望能省些内存 使用了自己写的Kafka lib 虚 不过我们在生产环境已经使用近1年 kafka 版本从 9 1到2 都在使用 目前情况稳定 吞吐量在每天2 亿条以上 Go 语言 Web 应用开发系列教程 从新手到双手残废iris 框架的后台api项目简单好用的DDNS 自动更新域名解析到公网IP 支持阿里云 腾讯云dnspod Cloudflare 华为云 自己动手实现Lua 随书源代码php直播go直播 短视频 直播带货 仿比心 猎游 tt语音聊天 美女约玩 陪玩系统源码开黑 约玩源码 社区开源 云原生的多云和混合云融合平台 Jiajun的编程随想Golang语言社区 腾讯课堂 网易云课堂 字节教育课程PPT及代码基于GF Go Frame 的后台管理系统带你了解一下Golang的市场行情mysql表结构自动同步工具 目前只支持字段 索引的同步 分区等高级功能暂不支持 基于Kubernetes的PaaS平台流媒体NetFlix解锁检测脚本稳定分支2 9 X 版本已更新 由 Golang语言游戏服务器 维护 全球服游戏服务器及区域服框架 目前协议支持websocket KCP TCP及RPC 采用状态同步 帧同步内测 愿景 打造MMO多人竞技游戏框架 功能持续更新中 基于 Golang 类似知乎的私有部署问答应用 包含问答 评论 点赞 管理后台等功能全新的开源漏洞测试框架 实现poc在线编辑 运行 批量测试 使用文档 XAPI MANAGER 专业实用的开源接口管理平台 为程序开发者提供一个灵活 方便 快捷的API管理工具 让API管理变的更加清晰 明朗 如果你觉得xApi对你有用的话 别忘了给我们点个赞哦 qq协议的golang实现 移植于miraigo版本极简工作流引擎全平台Go开源内网渗透扫描器框架 Windows Linux Mac内网渗透 使用它可轻松一键批量探测C段 B段 A段存活主机 高危漏洞检测MS17 1 SmbGhost 远程执行SSH Winrm 密码爆破端口扫描服务识别PortScan指纹识别多网卡主机 端口扫描服务识别PortScan iikira BaiduPCS Go原版基础上集成了分享链接 秒传链接转存功能 e签宝安全团队积累十几年的安全经验 都将对外逐步开放 首开的Ehoney欺骗防御系统 该系统是基于云原生的欺骗防御系统 也是业界唯一开源的对标商业系统的产品 欺骗防御系统通过部署高交互高仿真蜜罐及流量代理转发 再结合自研密签及诱饵 将攻击者攻击引导到蜜罐中达到扰乱引导以及延迟攻击的效果 可以很大程度上保护业务的安全 护网必备良药漂亮的Go语言通用后台管理框架 包含计划任务 MySQL管理 Redis管理 FTP管理 SSH管理 服务器管理 Caddy配置 云存储管理等功能 微信支付 WeChat Pay SDK for Golang用于监控系统的日志采集agent 可无缝对接open falcon阿里巴巴mysql数据库binlog的增量订阅 消费组件 Canal 的 go 客户端 github com alibaba canal 用于比较2个redis数据是否一致 支持单节点 主从 集群版 以及多种proxy 支持同构以及异构对比 redis的版本支持2 x 5 x 使用go micro微服务实现的在线电影院订票系统后端一站式微服务框架 提供API web websocket RPC 任务调度 消息消费服务器红蓝对抗跨平台远控工具Interchain protocol 跨链协议简单易用 足够轻量 性能好的 Golang 库一个go echo vue 开发的快速 简洁 美观 前后端分离的个人博客系统 blog 也可方便二次开发为CMS 内容管理系统 和各种企业门户网站 正在更新权限管理 hauth项目 不是一个前端or后台框架 而是一个集成权限管理 菜单资源管理 域管理 角色管理 用户管理 组织架构管理 操作日志管理等等的快速开发平台． hauth是一个基础产品 在这个基础产品上 根据业务需求 快速的开发应用服务．账号 admin 密码 123456通用的数据验证与过滤库 使用简单 内置大部分常用验证 过滤器 支持自定义验证器 自定义消息 字段翻译 CTF AWD Attack with Defense 线下赛平台 AWD platform 欢迎 Star 蓝鲸智云容器管理平台 BlueKing Container Service 程序员如何优雅的挣零花钱 2 版 升级为小书了 一个 PHP 微信 SDKAV 电影管理系统 avmoo javbus javlibrary 爬虫 线上 AV 影片图书馆 AV 磁力链接数据库ThinkPHP Framework 十年匠心的高性能PHP框架 最全的前端资源汇总仓库 包括前端学习 开发资源 求职面试等 多语言多货币多入口的开源电商 B2C 商城 支持移动端vue app html5 微信小程序微店 微信小程序商城等可能是我用过的最优雅的 Alipay 和 WeChat 的支付 SDK 扩展包了 基于词库的中文转拼音优质解决方案 我用爬虫一天时间“偷了”知乎一百万用户 只为证明PHP是世界上最好的语言 所使用的程序微信 SDK for Laravel 基于 overtrue wechat开源在线教育点播系统 一款满足你的多种发送需求的短信发送组件 基于 Laravel 的后台系统构建工具 Laravel Admin 使用很少的代码快速构建一个功能完善的高颜值后台系统 内置丰富的后台常用组件 开箱即用 让开发者告别冗杂的HTML代码一个想帮你总结所有类型的上传漏洞的靶场优雅的渐进式PHP采集框架 Laravel 电商实战教程的项目代码Payment是php版本的支付聚合第三方sdk 集成了微信支付 支付宝支付 招商一网通支付 提供统一的调用接口 方便快速接入各种支付 查询 退款 转账能力 服务端接入支付功能 方便 快捷 SPF Swoole PHP Framework 世界第一款基于Swoole扩展的PHP框架 开发者是Swoole创始人 A Wonderful WordPress Theme 樱花庄的白猫博客主题图床 此项目已弃用 基于 ThinkPHP 基础开发平台 登录账号密码都是 admin PanDownload网页复刻版一个开源的网址导航网站项目 您可以拿来制作自己的网址导航 使用PHP Swoole实现的网页即时聊天工具 独角数卡 发卡 开源式站长自动化售货解决方案 高效 稳定 快速 卡密商城系统 高效安全的在线卡密商城 ️命令行模式开发框架ShopXO免费开源商城系统 国内领先企业级B2C免费开源电商系统 包含PC h5 微信小程序 支付宝小程序 百度小程序 头条 抖音小程序 QQ小程序 APP 多商户 遵循MIT开源协议发布 基于 ThinkPHP5 1框架研发Wizard是一款开源的文档管理工具 支持Markdown Swagger Table类型的文档 Swoole MySQL Proxy 一个基于 MySQL 协议 Swoole 开发的MySQL数据库连接池 学习资源整合Freenom域名自动续期一个好玩的Web安全 漏洞测试平台一个基于Yii2高级框架的快速开发应用引擎蓝天采集器是一款免费的数据采集发布爬虫软件 采用php mysql开发 可部署在云服务器 几乎能采集所有类型的网页 无缝对接各类CMS建站程序 免登录实时发布数据 全自动无需人工干预 是网页大数据采集软件中完全跨平台的云端爬虫系统免费开源的中文搜索引擎 采用 C C 编写 基于 xapian 和 scws 提供 PHP 的开发接口和丰富文档WDScanner平台目前实现了如下功能 分布式web漏洞扫描 客户管理 漏洞定期扫描 子域名枚举 端口扫描 网站爬虫 暗链检测 坏链检测 网站指纹搜集 专项漏洞检测 代理搜集及部署等功能 ️兰空图床图标工场 移动应用图标生成工具 一键生成所有尺寸的应用图标和启动图 Argon 一个轻盈 简洁的 WordPress 主题Typecho Fans插件作品目录PHP代码审计分段讲解一个结构清晰的 易于维护的 现代的PHP Markdown解析器百度贴吧云签到 在服务器上配置好就无需进行任何操作便可以实现贴吧的全自动签到 配合插件使用还可实现云灌水 点赞 封禁 删帖 审查等功能 注意 Gitee 原Git osc 仓库将不再维护 目前唯一指定的仓库为 Github 本项目没有官方交流群 如需交流可以直接使用Github的Discussions 没有商业版本 目前贴吧云签到由社区共同维护 不会停止更新 PR 通常在一天内处理 微信调试 API调试和AJAX的调试的工具 能将日志通过WebSocket输出到Chrome浏览器的console中 結巴 中文分詞 做最好的 PHP 中文分詞 中文斷詞組件EleTeam开源项目 电商全套解决方案之PHP版 Shop for PHP Yii2 一个类似京东 天猫 淘宝的商城 有对应的APP支持 由EleTeam团队维护 RhaPHP是微信第三方管理平台 微信公众号管理系统 支持多公众号管理 CRM会员管理 小程序开发 APP接口开发 几乎集合微信功能 简洁 快速上手 快速开发微信各种各样应用 简洁 好用 快速 项目开发快几倍 群 656868 一刻社区后端 API 源码 新 微信服务号 微信小程序 微信支付 支付宝支付苹果cms v1 maccms v1 麦克cms 开源cms 内容管理系统 视频分享程序 分集剧情程序 网址导航程序 文章程序 漫画程序 图片程序一个PHP文件搞定支付宝支付系列 包括电脑网站支付 手机网站支付 现金红包 消费红包 扫码支付 JSAPI支付 单笔转账到支付宝账户 交易结算 分账 分润 网页授权获取用户信息等restful api风格接口 APP接口 APP接口权限 oauth2 接口版本管理 接口鉴权基于企业微信的开源SCRM应用开发框架 引擎 也是一套通用的企业私域流量管理系统 API接口大全不断更新中 欢迎Fork和Star 1 一言 古诗句版 api 2 必应每日一图api 3 在线ip查询 4 m3u8视频在线解析api 5 随机生成二次元图片api 6 快递查询api 支持国内百家快递 7 flv视频在线解析api 8 抖音视频无水印解析api 9 一句话随机图片api 1 QQ用户信息获取api 11 哔哩哔哩封面图获取api 12 千图网58pic无水印解析下载api 13 喜马拉雅主播FM数据采集api 14 网易云音乐api 15 CCTV央视网视频解析api 16 微信运动刷步数api 17 皮皮搞笑 基于swoole的定时器程序 支持秒级处理群 656868 ️ Saber PHP异步协程HTTP客户端微信支付单文件版 一个PHP文件搞定微信支付系列 包括原生支付 扫码支付 H5支付 公众号支付 现金红包 企业付款到零钱等 新增V3版 一个还不错的图床工具 支持Mac Win Linux服务器 支持压缩后上传 添加图片或文字水印 多文件同时上传 同时上传到多个云 右击任意文件上传 快捷键上传剪贴板截图 Web版上传 支持作为Mweb Typora发布图片接口 作为PicGo ShareX uPic等的自定义图床 支持在服务器上部署作为图床接口 支持上传任意格式文件 可能是我用过的最优雅的 Alipay 和 WeChat 的 laravel 支付扩展包了上传大文件的Laravel扩展包开发内功修炼Laravel核心代码学习南京邮电大学开源 Online Judge QQ群 6681 8264 免费IP地址数据库 已支持IPV4 IPV6 结构化输出为国家 省 市 县 运营商 中文数据库 方便实用 laravel5 5和vue js结合的前后端分离项目模板 后端使用了laravel的LTS版本 5 5 前端使用了流行的vue element template项目 作为程序的起点 可以直接以此为基础来进行业务扩展 模板内容包括基础的用户管理和权限管理 日志管理 集成第三方登录 整合laravel echo server 实现了websocket 做到了消息的实时推送 并在此基础上 实现了聊天室和客服功能 权限管理包括后端Token认证和前端vue js的动态权限 解决了前后端完整分离的情况下 vue js的认证与权限相关的痛点 已在本人的多个项目中集成使用 Web安全之机器学习入门 网易云音乐升级APIPHP 集成支付 SDK 集成了支付宝 微信支付的支付接口和其它相关接口的操作 支持 php fpm 和 Swoole 所有框架通用 宇润PHP全家桶技术支持群 17916227MDClub 社区系统后端代码imi 是基于 Swoole 的 PHP 协程开发框架 它支持 Http2 WebSocket TCP UDP MQTT 等主流协议的服务开发 特别适合互联网微服务 即时通讯聊天im 物联网等场景 QQ群 17916227WordPress 版 WebStack 导航主题 nav iowen cnLive2D 看板娘插件 www fghrsh net post 123 html 上使用的后端 API简单搜索 一个简单的前端界面 用惯了各种导航首页 满屏幕尽是各种不厌其烦的广告和资讯 尝试自己写个自己的主页 国内各大CTF赛题及writeup整理收集自网络各处的 webshell 样本 用于测试 webshell 扫描器检测率 PHP微信SDK 微信平台 微信支付 码小六 GitHub 代码泄露监控系统PHP表单生成器 快速生成现代化的form表单 支持前后端分离 内置复选框 单选框 输入框 下拉选择框 省市区三级联动 时间选择 日期选择 颜色选择 文件 图片上传等17种常用组件 悟空CRM 基于TP5 vue ElementUI的前后端分离CRM系统V免签PHP版 完全开源免费的个人免签约解决方案Composer 全量镜像发布于2 17年3月 曾不间断运行2年多 这个开源有助于理解 Composer 镜像的工作原理一个多彩 轻松上手 体验完善 具有强大自定义功能的WordPress主题 基于Sakura主题全球免费代理IP库 高可用IP 精心筛选优质IP 2s必达LaraCMS 是在学习 laravel web 开发实战进阶 实战构架 API 服务器 过程中产生的一个业余作品 试图通过简单的方式 快速构建一套基本的企业站同时保留很灵活的扩展能力和优雅的代码方式 当然这些都得益Laravel的优秀设计 同时LaraCMS 也是一个学习Laravel 不错的参考示例 已停止维护 HookPHP基于C扩展搭建内置AI编程的架构系统 支持微服务部署 热插拔业务组件 集成业务模型 权限模型 UI组件库 多模板 多平台 多域名 多终端 多语言 含常驻内存 前后分离 API平台 LUA QQ群 67911638 中华人民共和国居民身份证 中华人民共和国港澳居民居住证以及中华人民共和国台湾居民居住证号码验证工具 PHP 版 最简单的91porn爬虫php版本Fend 是一款短小精悍 可在 FPM Swoole 服务容器平滑切换的高性能PHP框架 no evil 实现过滤敏感词汇 基于确定有穷自动机 DFA 算法 支持composer安装扩展Z BlogPHP博客程序IYUU自动辅种工具 目前能对国内大部分的PT站点自动辅种 支持下载器集群 支持多盘位 支持多下载目录 支持远程连接等 果酱小店 基于 Laravel swoole 小程序的开源电商系统 优雅与性能兼顾 這是一份純靠北工程師的專案 請好好愛護它 謝謝 EC ecjia 到家是一款可开展O2O业务的移动电商系统 它包含 移动端APP 采用原生模式开发 覆盖使用iOS 及Android系统的移 动终端 后台系统 针对平台日常运营维护的平台后台 针对入驻店铺管理的商家后台 独立并行 移动端H5 能够灵活部署于微信及其他APP 网页等 Material Design 指南的中文翻译 一个纯php分词 thinkphp5 1 layui 实现的带rbac的基础管理后台 方便快速开发法使用百度pcs上传脚本目前最全的前端开发面试题及答案樱花内网穿透网站源代码 2 2 重制版MeepoPS是Meepo PHP Socket的缩写 旨在提供稳定的Socket服务 可以轻松构建在线实时聊天 即时游戏 视频流媒体播放等 基础目录 聚合所有其他目录 包含文档和例子基于 Vue js 的简洁一般强大的 WordPress 单栏博客主题阿里云打造Laravel最好的OSS Storage扩展 网上在线商城 综合网上购物平台swoolefy是一个基于swoole实现的轻量级 高性能 协程级 开放性的API应用服务框架基于redis实现高可用 易拓展 接入方便 生产环境稳定运行的延迟队列 一款基于WordPress开发的高颜值的自适应主题 支持白天与黑夜模式 无刷新加载等 阿里云 OSS 官方 SDK 的 Composer 封装 支持任何 PHP 项目 包括 Laravel Symfony TinyLara 等等 此插件将你的WordPress接入本土生态体系之中 使之更适合国内应用环境PHP的服务化框架 适用于Api Server Rpc Server 帮助原生PHP项目转向微服务化 出色的性能与支持高并发的协程相结合基于ThinkPHP V6 开发的面向API的后台管理系统 PHP Swoole 开发的在线同步点歌台 支持自由点歌 切歌 调整排序 删除指定音乐以及基础权限分级信呼 免费开源的办公OA系统 包括APP pc上客户端 REIM即时通信 服务端等 让每个企业单位都有自己的办公系统 来客电商 微信小程序商城 APP商城 公众号商城 PC商城系统 支付宝小程序商城 抖音小程序商城 百度小程序电商系统 前后端代码全部开源 注重界面美感与用户体验 打造独特电商系统生态圈哔哩哔哩 Bilibili B 站主站助手 直播助手 直播抽奖 挂机升级 贴心小棉袄脚本 Lv6 离你仅有一步之遥 PHP 版 Personal 一个运用php与swoole实现的统计监控系统短视频去水印 抖音 皮皮虾 火山 微视 微博 绿洲 最右 轻视频 快手 全民小视频 巴塞电影 陌陌 Before避风 开眼 Vue Vlog 小咖秀 皮皮搞笑 全民K歌 西瓜视频 中国农历 阴历 与阳历 公历 转换与查询工具AoiAWD 专为比赛设计 便携性好 低权限运行的EDR系统 项目管理系统后端接口ThinkPHP 队列支持Typecho Theme Aria 书写自己的篇章PHP 中文工具包 支持汉字转拼音 拼音分词 简繁互转 数字 金额大写 QQ群 17916227Yii2 community 请访问淘客5合一SDK 支持淘宝联盟 京东联盟 多多进宝 唯品会 苏宁基于 thinkphp 开发的的 blogMojito Admin 基于 Laravel Vue Element 构建的后台管理系统一个经典的XSS渗透管理平台一款基于 RageFrame2 的免费开源的基础销售功能的商城基于Laravel 5 4 的开发的博客系统 代号 myPersimmon证件照片排版在线生成器 在一张6寸的照片上排版多张证件照清华大学计算机学科推荐学术会议和期刊列表WordPress响应式免费主题 Art Blog唯品秀博客 weipxiu com 备用域名weipxiu cn 开源给小伙伴免费使用 如使用过程有任何问题 在线技术支持QQ 欢迎打扰 原创不易 如喜欢 请多多打赏 演示 EwoMail是基于Linux的企业邮箱服务器 集成了众多优秀稳定的组件 是一个快速部署 简单高效 多语言 安全稳定的邮件解决方案 笔记本新版简单强大的无数据库的图床2 版 演示地址 Bilibili B 站自动领瓜子 直播助手 直播挂机脚本 主站助手 PHP 版微信群二维码活码工具 生成微信群活码 随时可以切换二维码 短视频的PHP拓展包 集成各大短视频的去水印功能 抖音 快手 微视主流短视频 PHP去水印一个PHPer的升级之路酷瓜云课堂 在线教育 网课系统 网校系统 知识付费系统 不加密不阉割 1 %全功能开源 可免费商用 框架主要使用ThinkPHP6 layui 拥有完善的权限的管理模块以及敏捷的开发方式 让你开发起来更加的舒服 laravel5 5搭建的后台管理 和 api服务 的小程序商城基于ThinkPHP5 AdminLTE的后台管理系统魔改版本 为 OLAINDEX 添加多网盘挂载及一些小修复海豚PHP 基于ThinkPHP5 1 41LTS的快速开发框架挂载Teambition文件 可直链分享 支持网盘 需申请 和项目文件 无需邀请码 准确率99 9%的ip地址定位库laravel ant design vue 权限后台PHP 第三方登录授权 SDK 集成了QQ 微信 微博 Github等常用接口 支持 php fpm 和 Swoole 所有框架通用 QQ群 17916227抖音去水印PHP版接口一个分布式统计监控系统 包含PHP客户端 服务端整合多接口的IP查询工具 基于阿里云OSS的WordPress远程附件支持插件 后会有期 开箱即用的Laravel后台扩展 前后端分离 后端控制前端组件 无需编写vue即可创建一个的项目 丰富的表单 表格组件 强大的自定义组件功能 yii2 swoole 让yii2运行在swoole上胖鼠采集 WordPress优秀开源采集插件CatchAdmin是一款基于thinkphp6 和 element admin 开发的后台管理系统 基于 ServiceProvider 系统模块完全接耦 随时卸载安装模块 提供了完整的权限和数据权限等功能 大量内置的开发工具提升你的开发体验 官网地址 微信公众平台php版开发包微信小程序 校园小情书后台源码 好玩的表白墙 告白墙 功能全面的PHP命令行应用库 提供控制台参数解析 命令运行 颜色风格输出 用户信息交互 特殊格式信息显示基于 chinese poetry 数据整理的一份 mysql 格式数据帮助 thinkphp 5 开发者快速 轻松的构建Api hyperf admin 是基于 hyperf vue 的配置化后台开发工具 微信支付php 写的视频下载工具 现已支持 Youku Miaopai 腾讯 XVideos Pornhub 91porn 微博酷燃 bilibili 今日头条 芒果TVCorePress 主题 一款高性能 高颜值的WordPress主题快链电商 直播电商 分销商城 微信小程序商城 APP商城 公众号商城 PC商城系统 支付宝小程序商城 抖音小程序商城 百度小程序电商系统 前后端代码全部开源 Laravel vue开发 成熟商用项目 shop mall 商城 电商 利用 PHP cURL 转发 Disqus API 请求可能是最优雅 简易的淘宝客SDKUniAdmin是一套渐进式模块化开源后台 采用前后端分离技术 数据交互采用json格式 功能低耦合高内聚 核心模块支持系统设置 权限管理 用户管理 菜单管理 API管理等功能 后期上线模块商城将打造类似composer npm的开放式插件市场 同时我们将打造一套兼容性的API标准 从ThinkPHP5 1 Vue2开始 逐步吸引爱好者共同加入 以覆盖等多语言框架 PHP 多接口获取快递物流信息包LightCMS 是一个基于 Laravel 开发的轻量级 CMS 系统 也可以作为一个通用的后台管理框架使用单点登录系统快乐二级域名分发系统Typecho Theme Story 爱上你我的故事 一个轻量化的留言板 记事本 社交系统 博客 人类的本质是 咕咕咕？微信域名拦截检测 QQ域名拦截检测 t xzkxb com 查询有缓存 如需实时查询请自行部署 高性能分布式并发锁 行为限流Emlog是一款基于PHP和MySQL的功能强大的博客及CMS建站系统 追求快速 稳定 简单 舒适的建站体验Hyperf admin 基于Hyperf Element UI 通用管理后台企业仓库管理系统HisiPHP V2版是基于ThinkPHP5 1和Layui开发的后台框架 承诺永久免费开源 您可用于学习和商用 但须保留版权信息正常显示 如果HisiPHP对您有帮助 您可以点击右上角 Star 支持一下哦 谢谢 使用PHP开发的简约导航 书签管理系统 软擎是基于 Php 7 2 和 Swoole 4 4 的高性能 简单易用的开发框架 支持同时在 Swoole Server 和 php fpm 两种模式下运行 内置了服务 集成了大量成熟的组件 可以用于构建高性能的Web系统 API 中间件 基础服务等等 个人发卡源码 发卡系统 二次元发卡系统 二次元发卡源码 发卡程序 动漫发卡 PHP发卡源码聊天应用 php实现的dht爬虫搭建的webim客服系统 即时通讯一些实用的python脚本同城拼车微信小程序后端代码 一个响应式干净和简洁优雅的 Typecho 主题php仓库进销存深度学习5 问 以问答形式对常用的概率知识 线性代数 机器学习 深度学习 计算机视觉等热点问题进行阐述 以帮助自己及有需要的读者 全书分为18个章节 5 余万字 由于水平有限 书中不妥之处恳请广大读者批评指正 未完待续 如有意合作 联系scutjy2 15 163 com 版权所有 违权必究 Tan 2 18 6题解 记录自己的leetcode解题之路 最全中华古诗词数据库 唐宋两朝近一万四千古诗人 接近5 5万首唐诗加26万宋诗 两宋时期1564位词人 21 5 首词 uni app 是使用 Vue 语法开发小程序 H5 App的统一框架采用自身模块规范编写的前端 UI 框架 遵循原生 HTML CSS JS 的书写形式 极低门槛 拿来即用 我是依扬 木易杨 公众号 高级前端进阶 作者 每天搞定一道前端大厂面试题 祝大家天天进步 一年后会看到不一样的自己 YApi 是一个可本地部署的 打通前后端及QA的 可视化的接口管理平台小程序组件化开发框架网易云音乐 Node js API service基于 Vue js 的小程序开发框架 从底层支持 Vue js 语法和构建工具体系 ECMAScript 6入门 是一本开源的 JavaScript 语言教程 全面介绍 ECMAScript 6 新增的语法特性 谷粒 Chrome插件英雄榜 为优秀的Chrome插件写一本中文说明书 让Chrome插件英雄们造福人类公众号 加1 同步更新前端面试每日 3 1 以面试题来驱动学习 提倡每日学习与思考 每天进步一点 每天早上5点纯手工发布面试题 死磕自己 愉悦大家 4 道前端面试题全面覆盖小程序 软技能 本文原文由知名 Hacker Eric S Raymond 所撰寫 教你如何正確的提出技術問題並獲得你滿意的答案 千古前端图文教程 超详细的前端入门到进阶学习笔记 从零开始学前端 做一名精致优雅的前端工程师 公众号 千古壹号 作者 book Node js 包教不包会 by alsotang收集所有区块链 BlockChain 技术开发相关资料 包括Fabric和Ethereum开发资料轻量 可靠的小程序 UI 组件库微信小程序商城 微信小程序微店一个可以观看国内主流视频平台所有视频的客户端可伸缩布局方案基于 node js Mongodb 构建的后台系统 js 源码解析磁力链接聚合搜索中华人民共和国行政区划 省级 省份直辖市自治区 地级 城市 县级 区县 乡级 乡镇街道 村级 村委会居委会 中国省市区镇村二级三级四级五级联动地址数据 Web接口管理工具 开源免费 接口自动化 MOCK数据自动生成 自动化测试 企业级管理 阿里妈妈MUX团队出品 阿里巴巴都在用 1 公司的选择 RAP2已发布请移步至github com thx rap2 delosKuboard 是基于 Kubernetes 的微服务管理界面 同时提供 Kubernetes 免费中文教程 入门教程 最新版本的 Kubernetes v1 2 安装手册 k8s install 在线答疑 持续更新 ApacheCN 数据结构与算法译文集 chick 是使用 Node js 和 MongoDB 开发的社区系统一个非常适合IT团队的在线API文档 技术文档工具 Chinese sticker pack More joy 表情包的博物馆 Github最有毒的仓库 中国表情包大集合 聚欢乐 高颜值的第三方网易云播放器 支持 Windows macOS Linux vue2 vue router vuex 入门项目网易云音乐第三方 Flutter实战 电子书 一套代码运行多端 一端所见即多端所见 计算机速成课 Crash Course 字幕组 全4 集 2 18 5 1 精校完成 一个 react redux 的完整项目 和 个人总结中文独立博客列表CSS Inspiration 在这里找到写 CSS 的灵感 rich text 富文本编辑器 汉字拼音 hàn zì pīn yīn Chrome插件开发全攻略 配套完整Demo 欢迎clone体验微信调试 各种WebView样式调试 手机浏览器的页面真机调试 便捷的远程调试手机页面 抓包工具 支持 HTTPS 无需USB连接设备 master分支 渲染器 微信小程序组件 API 云开发示例简悦 SimpRead 让你瞬间进入沉浸式阅读的扩展让H5制作像搭积木一样简单 轻松搭建H5页面 H5网站 PC端网站 LowCode平台 一套组件化 可复用 易扩展的微信小程序 UI 组件库这是一个数据可视化项目 能够将历史数据排名转化为动态柱状图图表微信小程序图表charts组件 Charts for WeChat small app类似易企秀的H5制作 建站工具 可视化搭建系统 一个在你编程时疯狂称赞你的 VSCode 扩展插件全家桶后台管理框架解锁网易云音乐客户端变灰歌曲美观易用的React富文本编辑器 基于draft js开发一个致力于微信小程序和 Web 端同构的解决方案從零開始學 ReactJS ReactJS 1 1 是一本希望讓初學者一看就懂的 React 中文入門教學書 由淺入深學習 ReactJS 生態系源码解读 系列文章 完 我就是来分享脚本玩玩的开发者边车 github打不开 github加速 git clone加速 git release下载加速 stackoverflow加速vue源码逐行注释分析 4 多m的vue源码程序流程图思维导图 diff部分待后续更新 微信小程序解决方案 1KB javascript 覆盖状态管理 跨页通讯 插件开发和云数据库开发给老司机用的一个番号推荐系统 FeHelper Web前端助手记录成长的过程哔哩哔哩 bilibili com 辅助工具 可以替换播放器 推送通知并进行一些快捷操作提供了百度坐标 BD 9 国测局坐标 火星坐标 GCJ 2 和WGS84坐标系之间的转换F2etest是一个面向前端 测试 产品等岗位的多浏览器兼容性测试整体解决方案 ️ 阿里飞猪 很易用的中后台 表单 表格 图表 解决方案CRMEB Min 前后端分离版自带客服系统 是CRMEB品牌全新推出的一款轻量级 高性能 前后端分离的开源电商系统 完善的后台权限管理 会员管理 订单管理 产品管理 客服管理 CMS管理 多端管理 页面DIY 数据统计 系统配置 组合数据管理 日志管理 数据库管理 一键开通短信 产品采集 物流查询等接口 React技术揭秘 一本自顶向下的React源码分析书微信小程序 基于wepy 商城 微店 微信小程序 欢迎学习交流大屏数据可视化Pytorch 中文文档经典的网页对话框组件 强大的动态表单生成器 简洁 易用 灵活的微信小程序组件库 一款 Material Design 风格的 Hexo 主题签到一个帮助你自动申请京东价格保护的chrome拓展后台admin前端模板 基于 layui 编写的最简洁 易用的后台框架模板 只需提供一个接口就直接初始化整个框架 无需复杂操作 小程序生成图片库 轻松通过 json 方式绘制一张可以发到朋友圈的图片安卓应用层抓包通杀脚本一个轻量的工具集合婚礼大屏互动 微信请柬一站式解决方案mili 是一个开源的社区系统 界面优雅 功能丰富 丝般顺滑的触摸运动方案做最好的接口管理平台Mpx 一款具有优秀开发体验和深度性能优化的增强型跨端小程序框架省市区县乡镇三级或四级城市数据 带拼音标注 坐标 行政区域边界范围 2 21年 7月 3日最新采集 提供csv格式文件 支持在线转成多级联动js代码 通用json格式 提供软件转成shp geojson sql 导入数据库 带浏览器里面运行的js采集源码 综合了中华人民共和国民政部 国家统计局 高德地图 腾讯地图行政区划数据在vscode中用于生成文件头部注释和函数注释的插件 经过多版迭代后 插件 支持所有主流语言 功能强大 灵活方便 文档齐全 食用简单 觉得插件不错的话 点击右上角给个Star ️呀 JAVClub 让你的大姐姐不再走丢️你想要的最全 Android 进阶路线知识图谱 干货资料收集 开发者推荐阅读的书籍 2 2 淘宝 京东 支付宝双十一 双11全民养猫 全民营业自动化脚本 全额奖励 防检测 一款高性能敏感词 非法词 脏字 检测过滤组件 附带繁体简体互换 支持全角半角互换 汉字转拼音 模糊搜索等功能 前端博客 关注基础知识和性能优化 vue cli4配置vue config js持续更新PT 助手 Plus 为 Google Chrome 和 Firefox 浏览器插件 Web Extensions 主要用于辅助下载 PT 站的种子 基于vue2 koa2的 H5制作工具 让不会写代码的人也能轻松快速上手制作H5页面 类似易企秀 百度H5等H5制作 建站工具最全最新中国省 市 地区json及sql数据首个 Taro 多端统一实例 网易严选 小程序 H5 React Native By 趣店 FED地理信息可视化库企业级 Node js 应用性能监控与线上故障定位解决方案 Node js区块链开发 注 新版代码已开源 请star支持哦 基于Auto js的蚂蚁森林能量自动收取脚本 结巴 中文分词的Node js版本 译 面向机器学习的特征工程webfunny是一款轻量级的前端监控系统 webfunny也是一款前端性能监控系统 无埋点监控前端日志 实时分析前端健康状态一个实现汉字与拼音互转的小巧web工具库 演示地址 前端进阶 优质博文 一个 Chrome 插件 将 Google CDN 替换为国内的 Vue ElementUI构建的CMS开发框架超完整的React Native项目 功能丰富 适合学习和日常使用 GSYGithubApp系列的优势 我们目前已经拥有四个版本 功能齐全 项目框架内技术涉及面广 完成度高 配套文章 适合全面学习 对比参考 开源Github客户端App 更好的体验 更丰富的功能 旨在更好的日常管理和维护个人Github 提供更好更方便的驾车体验Σ 同款Weex版本同款Flutter版本 https github com CarGu 群 宇宙最强的前端面试指南 lucifer ren fe interview 微慕小程序开源版 WordPress版微信小程序函数式编程指北中文版Node js面试题 侧重后端应用与对Node核心的理解jQuery源码解析小白入坑vue三部曲 关于 vue 在工作的使用问题总结 请看博客 sunseekers github io 一直保持更新基于 Vue 的 PWA 解决方案 帮助开发者快速搭建 PWA 应用 解决接入 PWA 的各种问题ThinkCMF是一款支持Swoole的开源内容管理框架 基于ThinkPHP开发 同时支持PHP FPM和Swoole双模式 让WEB开发更快 微信小程序图片裁剪工具前端知识月刊Next Terminal是一个轻量级堡垒机系统 易安装 易使用 支持RDP SSH VNC Telnet Kubernetes协议 微信小程序 日历组件 基于 ueditor的更现代化的富文本编辑器 支持HTTPS基于JavaScript React Vue2的流程图组件 采用Spring MyBatis Shiro框架 开发的一套权限系统 极低门槛 拿来即用 设计之初 就非常注重安全性 为企业系统保驾护航 让一切都变得如此简单 QQ群 32478 2 4 145799952 一个前端的博客 春松客服 多渠道智能客服系统 开源客服系统 机器人客服一个工作流平台小程序 小游戏以及 Web 通用 Canvas 渲染引擎 在线工具秘籍 为在线工具写一本优质说明书 让在线工具造福人类 前端内参 有关于JavaScript 编程范式 设计模式 软件开发的艺术等大前端范畴内的知识分享 旨在帮助前端工程师们夯实技术基础以通过一线互联网企业技术面试 ️ vCards 中国黄页 优化 iOS Android 来电 信息界面体验各平台的分流规则 复写规则及自动化脚本 基于vue2 的实时聊天项目 图片剪裁上传组件 等笔记快速分享 GoogleDrive OneDrive 每日时报 以前端技术体系为主要分享课题 根据 文章 工具 新闻 视频几大板块作为主要分类 一款高效 高性能的帧动画生成工具 停止维护 一个在线音乐播放器 仅 UI 无功能 小程序富文本组件 支持渲染和编辑 html 支持在微信 QQ 百度 支付宝 头条和 uni app 平台使用基于 electron vue 开发的音乐播放器 界面模仿QQ音乐 技术栈欢迎starweui 是在weui和zepto基础上开发的增强UI组件 目前分为表单 基础 组件 js插件四大类 共计百余项功能 是最全的weui样式同步和更新大佬脚本库 更新懒人配置腾讯云即时通信 IM 服务 国内下载镜像 基于 Vue 的小程序开发框架React 16 8打造精美音乐WebAppWK系列开发框架 V1至V5 Java开源企业级开发框架 单应用 微服务 分布式 ️中国 省市区 三级联动 地址选择器 微信小程序2d动画库 分布式 Redis缓存 Shiro权限管理 Spring Session单点登录 Quartz分布式集群调度 Restful服务 QQ 微信登录 App token登录 微信 支付宝支付 日期转换 数据类型转换 序列化 汉字转拼音 身份证号码验证 数字转人民币 发送短信 发送邮件 加密解密 图片处理 excel导入导出 FTP SFTP fastDFS上传下载 二维码 XML读写 高精度计算 系统配置工具类等等 EduSoho 网络课堂是由杭州阔知网络科技有限公司研发的开源网校系统 EduSoho 包含了在线教学 招生和管理等完整功能 让教育机构可以零门槛建立网校 成功转型在线教育 EduSoho 也可作为企业内训平台 帮助企业实现人才培养 自用的一些乱七八糟 油猴脚本 为刚刚学习php语言以及web网站开发整理的一套资源 有视频 实战代码 学习路径等 会持续更新 This is a goindex theme 一个goindex的扩展主题 NumPy官方中文文档 完整版 搭建移动端开发 基于适配方案 axios封装 构建手机端模板脚手架 后台管理 脚手架接口 从简单开始 PhalApi简称π框架 一个轻量级PHP开源接口框架 专注于接口服务开发 前端特效存档Chrome浏览器 抢购 秒杀插件 秒杀助手 定时自动点击云存储管理客户端 支持七牛云 腾讯云 青云 阿里云 又拍云 亚马逊S3 京东云 仿文件夹管理 图片预览 拖拽上传 文件夹上传 同步 批量导出URL等功能font carrier是一个功能强大的字体操作库 使用它你可以随心所欲的操作字体 让你可以在svg的维度改造字体的展现形状 CRN是Ctrip React Native简称 由携程无线平台研发团队基于React Native框架优化 定制成稳定性和性能更佳 也更适合业务场景的跨平台开发框架 油猴脚本页面浮窗广告完全过滤净化 国服最强最全最新CSDN脚本微信小程序即时通讯模板 使用WebSocket通信小程序反编译 支持分包 “想学吗”个人知识管理与自媒体营销工具 超多经典 Canvas 实例 动态离子背景 炫彩小球 贪吃蛇 坦克大战 是男人就下1 层 心形文字等 Vue UEditor v model双向绑定 HQChart H5 微信小程序 沪深 港股 数字货币 期货 美股 K线图 kline 走势图 缩放 拖拽 十字光标 画图工具 截图 筹码图 分析家语法 通达信语法 麦语法 第3方数据替换接口基于koa2的标准前后端分离框架 一款企业信息化开发基础平台 拟集成OA 办公自动化 CMS 内容管理系统 等企业系统的通用业务功能 JeePlatform项目是一款以SpringBoot为核心框架 集ORM框架Mybatis Web层框架SpringMVC和多种开源组件框架而成的一款通用基础平台 代码已经捐赠给开源中国社区基于inception的自动化SQL操作平台 支持SQL执行 LDAP认证 发邮件 OSC SQL查询 SQL优化建议 权限管理等功能 支持docker镜像是一款专门面向个人 团队和小型组织的私有网盘系统 轻量 开源 完善 无论是在家庭 学校还是在办公室 您都能立刻开始使用它 了解更多请访问官方网站 Node js API 中文文档dubbo服务管理以及监控系统拯救B站的弹幕体验 对抗假消息系列项目之一 截屏 实锤？相信你就输了 ”突破性“更新 支持修改任何网站 ️一个简洁 优雅且高效的 Hugo 主题Vue js 示例项目 简易留言板 本项目拥有完善的文档说明与注释 让您快速上手 Vue js 开发? Vue Validator? Vuex?最佳实践基于 Node js Koa2 实战开发的一套完整的博客项目网站 用 React 编写的基于Taro Dva构建的适配不同端 微信 百度 支付宝小程序 H5 React Native 等 的时装衣橱信息泄漏监控系统 伪装115浏览器干爆前端 一网打尽前端面试 学习路径 优秀好文等各类内容 帮助大家一年内拿到期望的 offer 前端性能监控系统 消息队列 高可用 集群等相关架构SpringBoot v2项目是努力打造springboot框架的极致细腻的脚手架 包括一套漂亮的前台 无其他杂七杂八的功能 原生纯净 中文文本标注工具 最全最新中国 省 市 区县 乡镇街道 json csv sql数据 一款轻巧的渐进式微信小程序框架 全网 1 w 阅读量的进阶前端技术博客仓库 Vue 源码解析 React 深度实践 TypeScript 进阶艺术 工程化 性能优化实践 完整开源 Java快速开发平台 基于Spring SpringMVC Mybatis架构 MStore提供更多好用的插件与模板 文章 商城 微信 论坛 会员 评论 支付 积分 工作流 任务调度等 同时提供上百套免费模板任意选择 价值源自分享 铭飞系统不仅一套简单好用的开源系统 更是一整套优质的开源生态内容体系 铭飞的使命就是降低开发成本提高开发效率 提供全方位的企业级开发解决方案 每月28定期更新版本WeHalo 简约风 的微信小程序版博客 基于 vue2 vuex 构建一个具有 45 个页面的大型单页面应用基于Vue3 Element Plus 的后台管理系统解决方案基于 vue element ui 的后台管理系统鲜亮的高饱和色彩 专注视觉的小程序组件库 ️ 跨平台桌面端视频资源播放器 简洁无广告 免费高颜值 后台管理主线版本基于三者并行开发维护 同时支持电脑 手机 平板 切换分支查看不同的vue版本 element plus版本已发布 vue3 vue3 vue vue3 x vue js 程序无国界 但程序员有国界 中国国家尊严不容挑衅 如果您在特殊时期 mall admin web是一个电商后台管理系统的前端项目 基于Vue Element实现 主要包括商品管理 订单管理 会员管理 促销管理 运营管理 内容管理 统计报表 财务管理 权限管理 设置等功能 一款完善的安全评估工具 支持常见 web 安全问题扫描和自定义 poc 使用之前务必先阅读文档Vue数据可视化组件库 类似阿里DataV 大屏数据展示 提供SVG的边框及装饰 图表 水位图 飞线图等组件 简单易用 长期更新 React版已发布 UI表单设计及代码生成器基于Vue的可视化表单设计器 让表单开发简单而高效 基于vue的高扩展在线网页制作平台 可自定义组件 可添加脚本 可数据统计 vue后台管理框架 精致的下拉刷新和上拉加载 js框架 支持vue 完美运行于移动端和主流PC浏览器 基于vue2 vuex element ui后台管理系统 Vue js高仿饿了么外卖App课程源码 coding imooc com class 74 html京东风格移动端 Vue2 Vue3 组件库eladmin前端源码 项目基于的前后端分离后台管理系统 权限控制采用 RBAC 菜单动态路由资源采集站在线播放uView UI 是uni app生态最优秀的UI框架 全面的组件和便捷的工具会让您信手拈来 如鱼得水Vue2 全家桶仿 微信App 项目 支持多人在线聊天和机器人聊天前端vue 后端koa 全栈式开发bilibili首页 A magical vue admin 记得star互联网大厂内推及大厂面经整理 并且每天一道面试题推送 每天五分钟 半年大厂中 Vue3 全家桶 Vant 搭建大型单页面商城项目 新蜂商城 Vue3 版本 技术栈为基于Vue开发的XMall商城前台页面 PC端 Vue全家桶 Vant 搭建大型单页面电商项目 ddbuy 7 orange cn前后端分离权限管理系统 精力有限 停止维护 用 Vue js 开发的跨三端应用Prototyping Tool For Vue Devs 适用于Vue的原型工具实战商城 基于Vue2 高仿微信App的单页应用electron跨平台音乐播放器 可搜网易云 QQ音乐 虾米音乐 支持QQ 微博 Github登录 云歌单 支持一键导入音乐平台歌单ThorUI组件库 轻量 简洁的移动端组件库 组件文档地址 thorui cn doc 最近更新时间 2 21 5 28uni app框架演示示例 Electron Vue 仿网易云音乐windows客户端 基于 Vue2 Vue CLI3 的高仿网易云 mac 客户端播放器 PC Online Music PlayerGitHub 泄露监控系统pear 梨子 轻量级的在线项目 任务协作系统 远程办公协作自选基金助手是一款Chrome扩展 用来快速获取关注基金的实时数据 查看自选基金的实时估值情况支持 markdown 渲染的博客前台展示Vue 音乐搜索 播放 Demo 一个基于 vue2 vue3 的 大转盘 九宫格 抽奖插件奖品 文字 图片 颜色 按钮均可配置 支持同步 异步抽奖 概率前 后端可控 自动根据 dpr 调整清晰度适配移动端 基于 Vue 的在线音乐播放器 PC Online music player美团饿了吗外卖红包外卖优惠券 先领红包再下单 外卖红包优惠券 cps分成 别人领红包下单 你拿佣金 讨论如何构建一套可靠的大型分布式系统用 vue 写小程序 基于 mpvue 框架重写 weui 基于Vue框架构建的github数据可视化平台使用GitHub API 搭建一个可动态发布文章的博客可视化拖拽组件库 DEMO基于开源组件 Inception SQLAdvisor SOAR 的SQL审核 SQL优化的Web平台显示当前网站的所有可用Tampermonkey脚本 专注Web与算法无缝滚动component精通以太坊 中文版 网页模拟桌面基于的多模块前后端分离的博客项目网易云音乐 QQ音乐 咪咕音乐 第三方 web端 可播放 vip 下架歌曲 基于权限管理的后台管理系统基于 Node js 的开源个人博客系统 采用 Nuxt Vue TypeScript 技术栈 一款简洁高效的VuePress知识管理 博客 blog 主题基于uni app的ui框架基于Vue Vuex iView的电子商城网站 Vue2 全家桶 Vant 搭建大型单页面商城项目 新蜂商城前后端分离版本 前端Vue项目源码酷狗 ️ 极客猿梦导航 独立开发者的导航站 Vue SpringBoot MyBatis 音乐网站基于 RageFrame2 的一款免费开源的基础商城销售功能的开源微商城 基于vue2 的网易云音乐播放器 api来自于NeteaseCloudMusicApi v2 为最新版本编写的一套后台管理系统全栈开发王者荣耀手机端官网和管理后台基于 Vue3 x TypeScript 的在线演示文稿应用 实现PPT幻灯片的在线编辑 演示 基于vue2 生态的后台管理系统模板开发的后台管理系统 Vchat 从头到脚 撸一个社交聊天系统 vue node mongodb h5编辑器类似maka 易企秀 账号 密码 admin996 公司展示 讨论Vue Spring boot前后端分离项目 wh web wh server的升级版 基于element ui的数据驱动表单组件基于 GitHub API 开发的图床神器 图片外链使用 jsDelivr 进行 CDN 加速 免下载 免安装 打开网站即可直接使用 免费 稳定 高效 ️ Vue初 中级项目 CnodeJS社区重构预览 DEMO 基于vue2全家桶实现的 仿移动端QQ基于Vue Echarts 构建的数据可视化平台 酷炫大屏展示模板和组件库 持续更新各行各业实用模板和炫酷小组件 基于Spring Boot的在线考试系统 预览地址 129 211 88 191 账户分别是admin teacher student 密码是admin123 6pan 6盘小白羊 第二版 vue3 antd typescript on bone and knife 基于Vue 全家桶 2 x 制作的美团外卖APP 本项目是一款基于 Avue 的表单设计器 拖拽式操作让你快速构建一个表单 一刻社区前端源码基于Vue iView Admin开发的XBoot前后端分离开放平台前端 权限可控制至按钮显示 动态路由权限菜单 多语言 简洁美观 前后端分离 ️一个开源的社区程序 临时测试站 https t myrpg cnecharts地图geoJson行政边界数据的实时获取与应用 省市区县多级联动下钻 真正意义的下钻至县级 附最新geoJson文件下载 Vue的Nuxt js服务端渲染框架 NodeJS为后端的全栈项目 Docker一键部署 面向小白的完美博客系统vue瀑布流组件 vue waterfall easy 2 x Ego 移动端购物商城 vue vuex ruoter webpack Vue js Node js Mongodb 前后端分离的个人博客头像加口罩小程序 基于uniapp使用vue快速实现 广告月收入4k 基于vue3 的管理端模板教你如何打造舒适 高效 时尚的前端开发环境基于 Flask 和 Vue js 前后端分离的微型博客项目 支持多用户 Markdown文章 喜欢 收藏文章 粉丝关注 用户评论 点赞 动态通知 站内私信 黑名单 邮件支持 管理后台 权限管理 RQ任务队列 Elasticsearch全文搜索 Linux VPS部署 Docker容器部署等基于 vue 和 heyui 组件库的中后端系统 admin heyui topVue 轻量级后台管理系统基础模板uni app项目插件功能集合We川大小程序 scuplus 使用wepy开发的完善的校园综合小程序 4 页面 前后端开源 包括成绩 课表 失物招领 图书馆 新闻资讯等等常见校园场景功能一个全随机的刷装备小游戏一个vue全家桶入门Demo 是一個可以幫助您 Vue js 的項目測試及偵錯的工具 也同時支持 Vuex及 Vue Router 微信公众号管理系统 包含公众号菜单管理 自动回复 素材管理 模板消息 粉丝管理 ️等功能 前后端都开源免费 基于vue 的管理后台 配合Blog Core与Blog Vue等多个项目使用海风小店 开源商城 微信小程序商城管理后台 后台管理 VUE IT之家第三方小程序版客户端 使用 mpvue 开发 兼容 web vue 可以拖拽排序的树形表格 现代 Web 开发语法基础与工程实践 涵盖 Web 开发基础 前端工程化 应用架构 性能与体验优化 混合开发 React 实践 Vue 实践 WebAssembly 等多方面 数据大屏可视化编辑器一个适用于摄影从业者 爱好者 设计师等创意行业从业者的图像工具箱 武汉大学图书馆助手 桌面端基于form generator 仿钉钉审批流程创建 表单创建 流程节点可视化配置 必填条件及校验 一个完整electron桌面记账程序 技术栈主要使用electron vue vuetify 开机自动启动 自动更新 托盘最小化 闪烁等常用功能 Nsis制作漂亮的安装包 程序猿的婚礼邀请函 一个基于vue和element ui的树形穿梭框及邮件通讯录版本见示例见 基于Gin Vue Element UI的前后端分离权限管理系统的前端模块通用书籍阅读APP BookChat 的 uni app 实现版本 支持多端分发 编译生成Android和iOS 手机APP以及各平台的小程序基于Vue3的Material design风格移动端组件库进阶资深前端开发在线考试系统 springboot vue前后端分离的一个项目 ️ 无后端的仿 YouTube Live Chat 风格的简易 Bilibili 弹幕姬vue后端管理系统界面 基于ui组件iviewBilibili直播弹幕库 for Mac Windows LinuxVue高仿网易云音乐 基本实现网易云所有音乐 MV相关功能 现已更新到第二版 仅用于学习 下面有详细教程 武汉大学图书馆助手 移动端Zeus基于Golang Gin casbin 致力于做企业统一权限 账号中心管理系统 包含账号管理 数据权限 功能权限 应用管理 多数据库适配 可docker 一键运行 社区活跃 版本迭代快 加群免费技术支持 Vue高仿网易云音乐 Vue入门实践 在线预览 暂时停止基于 Vue 和 ElementUI 构建的一个企业级后台管理系统 ️ 跨平台移动端视频资源播放器 简洁免费 ZY Player 移动端 APP 基于 Uni app 开发 Vue实战项目基于参考小米商城 实现的电商项目 h5制作 移动端专题活动页面可视化编辑仿钉钉审批流程设置动态表单页面设计 自动生成页面微前端项目实战vue项目 基于vue3 qiankun2 进阶版 github com wl ui wl mfe基于 d2 admin的RBAC权限管理解决方案VueNode 是一套基于的前后端分离项目 基于仿京东淘宝的 移动端H5电商平台 巨树 基于ztree封装的Vue树形组件 轻松实现海量数据的高性能渲染 微信红包封面领取 用户观看视频广告或者邀请用户可获取微信红包序列号基于 Vue 的可视化布局编辑器插件kbone ui 是一套能同时支持 小程序 kbone 和 vue 框架开发的多端 UI 库 PS 新版 kbone ui 已出炉并迁移到 kbone 主仓库 此仓库仅做旧版维护之用 一个vue的个人博客项目 配合 net core api教程 打造前后端分离Tumo Blog For Vue js 前后端分离bpmn js流程设计器组件 基于vue elementui美化属性面板 满足9 %以上的业务需求专门为 Weex 前端开发者打造的一套高质量UI框架 想用vue把我现在的个人网站重新写一下 新的风格 新的技术 什么都是新的 本项目是一个在线聊天系统 最大程度的还原了Mac客户端QQ vue cli3 后台管理模板 heart 基于vue2和vuex的复杂单页面应用 2 页面53个API 仿实验楼 基于 Vue Koa 的 WebDesktop 视窗系统 Jeebase是一款前后端分离的开源开发框架 基于开发 一套SpringBoot后台 两套前端页面 可以自由选择基于ElementUI或者AntDesign的前端界面 二期会整合react前端框架 Ant Design React 在实际应用中已经使用这套框架开发了CMS网站系统 社区论坛系统 微信小程序 微信服务号等 后面会逐步整理开源 本项目主要目的在于整合主流技术框架 寻找应用最佳项目实践方案 实现可直接使用的快速开发框架 使用 vue cli3 搭建的vue vuex router element 开发模版 集成常用组件 功能模块JEECG BOOT APP 移动解决方案 采用uniapp框架 一份代码多终端适配 同时支持APP 小程序 H5 实现了与JeecgBoot平台完美对接的移动解决方案 目前已经实现登录 用户信息 通讯录 公告 移动首页 九宫格等基础功能 明日方舟工具箱 支持中台美日韩服vue的验证码插件这里有一些标准组件库可能没有的功能组件 已有组件 放大镜 签到 图片标签 滑动验证 倒计时 水印 拖拽 大家来找茬 基于Vue2 Nodejs MySQL的博客 有后台管理系统\",\"html_url\":\"https://github.com/gege-circle/.github\",\"stargazers_count\":1680,\"forks_count\":144,\"open_issues_count\":919,\"updated_at\":\"2025-11-19T12:50:33Z\",\"created_at\":\"2020-09-20T04:47:25Z\",\"topics\":[\"a-soul\",\"acfun\",\"bilibili\",\"china\",\"gege-circle\",\"message-board\",\"vtuber\",\"vup\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":162919890,\"name\":\"awesome-ai-ml-dl\",\"full_name\":\"neomatrix369/awesome-ai-ml-dl\",\"description\":\"Awesome Artificial Intelligence, Machine Learning and Deep Learning as we learn it. Study notes and a curated list of awesome resources of such topics.\",\"html_url\":\"https://github.com/neomatrix369/awesome-ai-ml-dl\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":1608,\"forks_count\":361,\"open_issues_count\":3,\"updated_at\":\"2025-11-19T17:17:26Z\",\"created_at\":\"2018-12-23T19:55:49Z\",\"topics\":[\"ai\",\"algorithms\",\"artificial-intelligence\",\"cloud-devops\",\"data\",\"data-generation\",\"deep-learning\",\"dl\",\"docker\",\"graal\",\"graalvm\",\"intelligent-systems\",\"machine-intelligence\",\"machine-learning\",\"mathematica\",\"ml\",\"natural-language-processing\",\"neural-networks\",\"nlp\",\"time-series\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":376740449,\"name\":\"yt-channels-DS-AI-ML-CS\",\"full_name\":\"benthecoder/yt-channels-DS-AI-ML-CS\",\"description\":\"A comprehensive list of 180+ YouTube Channels for Data Science,  Data Engineering, Machine Learning, Deep learning, Computer Science, programming, software engineering, etc.\",\"html_url\":\"https://github.com/benthecoder/yt-channels-DS-AI-ML-CS\",\"stargazers_count\":1377,\"forks_count\":141,\"open_issues_count\":1,\"updated_at\":\"2025-11-15T01:25:49Z\",\"created_at\":\"2021-06-14T07:31:26Z\",\"topics\":[\"ai\",\"artificial-intelligence\",\"awesome\",\"awesome-list\",\"coding\",\"data\",\"data-analysis\",\"data-engineering\",\"data-science\",\"deep-learning\",\"machine-learning\",\"math\",\"ml\",\"programming\",\"python\",\"resources\",\"software-engineering\",\"statistics\",\"web-development\",\"youtube\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"}]}"}]},{"type":"mcp_tool_result","tool_use_id":"mcptoolu_01AP6ME8z1AwGCuzW4RkrqNL","is_error":false,"content":[{"type":"text","text":"{\"total_count\":367,\"incomplete_results\":false,\"items\":[{\"id\":740303686,\"name\":\"sglang\",\"full_name\":\"sgl-project/sglang\",\"description\":\"SGLang is a fast serving framework for large language models and vision language models.\",\"html_url\":\"https://github.com/sgl-project/sglang\",\"language\":\"Python\",\"stargazers_count\":20226,\"forks_count\":3453,\"open_issues_count\":1453,\"updated_at\":\"2025-11-19T21:20:35Z\",\"created_at\":\"2024-01-08T04:15:52Z\",\"topics\":[\"blackwell\",\"cuda\",\"deepseek\",\"deepseek-r1\",\"deepseek-v3\",\"deepseek-v3-2\",\"gpt-oss\",\"inference\",\"kimi\",\"llama\",\"llama3\",\"llava\",\"llm\",\"llm-serving\",\"moe\",\"openai\",\"pytorch\",\"qwen3\",\"transformer\",\"vlm\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1001935685,\"name\":\"LLMs-from-scratch\",\"full_name\":\"Lamorati92/LLMs-from-scratch\",\"description\":\"📚 Build and train your own GPT-like Large Language Model from scratch with clear guidance and real code examples.\",\"html_url\":\"https://github.com/Lamorati92/LLMs-from-scratch\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":1,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-19T19:46:37Z\",\"created_at\":\"2025-06-14T11:03:23Z\",\"topics\":[\"advanced-neural-network\",\"agents\",\"book\",\"classification\",\"deep-learning\",\"deep-neural-networks\",\"flan-t5\",\"from-scratch\",\"gpt\",\"language-model\",\"llm\",\"llms-book\",\"mcp\",\"nlp\",\"prompt-engineering\",\"roberta\",\"rust\",\"transformer\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":735425203,\"name\":\"transformerlab-app\",\"full_name\":\"transformerlab/transformerlab-app\",\"description\":\"Open Source Application for Advanced LLM + Diffusion Engineering: interact, train, fine-tune, and evaluate large language models on your own computer.\",\"html_url\":\"https://github.com/transformerlab/transformerlab-app\",\"language\":\"Python\",\"stargazers_count\":4533,\"forks_count\":455,\"open_issues_count\":62,\"updated_at\":\"2025-11-19T19:45:52Z\",\"created_at\":\"2023-12-24T22:09:14Z\",\"topics\":[\"diffusion\",\"diffusion-models\",\"electron\",\"llama\",\"llms\",\"lora\",\"mlx\",\"rlhf\",\"stability-diffusion\",\"transformers\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":976708748,\"name\":\"transformerlab-app\",\"full_name\":\"SOMIR420/transformerlab-app\",\"description\":\"Open Source Application for Advanced LLM Engineering: interact, train, fine-tune, and evaluate large language models on your own computer.\",\"html_url\":\"https://github.com/SOMIR420/transformerlab-app\",\"language\":\"TypeScript\",\"stargazers_count\":1,\"forks_count\":2,\"open_issues_count\":1,\"updated_at\":\"2025-11-19T18:41:44Z\",\"created_at\":\"2025-05-02T15:48:08Z\",\"topics\":[\"electron\",\"llama\",\"lora\",\"mlx\",\"rlhf\",\"transformers\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1088106903,\"name\":\"LLM-Evolution-Metadata-Repository\",\"full_name\":\"Harsidak/LLM-Evolution-Metadata-Repository\",\"description\":\"This repositry Showcases the historical timeline of Large Language Models(LLMs) from the transformers model to current state of art models\",\"html_url\":\"https://github.com/Harsidak/LLM-Evolution-Metadata-Repository\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-17T06:44:55Z\",\"created_at\":\"2025-11-02T10:26:28Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1096839708,\"name\":\"gpu-llm-on-wsl\",\"full_name\":\"colinmxs/gpu-llm-on-wsl\",\"description\":\"Dockerized environment for running quantized large language models with GPU acceleration on Windows via WSL2. Includes PyTorch, Hugging Face Transformers, and bitsandbytes for efficient local inference with Llama 2, Mistral, and other 7B/13B models.\",\"html_url\":\"https://github.com/colinmxs/gpu-llm-on-wsl\",\"language\":\"Python\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-17T03:26:18Z\",\"created_at\":\"2025-11-15T03:24:48Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1097796923,\"name\":\"nlp-llm-journey\",\"full_name\":\"Vishakha93/nlp-llm-journey\",\"description\":\"Learning log and code playground for my journey from core NLP concepts (tokenization, BPE, language modeling) to transformer internals and large language model systems (GPU memory, model parallelism, LoRA, deployment).\",\"html_url\":\"https://github.com/Vishakha93/nlp-llm-journey\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-16T21:30:59Z\",\"created_at\":\"2025-11-16T20:50:02Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1096951982,\"name\":\"Generative-AI-Assignment-Using-google_flan_t5_small\",\"full_name\":\"Barrcardavis/Generative-AI-Assignment-Using-google_flan_t5_small\",\"description\":\"The goal of this assignment is to gain practical experience using state-of-the-art Large Language Models (LLMs) via the Hugging Face Transformers library. You will select a model and implement one core generative task, experimenting with key parameters.\",\"html_url\":\"https://github.com/Barrcardavis/Generative-AI-Assignment-Using-google_flan_t5_small\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-16T20:28:57Z\",\"created_at\":\"2025-11-15T08:50:23Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1033686445,\"name\":\"GEN-AI-CHATBOT\",\"full_name\":\"SH4IKVT/GEN-AI-CHATBOT\",\"description\":\"A conversational chatbot built with LangChain, Hugging Face Transformers, and Streamlit, designed to interact with users intelligently using powerful LLMs like flan-t5. This project demonstrates how to integrate large language models into a web interface for natural language understanding, question answering, and educational support.\",\"html_url\":\"https://github.com/SH4IKVT/GEN-AI-CHATBOT\",\"language\":\"Python\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-15T11:56:04Z\",\"created_at\":\"2025-08-07T07:38:12Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1083103295,\"name\":\"Prompt-Injection-Detector\",\"full_name\":\"Kiraneswar/Prompt-Injection-Detector\",\"description\":\"A machine learning model to detect and prevent prompt injection attacks in Large Language Models (LLMs). Uses DistilBERT for binary classification of safe vs. malicious prompts, achieving 99% accuracy in identifying jailbreak attempts and harmful instructions. Built with PyTorch and Hugging Face Transformers.\",\"html_url\":\"https://github.com/Kiraneswar/Prompt-Injection-Detector\",\"language\":\"Python\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-15T07:01:24Z\",\"created_at\":\"2025-10-25T10:56:55Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1078506818,\"name\":\"LLM-from-Scratch\",\"full_name\":\"saeedmohseni97/LLM-from-Scratch\",\"description\":\"From tokenizer to transformer — a hands-on exploration of how large language models learn, implemented and trained entirely from scratch on TinyStories using PyTorch. \",\"html_url\":\"https://github.com/saeedmohseni97/LLM-from-Scratch\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-14T22:25:43Z\",\"created_at\":\"2025-10-17T21:05:26Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1096717910,\"name\":\"LLM-From-Scratch\",\"full_name\":\"Aayanpatel05/LLM-From-Scratch\",\"description\":\"An end-to-end implementation of a GPT-style large language model built from scratch using Python and PyTorch, including tokenizer, attention mechanism, transformer blocks, pretraining, and finetuning.\",\"html_url\":\"https://github.com/Aayanpatel05/LLM-From-Scratch\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-14T21:23:41Z\",\"created_at\":\"2025-11-14T20:59:52Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1086711806,\"name\":\"DocuMind-LLM\",\"full_name\":\"ramarav/DocuMind-LLM\",\"description\":\" 🤖 Chat with your documents using Large Language Models (LLMs), FAISS, and Hugging Face Transformers — all running locally with Streamlit.\",\"html_url\":\"https://github.com/ramarav/DocuMind-LLM\",\"language\":\"Python\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":3,\"updated_at\":\"2025-11-14T13:37:53Z\",\"created_at\":\"2025-10-30T19:46:45Z\",\"topics\":[\"ai\",\"deep-learning\",\"document-ai\",\"faiss\",\"generative-ai\",\"hugging-face\",\"langchain\",\"llm\",\"machine-learning\",\"nlp\",\"pdf\",\"python\",\"qna\",\"rag\",\"streamlit\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1025317990,\"name\":\"valerie.c\",\"full_name\":\"teleprint-me/valerie.c\",\"description\":\"A Large Language Model written completely from scratch in pure C.\",\"html_url\":\"https://github.com/teleprint-me/valerie.c\",\"language\":\"C\",\"stargazers_count\":2,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-13T20:42:21Z\",\"created_at\":\"2025-07-24T04:46:14Z\",\"topics\":[\"c\",\"llm\",\"transformer\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":1088110614,\"name\":\"LLM_memory_constraints\",\"full_name\":\"ShayanShahrabi/LLM_memory_constraints\",\"description\":\"Official repository of \\\"Cognitive Diversity in Artificial Minds - Investigating the Performance of Large Language Models Under Memory Constraints\\\" - HNC2025\",\"html_url\":\"https://github.com/ShayanShahrabi/LLM_memory_constraints\",\"language\":\"Python\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-11-13T10:54:43Z\",\"created_at\":\"2025-11-02T10:36:00Z\",\"topics\":[\"attention-mechanism\",\"llm\",\"neuroscience\",\"transformer\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"}]}"}]},{"type":"mcp_tool_result","tool_use_id":"mcptoolu_014tJigNv1KkKLTDyj6J9Wg5","is_error":false,"content":[{"type":"text","text":"{\"total_count\":12,\"incomplete_results\":false,\"items\":[{\"id\":902059461,\"name\":\"Lang-NAS\",\"full_name\":\"kyegomez/Lang-NAS\",\"description\":\"The Prompt Guided Neural Architecture Search (PG-NAS) is a PyTorch-based framework designed to generate neural network architectures based on textual prompts.\",\"html_url\":\"https://github.com/kyegomez/Lang-NAS\",\"language\":\"Python\",\"stargazers_count\":4,\"forks_count\":0,\"open_issues_count\":1,\"updated_at\":\"2024-12-18T16:35:37Z\",\"created_at\":\"2024-12-11T20:35:19Z\",\"topics\":[\"agoralab\",\"ai\",\"meta\",\"ml\",\"nas\",\"neural-architecture-search\",\"pytorch\",\"research\",\"tensorflow\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":545757424,\"name\":\"Plant-leaf-diseases-detection-\",\"full_name\":\"assalaabnk/Plant-leaf-diseases-detection-\",\"description\":\"The automated identification of plant diseases based on plant leaves is a huge breakthrough. Furthermore, early and accurate detection of plant diseases positively impacts crop productivity and quality. However, managing the accessibility of early plant disease detection is crucial. This work has environmental goals aiming to save plants from different threatening diseases by providing early detection of the affected leaves. We studied the performance of different Convolutional Neural Network (CNN) architectures in predicting 26 diseases for 14 plant species. The work studied the complexity of the system and compared the two main deep learning frameworks, TensorFlow and PyTorch, to get the most accurate results with higher accuracy. Using the \\u0026quot;New PlantVillage Dataset\\u0026quot; from Kaggle [1], the TensorFlow models achieved an accuracy of 90,94% for the basic CCN architecture, and 95,59% for the Transfer Learning architecture with VGG19. Whereas the PyTorch models achieved an accuracy of 93,47% for the basic CCN architecture, and 98,53% for the Transfer Learning architecture with ResNet34. Finally, after examining the feasibility of the model\\u0026#39;s implementation and discussing the main problems that may be encountered, the models were deployed in a mobile application using the Tflite and torch mobile flutter SDK to let them as an internal feature in the mobile without the need for any access to the cloud, which is known as edge AI.\",\"html_url\":\"https://github.com/assalaabnk/Plant-leaf-diseases-detection-\",\"language\":\"Dart\",\"stargazers_count\":3,\"forks_count\":0,\"open_issues_count\":1,\"updated_at\":\"2024-11-23T09:18:18Z\",\"created_at\":\"2022-10-04T23:37:15Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":977690917,\"name\":\"VanillAI\",\"full_name\":\"batuhaneralp/VanillAI\",\"description\":\"VanillAI is a minimalist ML/DL framework built with 100% vanilla Python – no NumPy, no PyTorch, no TensorFlow. This project is for those who want to learn, teach, or deeply understand the mechanics of neural networks and optimization algorithms from the ground up. Requirements? Python.\",\"html_url\":\"https://github.com/batuhaneralp/VanillAI\",\"language\":\"Python\",\"stargazers_count\":1,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-05-06T22:03:11Z\",\"created_at\":\"2025-05-04T19:04:31Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":537196572,\"name\":\"Symbl.ai-Building-Neural-Network-TensorFlow-PyTorch\",\"full_name\":\"contentlab-io/Symbl.ai-Building-Neural-Network-TensorFlow-PyTorch\",\"description\":\"Learn to create and train a neural network for the multi-class classification using TensorFlow and PyTorch. Discover which framework is suitable for beginners.\",\"html_url\":\"https://github.com/contentlab-io/Symbl.ai-Building-Neural-Network-TensorFlow-PyTorch\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2022-09-15T20:49:16Z\",\"created_at\":\"2022-09-15T20:30:36Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":665289313,\"name\":\"FAA-Airman-Qualification-Prediction\",\"full_name\":\"bins0000/FAA-Airman-Qualification-Prediction\",\"description\":\"Employed cutting-edge Machine Learning and Deep Learning algorithms, including SVM, Random Forest, and Neural Networks, to develop a dynamic framework for assessing pilots' qualification using Python and prominent ML libraries such as scikit-learn, TensorFlow, and PyTorch.\",\"html_url\":\"https://github.com/bins0000/FAA-Airman-Qualification-Prediction\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2023-07-12T01:39:29Z\",\"created_at\":\"2023-07-11T21:53:14Z\",\"topics\":[\"big-data\",\"deep-learning\",\"healthcare-data\",\"machine-learning\",\"neural-network\",\"pytorch\",\"sklearn\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":780439248,\"name\":\"Deep-Learning-projects\",\"full_name\":\"dorianDraper/Deep-Learning-projects\",\"description\":\"We embark on a journey through various deep learning projects, delving into diverse models, techniques, methods, and frameworks. From convolutional neural networks (CNNs) to recurrent neural networks (RNNs), from TensorFlow to PyTorch, we uncover the intricacies of cutting-edge AI technologies. \",\"html_url\":\"https://github.com/dorianDraper/Deep-Learning-projects\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2024-06-07T14:18:31Z\",\"created_at\":\"2024-04-01T13:40:47Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":653596113,\"name\":\"dsl\",\"full_name\":\"NeuralFlogo/dsl\",\"description\":\"Flogo is a Domain-Specific Language (DSL) designed to simplify and automate the creation of neural networks. It aims to abstract the technical complexity of frameworks like PyTorch or TensorFlow, allowing developers to model both the structure of neural networks and their training process using a more intuitive and straightforward language.\",\"html_url\":\"https://github.com/NeuralFlogo/dsl\",\"language\":\"Java\",\"stargazers_count\":0,\"forks_count\":1,\"open_issues_count\":0,\"updated_at\":\"2024-06-10T06:56:39Z\",\"created_at\":\"2023-06-14T11:00:31Z\",\"topics\":[\"ai\",\"artificial-intelligence\",\"deep-learning\",\"dsl\",\"intuitive\",\"neural-network\",\"simple\"],\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":926061147,\"name\":\".-AI-Powered-Image-Colorization-Solution\",\"full_name\":\"priyankachaudhari101/.-AI-Powered-Image-Colorization-Solution\",\"description\":\"An AI-powered solution that colorizes grayscale images using deep neural networks, inspired by ECCV 2016 \\u0026 SIGGRAPH 2017. It supports content restoration, e-commerce, and creative workflows, ensuring high-quality, vibrant results. Originally implemented in Caffe, it's scalable and adaptable to modern frameworks like TensorFlow/PyTorch\",\"html_url\":\"https://github.com/priyankachaudhari101/.-AI-Powered-Image-Colorization-Solution\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-02-02T13:10:08Z\",\"created_at\":\"2025-02-02T13:10:04Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":273421168,\"name\":\"camAIra\",\"full_name\":\"widiasamosir/camAIra\",\"description\":\"Application of Deep Learning Network for Enhance Low Resolution into High Resolution and Restoring Dark Images into Bright and Clear Image. Single Image Super-Resolution using ESRGAN as neural network to enhance Low-Res image into High-res Image. Using FCNN as a neural network for restoring dark images. Deep learning using Tensorflow, Pytorch, and ISR module as a framework model. For deploying model to website, using Flask and Flask Ngrok. Namely camAIra as Image Processing using Artificial Intelligence Technology.\",\"html_url\":\"https://github.com/widiasamosir/camAIra\",\"language\":\"Python\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2022-01-29T15:45:39Z\",\"created_at\":\"2020-06-19T06:29:25Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":970694571,\"name\":\"Deep-Learning\",\"full_name\":\"ijayslay/Deep-Learning\",\"description\":\"A collection of Deep Learning practicals implemented and practiced during class sessions. This repository is intended for students and beginners to understand the basics of neural networks, training pipelines, and real-world AI applications using frameworks like TensorFlow and PyTorch\",\"html_url\":\"https://github.com/ijayslay/Deep-Learning\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-08-31T20:13:51Z\",\"created_at\":\"2025-04-22T11:55:10Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"},{\"id\":719171394,\"name\":\"MLProjects\",\"full_name\":\"ChideraBel/MLProjects\",\"description\":\"ML projects worked on while learning new machine learning concepts and AI engineering. Includes linear, multi-linear \\u0026 logistic regression models as well as classification models. Additionally, learning concepts of classification and regression using deep learning neural networks. Frameworks used: PyTorch, Keras, TensorFlow, Pandas, Numpy, e.t.c\",\"html_url\":\"https://github.com/ChideraBel/MLProjects\",\"language\":\"Jupyter Notebook\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2024-01-06T09:59:12Z\",\"created_at\":\"2023-11-15T15:51:41Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"master\"},{\"id\":990878766,\"name\":\"Deep-learning\",\"full_name\":\"MUHAMMADKASHIF3322/Deep-learning\",\"description\":\"Exploring advanced deep learning techniques and optimizing neural network architectures! Passionate about model efficiency, hyperparameter tuning, and leveraging frameworks like TensorFlow and PyTorch for scalable AI solutions. Always eager to refine workflows and push the boundaries of performance. Let me know if you’d like any adjustments! 🚀\",\"html_url\":\"https://github.com/MUHAMMADKASHIF3322/Deep-learning\",\"stargazers_count\":0,\"forks_count\":0,\"open_issues_count\":0,\"updated_at\":\"2025-05-30T18:07:11Z\",\"created_at\":\"2025-05-26T19:30:13Z\",\"private\":false,\"fork\":false,\"archived\":false,\"default_branch\":\"main\"}]}"}]},{"type":"text","text":"Based on my research of AI developments on GitHub, here's a comprehensive overview of current trends and key projects:\n\n## **Top AI Development Trends on GitHub**\n\n### **1. Large Language Models (LLMs) - The Dominant Force**\n\n**Key Projects:**\n- **SGLang** (20.2k stars) - Fast serving framework for LLMs and vision language models\n  - Supports: DeepSeek, GPT, Llama, Llava, and other cutting-edge models\n  - Focus on inference optimization and model deployment\n\n- **LLMs from Scratch** - Educational projects teaching how to build GPT-like models\n  - Covers transformers, attention mechanisms, and tokenization\n  - Built with PyTorch from ground up\n\n### **2. ML/DL Frameworks & Tools**\n\n**PyTorch vs TensorFlow:** Both remain dominant, with PyTorch gaining traction in research:\n- **PyTorch**: Preferred for research and custom architectures\n- **TensorFlow**: Strong in production deployments and edge computing\n\n**Emerging Tools:**\n- SGLang (inference optimization)\n- TransformerLab App (4.5k stars) - Open-source app for LLM engineering with fine-tuning capabilities\n\n### **3. Key AI Domains**\n\n**Neural Architecture Search (NAS):**\n- Projects using prompt-guided architecture generation\n- Automating model design using natural language\n\n**Computer Vision:**\n- Image classification with CNNs\n- Object detection with YOLOv3\n- Image super-resolution and restoration\n- Plant disease detection systems\n\n**Natural Language Processing:**\n- Transformer implementations\n- Token processing and embeddings\n- Chat systems and Q&A models\n\n### **4. Practical Applications**\n\n- **Healthcare AI**: Medical image analysis, diagnostic systems\n- **Agriculture**: Crop disease detection using deep learning\n- **E-commerce**: Product recommendation engines\n- **Document AI**: RAG (Retrieval-Augmented Generation) systems\n- **Edge AI**: Mobile deployment with TensorFlow Lite and PyTorch Mobile\n\n### **5. Learning Resources**\n\nPopular repositories include:\n- **500+ AI/ML Projects** (28.8k stars) - Comprehensive collection with code\n- **Deep Learning Specialization** (4k stars) - Course materials and assignments\n- **Netron** (31.8k stars) - Neural network visualization tool\n- **Cheatsheets for AI** (15.3k stars) - Quick reference guides\n\n### **6. Current Hot Topics**\n\n✅ **Quantization & Efficiency** - Running large models on consumer hardware\n✅ **Fine-tuning & LoRA** - Customizing models for specific tasks  \n✅ **Vision Language Models** - Multimodal AI combining images and text\n✅ **Prompt Engineering** - Optimizing interactions with LLMs\n✅ **Model Deployment** - Making AI accessible in production\n✅ **Safety & Security** - Detecting prompt injection attacks\n\n### **7. Notable Technical Advances**\n\n- **DeepSeek models** - New competitive alternatives to ChatGPT\n- **Qwen-3** and other open-source LLMs\n- **MOE (Mixture of Experts)** architectures\n- **RLHF (Reinforcement Learning from Human Feedback)** refinements\n\n### **Recommendations for Getting Started**\n\n1. **For LLM work**: Start with SGLang or TransformerLab for practical experiments\n2. **For learning**: Use \"LLMs from Scratch\" for theoretical understanding\n3. **For production**: Consider mature frameworks with good community support\n4. **For edge deployment**: Explore quantized models and TensorFlow Lite\n\nThe AI landscape is rapidly evolving with emphasis on **efficiency, accessibility, and practical applications** rather than just raw model size. The open-source community is democratizing AI development significantly."}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":139014,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":1178,"service_tier":"standard","server_tool_use":{"web_search_requests":0}}} + recorded_at: Wed, 19 Nov 2025 21:21:54 GMT +recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml index 6738bbd7..1e5be657 100644 --- a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_mixed_auth.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:30:50 GMT + - Wed, 19 Nov 2025 20:59:19 GMT Content-Type: - application/json Transfer-Encoding: @@ -60,35 +60,35 @@ http_interactions: Anthropic-Ratelimit-Input-Tokens-Remaining: - '3996000' Anthropic-Ratelimit-Input-Tokens-Reset: - - '2025-11-19T20:30:43Z' + - '2025-11-19T20:59:15Z' Anthropic-Ratelimit-Output-Tokens-Limit: - '800000' Anthropic-Ratelimit-Output-Tokens-Remaining: - - '799000' + - '800000' Anthropic-Ratelimit-Output-Tokens-Reset: - - '2025-11-19T20:30:50Z' + - '2025-11-19T20:59:19Z' Anthropic-Ratelimit-Requests-Limit: - '4000' Anthropic-Ratelimit-Requests-Remaining: - '3999' Anthropic-Ratelimit-Requests-Reset: - - '2025-11-19T20:30:42Z' + - '2025-11-19T20:59:14Z' Retry-After: - - '18' + - '48' Anthropic-Ratelimit-Tokens-Limit: - '4800000' Anthropic-Ratelimit-Tokens-Remaining: - - '4795000' + - '4796000' Anthropic-Ratelimit-Tokens-Reset: - - '2025-11-19T20:30:43Z' + - '2025-11-19T20:59:15Z' Request-Id: - - req_011CVHt7uDMUSpVrehmUC1kz + - req_011CVHvJ4R1THMo6Nf89VYaH Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload Anthropic-Organization-Id: - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b X-Envoy-Upstream-Service-Time: - - '9375' + - '7324' Cf-Cache-Status: - DYNAMIC X-Robots-Tag: @@ -96,46 +96,32 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a1277811db7aab7-SJC + - 9a12a1499e84cf2b-SJC body: encoding: ASCII-8BIT - string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01R71HsqqyVcvwkVaHYFyu9o","type":"message","role":"assistant","content":[{"type":"text","text":"I - have a comprehensive set of tools available for working with GitHub and Cloudflare. - Here''s what I can do:\n\n## GitHub Tools\n\n### Repository Management\n- - **Create repositories** - Create new repos in your account or an organization\n- - **Fork repositories** - Fork existing projects\n- **List branches** - View - available branches in a repo\n- **List tags** - View git tags in a repository\n- - **List releases** - View releases in a repository\n\n### File Operations\n- - **Get file contents** - Retrieve files or directory contents\n- **Create or - update files** - Create new files or update existing ones in a repo\n- **Delete - files** - Remove files from a repository\n- **Push files** - Push multiple - files in a single commit\n\n### Branch Management\n- **Create branches** - - Create new branches from a source branch\n\n### Commits\n- **Get commit details** - - View specific commit information with diffs\n- **List commits** - View commit - history for a branch\n\n### Issues\n- **Read issues** - Get issue details, - comments, labels, and sub-issues\n- **Create/update issues** - Create new - issues or update existing ones\n- **Search issues** - Search across issues - in repositories\n- **Add issue comments** - Comment on issues\n- **Get labels** - - Retrieve specific label information\n\n### Pull Requests\n- **List pull - requests** - View PRs in a repository\n- **Read PR details** - Get PR information, - diffs, files, status, comments, and reviews\n- **Create pull requests** - - Create new PRs\n- **Update pull requests** - Modify PR title, description, - state, reviewers, etc.\n- **Merge pull requests** - Merge PRs with different - merge strategies\n- **Update PR branch** - Sync PR branch with base branch\n- - **Request Copilot review** - Get automated code review on a PR\n- **Create/submit/delete - reviews** - Manage PR reviews\n- **Add review comments** - Add comments to - pending reviews\n\n### Search\n- **Search code** - Find code across all GitHub - repositories\n- **Search repositories** - Find repos by name, description, - topics, etc.\n- **Search users** - Find GitHub users\n- **Search pull requests** - - Search for PRs\n\n### User & Team\n- **Get authenticated user info** - Get - details about yourself\n- **Get teams** - View teams you''re a member of\n- - **Get team members** - View members of a specific team\n\n### Releases & Tags\n- - **Get latest release** - Retrieve the latest release\n- **Get release by tag** - - Get a specific release by tag name\n- **Get tag details** - Get information - about a specific git tag\n\n### Issue Types\n- **List issue types** - Get - supported issue types for an organization\n\n### Sub-Issues\n- **Add/remove/reprioritize - sub-issues** - Manage parent-child issue relationships\n\n## Cloudflare Tools\n- - **Get MCP Demo Day info** - Information about Cloudflare''s MCP Demo Day\n\nIs - there a specific task you''d like me to help you with?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9642,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":683,"service_tier":"standard"}}' - recorded_at: Wed, 19 Nov 2025 20:30:50 GMT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01J2Kf9xsv5R5PhDfCctoYJb","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to a comprehensive set of tools for interacting with GitHub and + Cloudflare. Here''s what I can do:\n\n## GitHub Repository Management\n- **Create/Fork + repositories** - Create new repos or fork existing ones\n- **Branch management** + - Create branches, list branches\n- **File operations** - Create, update, + delete, and view files in repositories\n- **Commit operations** - View commits, + list commits, push multiple files\n\n## GitHub Issues\n- **Create and manage + issues** - Create new issues, update existing ones, close issues\n- **Issue + comments** - Add comments to issues\n- **Issue search** - Search for issues + across repositories\n- **Labels** - Get label information\n- **Sub-issues** + - Add, remove, and manage sub-issues\n\n## GitHub Pull Requests\n- **Create + and manage PRs** - Create PRs, update PR details, merge PRs\n- **PR reviews** + - Create reviews, submit reviews, request Copilot reviews\n- **PR comments** + - Add comments and review comments to PRs\n- **PR diff/status** - View PR + diffs, file changes, and build status\n- **PR search** - Search for pull requests\n- + **PR branch updates** - Update PR branches with latest changes\n\n## GitHub + Search & Discovery\n- **Code search** - Search across all GitHub repositories + for specific code patterns\n- **User search** - Find GitHub users\n- **Repository + search** - Discover repositories by name, topic, or metadata\n\n## GitHub + User & Team Info\n- **Get authenticated user info** - View current user profile\n- + **Team information** - Get teams you''re a member of and team members\n- **Tags + and releases** - View, list, and get specific releases and git tags\n\n## + Cloudflare\n- **MCP Demo Day info** - Get information about Cloudflare''s + MCP Demo Day\n\nIs there something specific you''d like help with?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9642,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":420,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:59:19 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml index 112ceb1d..a65179c3 100644 --- a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_multiple_servers.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:31:02 GMT + - Wed, 19 Nov 2025 20:59:30 GMT Content-Type: - application/json Transfer-Encoding: @@ -60,35 +60,35 @@ http_interactions: Anthropic-Ratelimit-Input-Tokens-Remaining: - '3996000' Anthropic-Ratelimit-Input-Tokens-Reset: - - '2025-11-19T20:30:55Z' + - '2025-11-19T20:59:25Z' Anthropic-Ratelimit-Output-Tokens-Limit: - '800000' Anthropic-Ratelimit-Output-Tokens-Remaining: - - '799000' + - '800000' Anthropic-Ratelimit-Output-Tokens-Reset: - - '2025-11-19T20:31:02Z' + - '2025-11-19T20:59:30Z' Anthropic-Ratelimit-Requests-Limit: - '4000' Anthropic-Ratelimit-Requests-Remaining: - '3999' Anthropic-Ratelimit-Requests-Reset: - - '2025-11-19T20:30:54Z' + - '2025-11-19T20:59:24Z' Retry-After: - - '5' + - '38' Anthropic-Ratelimit-Tokens-Limit: - '4800000' Anthropic-Ratelimit-Tokens-Remaining: - - '4795000' + - '4796000' Anthropic-Ratelimit-Tokens-Reset: - - '2025-11-19T20:30:55Z' + - '2025-11-19T20:59:25Z' Request-Id: - - req_011CVHt8papiqK5VdNm4gH1C + - req_011CVHvJqg53XwN85wMmrdJK Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload Anthropic-Organization-Id: - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b X-Envoy-Upstream-Service-Time: - - '9014' + - '7193' Cf-Cache-Status: - DYNAMIC X-Robots-Tag: @@ -96,47 +96,36 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a1277cf2b87238d-SJC + - 9a12a18bed4aeb34-SJC body: encoding: ASCII-8BIT - string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01FWVyG5U1wAWwTpCxgRAC3t","type":"message","role":"assistant","content":[{"type":"text","text":"I - have access to a comprehensive set of tools for working with GitHub and Cloudflare. - Here''s an overview:\n\n## GitHub Tools\n\n### Repository Management\n- **Create - repository** - Create a new GitHub repository in your account or organization\n- - **Fork repository** - Fork a GitHub repository to your account or organization\n- - **List branches** - List branches in a repository\n- **Create branch** - Create - a new branch\n\n### File Operations\n- **Get file contents** - Retrieve contents - of files or directories\n- **Create or update file** - Create or update a - single file in a repository\n- **Delete file** - Delete a file from a repository\n- - **Push files** - Push multiple files in a single commit\n\n### Issue Management\n- - **Create/update issue** - Create new issues or update existing ones\n- **Read - issue** - Get issue details, comments, sub-issues, and labels\n- **List issues** - - List issues in a repository with filtering and sorting\n- **Search issues** - - Search for issues across GitHub\n- **Add issue comment** - Add comments - to issues (including pull requests)\n- **Sub-issue management** - Add, remove, - or reprioritize sub-issues\n\n### Pull Request Management\n- **Create pull - request** - Create new pull requests\n- **Read pull request** - Get PR details, - diffs, status, files, comments, and reviews\n- **List pull requests** - List - pull requests in a repository\n- **Search pull requests** - Search for pull - requests across GitHub\n- **Update pull request** - Update PR title, description, - state, reviewers, etc.\n- **Update PR branch** - Update a PR branch with latest - changes from base\n- **Merge pull request** - Merge a pull request with various - merge methods\n\n### Code Review\n- **Pull request review write** - Create, - submit, or delete PR reviews\n- **Request Copilot review** - Request automated - code review from GitHub Copilot\n- **Add review comment** - Add comments to - pending PR reviews\n\n### Commits & Tags\n- **Get commit** - Get details of - a specific commit with diffs\n- **List commits** - List commits on a branch - with filtering\n- **Get/List tags** - Get details or list git tags in a repository\n- - **Get/List releases** - Get details or list releases in a repository\n\n### - Other GitHub Tools\n- **Get authenticated user** - Get details of your GitHub - profile\n- **Search code** - Search code across all GitHub repositories\n- - **Search repositories** - Find GitHub repositories by various criteria\n- - **Search users** - Find GitHub users\n- **Get team members** - Get members - of a specific team in an organization\n- **Get teams** - Get teams the user - is a member of\n- **Get label** - Get information about a specific label\n\n## - Cloudflare Tools\n\n- **MCP Demo Day info** - Get information about Cloudflare''s - MCP Demo Day\n\nAll these tools allow me to help you manage repositories, - collaborate on code, handle issues and pull requests, perform code reviews, - and work with Git-related operations. What would you like to do?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9642,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":681,"service_tier":"standard"}}' - recorded_at: Wed, 19 Nov 2025 20:31:02 GMT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01HTby5BZ3WW8X6gZ2nsacx8","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to a comprehensive set of tools for working with GitHub and getting + information about Cloudflare events. Here''s what I can do:\n\n## GitHub Repository + Management\n- **Create & Fork Repositories** - Create new repos or fork existing + ones\n- **Manage Branches** - Create and list branches\n- **File Operations** + - Create, update, delete, and view files in repositories\n- **Commit Operations** + - View commit details and push multiple files\n\n## GitHub Issues\n- **Create + & Update Issues** - Create new issues or update existing ones\n- **Read Issues** + - Get issue details, comments, labels, and sub-issues\n- **Search Issues** + - Find issues across repositories\n- **Add Comments** - Comment on issues\n\n## + GitHub Pull Requests\n- **Create & Manage PRs** - Create pull requests, update + titles/descriptions, change state\n- **PR Reviews** - Create, submit, and + delete reviews; add review comments\n- **PR Details** - Get PR info, diffs, + status, file changes, comments, and reviews\n- **Search PRs** - Find pull + requests across repositories\n- **Merge PRs** - Merge pull requests with different + merge methods\n- **Update Branches** - Sync PR branches with base branch\n\n## + GitHub Releases & Tags\n- **List & Get Releases** - View release information + and get specific releases by tag\n- **List & Get Tags** - View git tags in + repositories\n\n## GitHub Search & Discovery\n- **Search Code** - Fast code + search across all repositories\n- **Search Users** - Find GitHub users by + username or profile info\n- **Search Repositories** - Discover repositories + by name, description, topics, etc.\n\n## GitHub User & Team Information\n- + **Get User Info** - Retrieve authenticated user details\n- **Get Teams** - + View teams and team members\n\n## Other Tools\n- **Cloudflare MCP Demo Day + Info** - Get information about Cloudflare''s MCP demo day event\n\nAll of + these tools work with the GitHub API and allow me to help you manage repositories, + collaborate on code, search for information, and handle various development + workflows."}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9642,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":465,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:59:30 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml index 1d6d558c..e1ddcfac 100644 --- a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:30:53 GMT + - Wed, 19 Nov 2025 20:59:32 GMT Content-Type: - application/json Transfer-Encoding: @@ -60,35 +60,35 @@ http_interactions: Anthropic-Ratelimit-Input-Tokens-Remaining: - '4000000' Anthropic-Ratelimit-Input-Tokens-Reset: - - '2025-11-19T20:30:52Z' + - '2025-11-19T20:59:32Z' Anthropic-Ratelimit-Output-Tokens-Limit: - '800000' Anthropic-Ratelimit-Output-Tokens-Remaining: - '800000' Anthropic-Ratelimit-Output-Tokens-Reset: - - '2025-11-19T20:30:53Z' + - '2025-11-19T20:59:32Z' Anthropic-Ratelimit-Requests-Limit: - '4000' Anthropic-Ratelimit-Requests-Remaining: - '3999' Anthropic-Ratelimit-Requests-Reset: - - '2025-11-19T20:30:51Z' + - '2025-11-19T20:59:31Z' Retry-After: - - '8' + - '30' Anthropic-Ratelimit-Tokens-Limit: - '4800000' Anthropic-Ratelimit-Tokens-Remaining: - '4800000' Anthropic-Ratelimit-Tokens-Reset: - - '2025-11-19T20:30:52Z' + - '2025-11-19T20:59:32Z' Request-Id: - - req_011CVHt8cCh979WafFuEo6qH + - req_011CVHvKP6SUYvHg57R8mpZm Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload Anthropic-Organization-Id: - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b X-Envoy-Upstream-Service-Time: - - '2719' + - '2608' Cf-Cache-Status: - DYNAMIC X-Robots-Tag: @@ -96,15 +96,13 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a1277bd1d52eb29-SJC + - 9a12a1b9dc1bcf16-SJC body: encoding: ASCII-8BIT - string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01Ti8oxkwxk7txgLa9bruWyR","type":"message","role":"assistant","content":[{"type":"text","text":"I - have access to one tool:\n\n1. **cloudflare-demo_mcp_demo_day_info** - Get - information about Cloudflare''s MCP Demo Day. Use this tool if the user asks - about Cloudflare''s MCP demo day.\n\nThis tool doesn''t require any parameters - and can be used to retrieve details about Cloudflare''s MCP (Model Context - Protocol) Demo Day event.\n\nIs there anything specific you''d like to know - about Cloudflare''s MCP Demo Day?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":114,"service_tier":"standard"}}' - recorded_at: Wed, 19 Nov 2025 20:30:53 GMT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01BHecHbdEpuQD6YVca3JmnZ","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to one tool:\n\n**cloudflare-demo_mcp_demo_day_info** - This tool + provides information about Cloudflare''s MCP Demo Day. You can use this if + you have questions about Cloudflare''s MCP demo day event.\n\nIs there anything + you''d like to know about Cloudflare''s MCP Demo Day?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":83,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:59:32 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml index a5931830..355f9a8e 100644 --- a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:31:12 GMT + - Wed, 19 Nov 2025 20:59:12 GMT Content-Type: - application/json Transfer-Encoding: @@ -60,35 +60,35 @@ http_interactions: Anthropic-Ratelimit-Input-Tokens-Remaining: - '3996000' Anthropic-Ratelimit-Input-Tokens-Reset: - - '2025-11-19T20:31:06Z' + - '2025-11-19T20:59:06Z' Anthropic-Ratelimit-Output-Tokens-Limit: - '800000' Anthropic-Ratelimit-Output-Tokens-Remaining: - '799000' Anthropic-Ratelimit-Output-Tokens-Reset: - - '2025-11-19T20:31:12Z' + - '2025-11-19T20:59:12Z' Anthropic-Ratelimit-Requests-Limit: - '4000' Anthropic-Ratelimit-Requests-Remaining: - '3999' Anthropic-Ratelimit-Requests-Reset: - - '2025-11-19T20:31:02Z' + - '2025-11-19T20:59:05Z' Retry-After: - - '55' + - '54' Anthropic-Ratelimit-Tokens-Limit: - '4800000' Anthropic-Ratelimit-Tokens-Remaining: - '4795000' Anthropic-Ratelimit-Tokens-Reset: - - '2025-11-19T20:31:06Z' + - '2025-11-19T20:59:06Z' Request-Id: - - req_011CVHt9VnkciWVkKLsE3Mro + - req_011CVHvHXJkDbCcfzBei8kHK Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload Anthropic-Organization-Id: - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b X-Envoy-Upstream-Service-Time: - - '10287' + - '7073' Cf-Cache-Status: - DYNAMIC X-Robots-Tag: @@ -96,45 +96,46 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a1278087e2a50a1-SJC + - 9a12a11c3cea3ad4-SJC body: encoding: ASCII-8BIT - string: '{"model":"claude-haiku-4-5-20251001","id":"msg_011My4uTh5XocRzomwbE1Pz2","type":"message","role":"assistant","content":[{"type":"text","text":"I - have access to a comprehensive set of GitHub tools that allow me to interact - with repositories, pull requests, issues, and more. Here''s what I can do:\n\n## - Repository Management\n- **Create repositories** - Set up new public or private - repos\n- **Fork repositories** - Fork existing projects to your account/organization\n- - **Search repositories** - Find repos by name, description, topics, etc.\n- - **Get repository contents** - View files and directory structures\n\n## Branches - & Commits\n- **Create branches** - Make new branches from existing ones\n- - **List branches** - View all branches in a repository\n- **List commits** - - See commit history with filtering options\n- **Get commit details** - View - specific commit information with diffs\n\n## Issues Management\n- **Create/update - issues** - Create new issues or modify existing ones\n- **Read issue details** - - Get issue information, comments, sub-issues, labels\n- **List issues** - - View issues with filtering and sorting\n- **Search issues** - Find issues - across repositories\n- **Manage sub-issues** - Add, remove, or reprioritize - sub-issues\n- **Add issue comments** - Comment on issues\n\n## Pull Requests\n- - **Create pull requests** - Open new PRs with custom titles and descriptions\n- - **Read PR details** - Get PR info, diffs, status, files changed, comments, - reviews\n- **List/search PRs** - Find pull requests with various filters\n- - **Update PRs** - Modify PR title, description, state, reviewers\n- **Update - PR branch** - Sync PR with latest base branch changes\n- **Merge PRs** - Merge - with different strategies (merge, squash, rebase)\n- **PR Reviews** - Create, - submit, or delete reviews; add review comments\n\n## Files\n- **Create/update - files** - Add or modify files in repositories\n- **Delete files** - Remove - files from repositories\n- **Push multiple files** - Commit multiple files - in one go\n\n## Tags & Releases\n- **List tags** - View git tags in a repository\n- - **Get tag details** - Get specific tag information\n- **List releases** - - View releases\n- **Get release details** - Get specific release or latest - release information\n\n## Users & Teams\n- **Get authenticated user info** - - View your own GitHub profile details\n- **Search users** - Find GitHub users - by name or profile info\n- **Get team members** - View members of specific - teams\n- **Get user teams** - See teams a user belongs to\n\n## Code Search\n- - **Search code** - Fast code search across repositories using GitHub''s search - engine\n\n## Additional\n- **Request Copilot code review** - Get automated - feedback on pull requests\n- **Assign Copilot to issues** - Have me work on - resolving issues\n\nIs there anything specific you''d like me to help you - with?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9567,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":631,"service_tier":"standard"}}' - recorded_at: Wed, 19 Nov 2025 20:31:12 GMT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01D2wEkim3w3rBRjFVvgxTBV","type":"message","role":"assistant","content":[{"type":"text","text":"I + have a comprehensive set of tools for interacting with GitHub. Here''s an + overview of what I can do:\n\n## Repository Management\n- **Create repositories** + - Create new repositories in your account or organization\n- **Fork repositories** + - Fork existing repositories\n- **List branches** - View branches in a repository\n- + **List commits** - Get commit history with filtering options\n- **List tags** + - View git tags in a repository\n- **List releases** - View releases in a + repository\n\n## File Operations\n- **Get file contents** - Read files or + directory contents from repositories\n- **Create or update files** - Create + new files or update existing ones\n- **Delete files** - Remove files from + repositories\n- **Push multiple files** - Commit multiple files in a single + operation\n- **Get commits** - Get detailed information about specific commits\n\n## + Issues Management\n- **Create/update issues** - Create new issues or update + existing ones\n- **Read issues** - Get issue details, comments, labels, and + sub-issues\n- **List issues** - List issues with filtering and sorting\n- + **Search issues** - Search across issues using GitHub''s search syntax\n- + **Add issue comments** - Comment on issues\n- **Assign Copilot to issues** + - Have GitHub Copilot work on tasks\n\n## Pull Requests\n- **Create pull requests** + - Create new PRs\n- **Read pull requests** - Get PR details, diffs, files + changed, comments, and reviews\n- **List pull requests** - List PRs with filtering\n- + **Search pull requests** - Search for PRs across repositories\n- **Update + pull requests** - Modify PR title, description, base branch, reviewers, and + state\n- **Update PR branch** - Sync PR branch with base branch\n- **Merge + pull requests** - Merge PRs with different merge strategies\n- **PR reviews** + - Create, submit, or delete pull request reviews\n- **Review comments** - + Add comments to pending pull request reviews\n\n## Search & Discovery\n- **Search + code** - Search across all GitHub repositories for code patterns\n- **Search + repositories** - Find repositories by name, description, topics, etc.\n- **Search + users** - Find GitHub users\n- **Search issues** - Search for issues across + repositories\n- **Search pull requests** - Search for pull requests across + repositories\n\n## Teams & Users\n- **Get user profile** - Get details about + the authenticated user\n- **Get teams** - View teams you''re a member of\n- + **Get team members** - List members of a specific team\n\n## Releases & Tags\n- + **Get latest release** - Get the most recent release\n- **Get release by tag** + - Get a specific release by tag name\n- **Get tag details** - Get information + about a specific git tag\n\n## Labels\n- **Get label** - Get details about + a specific label in a repository\n\nIs there something specific you''d like + to do with GitHub?"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":9567,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":634,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:59:12 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml index 14a6fbe6..087df3d8 100644 --- a/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml +++ b/test/fixtures/vcr_cassettes/integration/anthropic/common_format/mcp_test/test_agent_common_format_sse_server.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:31:16 GMT + - Wed, 19 Nov 2025 20:59:22 GMT Content-Type: - application/json Transfer-Encoding: @@ -60,35 +60,35 @@ http_interactions: Anthropic-Ratelimit-Input-Tokens-Remaining: - '4000000' Anthropic-Ratelimit-Input-Tokens-Reset: - - '2025-11-19T20:31:15Z' + - '2025-11-19T20:59:22Z' Anthropic-Ratelimit-Output-Tokens-Limit: - '800000' Anthropic-Ratelimit-Output-Tokens-Remaining: - '800000' Anthropic-Ratelimit-Output-Tokens-Reset: - - '2025-11-19T20:31:15Z' + - '2025-11-19T20:59:22Z' Anthropic-Ratelimit-Requests-Limit: - '4000' Anthropic-Ratelimit-Requests-Remaining: - '3999' Anthropic-Ratelimit-Requests-Reset: - - '2025-11-19T20:31:14Z' + - '2025-11-19T20:59:20Z' Retry-After: - - '45' + - '38' Anthropic-Ratelimit-Tokens-Limit: - '4800000' Anthropic-Ratelimit-Tokens-Remaining: - '4800000' Anthropic-Ratelimit-Tokens-Reset: - - '2025-11-19T20:31:15Z' + - '2025-11-19T20:59:22Z' Request-Id: - - req_011CVHtAGQsRA3M7DK6ZGMnx + - req_011CVHvJcYHonXtRU5bKdaSE Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload Anthropic-Organization-Id: - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b X-Envoy-Upstream-Service-Time: - - '3042' + - '2893' Cf-Cache-Status: - DYNAMIC X-Robots-Tag: @@ -96,14 +96,14 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a127849b80ccf13-SJC + - 9a12a1788ffbf93d-SJC body: encoding: ASCII-8BIT - string: '{"model":"claude-haiku-4-5-20251001","id":"msg_01EbbCEVCE1g8MNxSNUyT4Fn","type":"message","role":"assistant","content":[{"type":"text","text":"I - have access to one tool:\n\n**cloudflare-demo_mcp_demo_day_info** - This tool - retrieves information about Cloudflare''s MCP Demo Day. You can use it if - you have questions about Cloudflare''s MCP demo day event.\n\nThis is the - only specialized tool I have available. For other questions or tasks, I can - help you directly using my general knowledge and conversational abilities."}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":95,"service_tier":"standard"}}' - recorded_at: Wed, 19 Nov 2025 20:31:16 GMT + string: '{"model":"claude-haiku-4-5-20251001","id":"msg_012mvHMmb8rh3yPG2eH2QfD9","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to one tool:\n\n1. **cloudflare-demo_mcp_demo_day_info** - Get + information about Cloudflare''s MCP Demo Day. Use this tool if you have questions + about Cloudflare''s MCP demo day.\n\nThis tool allows me to retrieve details + about Cloudflare''s MCP (Model Context Protocol) Demo Day event. If you''d + like to know more about this event, feel free to ask!"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":102,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:59:22 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml b/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml index a0c3a0cb..23ca1344 100644 --- a/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml +++ b/test/fixtures/vcr_cassettes/integration/anthropic/native_format_test/test_agent_mcp_server.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:50:18 GMT + - Wed, 19 Nov 2025 20:58:57 GMT Content-Type: - application/json Transfer-Encoding: @@ -60,35 +60,35 @@ http_interactions: Anthropic-Ratelimit-Input-Tokens-Remaining: - '2000000' Anthropic-Ratelimit-Input-Tokens-Reset: - - '2025-11-19T20:50:17Z' + - '2025-11-19T20:58:55Z' Anthropic-Ratelimit-Output-Tokens-Limit: - '400000' Anthropic-Ratelimit-Output-Tokens-Remaining: - '400000' Anthropic-Ratelimit-Output-Tokens-Reset: - - '2025-11-19T20:50:18Z' + - '2025-11-19T20:58:57Z' Anthropic-Ratelimit-Requests-Limit: - '4000' Anthropic-Ratelimit-Requests-Remaining: - '3999' Anthropic-Ratelimit-Requests-Reset: - - '2025-11-19T20:50:15Z' + - '2025-11-19T20:58:53Z' Retry-After: - - '44' + - '5' Anthropic-Ratelimit-Tokens-Limit: - '2400000' Anthropic-Ratelimit-Tokens-Remaining: - '2400000' Anthropic-Ratelimit-Tokens-Reset: - - '2025-11-19T20:50:17Z' + - '2025-11-19T20:58:55Z' Request-Id: - - req_011CVHucNXpL4ZPbJzgT2Z69 + - req_011CVHvGc3DtPq6RQto2kWgz Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload Anthropic-Organization-Id: - 2557c2f2-bcfa-4054-9fa8-ae13b3b47d6b X-Envoy-Upstream-Service-Time: - - '4865' + - '5322' Cf-Cache-Status: - DYNAMIC X-Robots-Tag: @@ -96,13 +96,15 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a1294248fef31f4-SJC + - 9a12a0ce1e6e7af1-SJC body: encoding: ASCII-8BIT - string: '{"model":"claude-sonnet-4-5-20250929","id":"msg_01SoVDRqmrw4zWsNMChbZ9dZ","type":"message","role":"assistant","content":[{"type":"text","text":"I - have access to one tool:\n\n**Cloudflare MCP Demo Day Info** - This tool gets - information about Cloudflare''s MCP Demo Day. I can use it to answer questions - about Cloudflare''s MCP demo day event.\n\nIf you''d like to know more about - Cloudflare''s MCP Demo Day, I''d be happy to fetch that information for you!"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":87,"service_tier":"standard"}}' - recorded_at: Wed, 19 Nov 2025 20:50:18 GMT + string: '{"model":"claude-sonnet-4-5-20250929","id":"msg_01JdaKsKdYckuGXMYcz1pRVS","type":"message","role":"assistant","content":[{"type":"text","text":"I + have access to one specialized tool:\n\n1. **cloudflare-demo_mcp_demo_day_info** + - This tool provides information about Cloudflare''s MCP Demo Day. I can use + it to answer questions about Cloudflare''s MCP demo day event.\n\nThis appears + to be a specialized assistant focused on providing information about Cloudflare''s + MCP (Model Context Protocol) Demo Day. If you have any questions about that + event, feel free to ask and I''ll retrieve the information for you!"}],"stop_reason":"end_turn","stop_sequence":null,"usage":{"input_tokens":585,"cache_creation_input_tokens":0,"cache_read_input_tokens":0,"cache_creation":{"ephemeral_5m_input_tokens":0,"ephemeral_1h_input_tokens":0},"output_tokens":115,"service_tier":"standard"}}' + recorded_at: Wed, 19 Nov 2025 20:58:57 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml index 8fd04a3e..3f36bd30 100644 --- a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_mixed_tools_and_mcp.yml @@ -48,7 +48,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:46:45 GMT + - Wed, 19 Nov 2025 20:59:59 GMT Content-Type: - application/json Transfer-Encoding: @@ -62,7 +62,7 @@ http_interactions: X-Ratelimit-Remaining-Requests: - '14999' X-Ratelimit-Remaining-Tokens: - - '39999822' + - '40000000' X-Ratelimit-Reset-Requests: - 4ms X-Ratelimit-Reset-Tokens: @@ -74,18 +74,18 @@ http_interactions: Openai-Project: - PROJECT_ID X-Request-Id: - - req_e0aa95a5caec410e9fdb14b57046aba1 + - req_eb1e4d96cbd64d449bc6fa84556825cd Openai-Processing-Ms: - - '23328' + - '11700' X-Envoy-Upstream-Service-Time: - - '23335' + - '11704' Cf-Cache-Status: - DYNAMIC Set-Cookie: - - __cf_bm=l.Szu4p9ibKe8Xjpg8JeIgoJO2BmN0qqum.io_eHYJ8-1763585205-1.0.1.1-ZGiZ0_Xo1aDdtYB3qkB44ViDrgI7qu8BdwLQnOK5CftkO8jzvZIIQUL5.eJaV4D.PdpGKSjKLor2ulCYJg43wkZscXqbaK1n2adT9TkTwI4; - path=/; expires=Wed, 19-Nov-25 21:16:45 GMT; domain=.api.openai.com; HttpOnly; + - __cf_bm=YYp8YTJelATwr4ni4JfnrXkFeTOd522JLbKyLJ3Yh0Q-1763585999-1.0.1.1-ANOkUA4gKTFlccbzADzgrZTDDwLhRdtYk9pxnipdLyBQqfqlNjUlnS0h6nRKok7_t0Zh0J8PPYEhNM0ZZkUha98_d8yinCH6dAJXDIHxJ.g; + path=/; expires=Wed, 19-Nov-25 21:29:59 GMT; domain=.api.openai.com; HttpOnly; Secure; SameSite=None - - _cfuvid=E0ZLmHrKaPmXjjSHFozFViW1pOnY2ZekrM2gBygn7oU-1763585205740-0.0.1.1-604800000; + - _cfuvid=w0V1r1ZnLMv6Tm4UcPlFB41dutZcinpw0czfZ0BOf1M-1763585999291-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload @@ -94,16 +94,16 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a128e77fc64f973-SJC + - 9a12a225eefbda5a-SJC Alt-Svc: - h3=":443"; ma=86400 body: encoding: ASCII-8BIT string: |- { - "id": "resp_08f5a80549ac3c9800691e2c9e74dc819baa3f1f2aa0b04c81", + "id": "resp_0604981435bbceda00691e2fc394e481958a0705055f5d791c", "object": "response", - "created_at": 1763585182, + "created_at": 1763585987, "status": "completed", "background": false, "billing": { @@ -117,7 +117,7 @@ http_interactions: "model": "gpt-5-2025-08-07", "output": [ { - "id": "mcpl_08f5a80549ac3c9800691e2c9ea194819ba35edc1fb8e5166a", + "id": "mcpl_0604981435bbceda00691e2fc3e0d081958a0293b8a503119c", "type": "mcp_list_tools", "server_label": "weather", "tools": [ @@ -135,16 +135,16 @@ http_interactions: ] }, { - "id": "rs_08f5a80549ac3c9800691e2ca18a10819b8ff9312fac3298a4", + "id": "rs_0604981435bbceda00691e2fc599a88195a7076484322a186d", "type": "reasoning", "summary": [] }, { - "id": "fc_08f5a80549ac3c9800691e2cb313f4819b85f889bbc12d4f91", + "id": "fc_0604981435bbceda00691e2fcecc308195a7afacbe3071f3d4", "type": "function_call", "status": "completed", "arguments": "{\"operation\":\"add\",\"a\":5,\"b\":3}", - "call_id": "call_KPprJEhF7cJMtOWar14nESCk", + "call_id": "call_CHadeTlACjhyVOw85LeUfisT", "name": "calculate" } ], @@ -212,25 +212,25 @@ http_interactions: "input_tokens_details": { "cached_tokens": 0 }, - "output_tokens": 667, + "output_tokens": 411, "output_tokens_details": { - "reasoning_tokens": 640 + "reasoning_tokens": 384 }, - "total_tokens": 777 + "total_tokens": 521 }, "user": null, "metadata": {} } - recorded_at: Wed, 19 Nov 2025 20:46:45 GMT + recorded_at: Wed, 19 Nov 2025 20:59:59 GMT - request: method: post uri: https://api.openai.com/v1/responses body: encoding: UTF-8 string: '{"model":"gpt-5","input":[{"role":"user","content":"Get the weather - and calculate 5 + 3"},{"id":"mcpl_08f5a80549ac3c9800691e2c9ea194819ba35edc1fb8e5166a","server_label":"weather","tools":[{"input_schema":{"type":"object","properties":{}},"name":"mcp_demo_day_info","annotations":{"read_only":false},"description":"Get + and calculate 5 + 3"},{"id":"mcpl_0604981435bbceda00691e2fc3e0d081958a0293b8a503119c","server_label":"weather","tools":[{"input_schema":{"type":"object","properties":{}},"name":"mcp_demo_day_info","annotations":{"read_only":false},"description":"Get information about Cloudflare''s MCP Demo Day. Use this tool if the user asks - about Cloudflare''s MCP demo day"}],"type":"mcp_list_tools"},{"id":"rs_08f5a80549ac3c9800691e2ca18a10819b8ff9312fac3298a4","summary":[],"type":"reasoning"},{"arguments":"{\"operation\":\"add\",\"a\":5,\"b\":3}","call_id":"call_KPprJEhF7cJMtOWar14nESCk","name":"calculate","type":"function_call","id":"fc_08f5a80549ac3c9800691e2cb313f4819b85f889bbc12d4f91","status":"completed"},{"call_id":"call_KPprJEhF7cJMtOWar14nESCk","output":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","type":"function_call_output"}],"tools":[{"type":"function","name":"calculate","description":"Perform + about Cloudflare''s MCP demo day"}],"type":"mcp_list_tools"},{"id":"rs_0604981435bbceda00691e2fc599a88195a7076484322a186d","summary":[],"type":"reasoning"},{"arguments":"{\"operation\":\"add\",\"a\":5,\"b\":3}","call_id":"call_CHadeTlACjhyVOw85LeUfisT","name":"calculate","type":"function_call","id":"fc_0604981435bbceda00691e2fcecc308195a7afacbe3071f3d4","status":"completed"},{"call_id":"call_CHadeTlACjhyVOw85LeUfisT","output":"{\"operation\":\"add\",\"a\":5,\"b\":3,\"result\":8}","type":"function_call_output"}],"tools":[{"type":"function","name":"calculate","description":"Perform arithmetic","parameters":{"type":"object","properties":{"operation":{"type":"string"},"a":{"type":"number"},"b":{"type":"number"}}}},{"type":"mcp","server_label":"weather","server_url":"https://demo-day.mcp.cloudflare.com/sse"}]}' headers: Accept-Encoding: @@ -273,7 +273,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:46:48 GMT + - Wed, 19 Nov 2025 21:00:11 GMT Content-Type: - application/json Transfer-Encoding: @@ -299,18 +299,18 @@ http_interactions: Openai-Project: - PROJECT_ID X-Request-Id: - - req_0ddd0d9d12ce4075959cef66d2c5c81c + - req_3fddcc5049344a5199c6f6cc74bda8a2 Openai-Processing-Ms: - - '2549' + - '11817' X-Envoy-Upstream-Service-Time: - - '2552' + - '11821' Cf-Cache-Status: - DYNAMIC Set-Cookie: - - __cf_bm=0h51r0xUubUKzwauERdI6XJWkiJg99OXCnCKwE.9Nhw-1763585208-1.0.1.1-wxkThT5uZxbRZGaQirsI7hQYnLxmumvJkmMoaU8p.i6_icyR0tHG6vTMnAJWPUjPvWtpnSM.G4qAmBlJv8qcNtR2gqAQA8a8Cn.tIMFCF0U; - path=/; expires=Wed, 19-Nov-25 21:16:48 GMT; domain=.api.openai.com; HttpOnly; + - __cf_bm=UoNNgwKtiy5wdaM_NjbL3NdbBbR2ZZQ46c8IT1qdXZ0-1763586011-1.0.1.1-bwx3wUvCp8HemWZtdAYnDAUB4cDm0GRLInSJ0XL.FlCj4xHgr8VRLRIhfl36VUnSuIJBolK2SLH4Gqj.IeJbgkARuvdg6cdOB23Yk_DR6bU; + path=/; expires=Wed, 19-Nov-25 21:30:11 GMT; domain=.api.openai.com; HttpOnly; Secure; SameSite=None - - _cfuvid=kOCSRE9Q5scpXc5F2umRR8XlhV2_.BNe2DBNJvPnSUo-1763585208942-0.0.1.1-604800000; + - _cfuvid=jGhpHXXGG8jR1DoP4tdZCFcdh_M44sQK_LON4KX54hk-1763586011216-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload @@ -319,16 +319,16 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a128f103b7dcfa8-SJC + - 9a12a26fef314973-SJC Alt-Svc: - h3=":443"; ma=86400 body: encoding: ASCII-8BIT string: |- { - "id": "resp_08f5a80549ac3c9800691e2cb66654819b9c1c0dc1f10d9118", + "id": "resp_0604981435bbceda00691e2fcf686c81958723ab0883c9eb44", "object": "response", - "created_at": 1763585206, + "created_at": 1763585999, "status": "completed", "background": false, "billing": { @@ -342,7 +342,7 @@ http_interactions: "model": "gpt-5-2025-08-07", "output": [ { - "id": "msg_08f5a80549ac3c9800691e2cb768e8819bac854261a573efc4", + "id": "msg_0604981435bbceda00691e2fda1d408195b0efaf7fda0a9468", "type": "message", "status": "completed", "content": [ @@ -350,7 +350,7 @@ http_interactions: "type": "output_text", "annotations": [], "logprobs": [], - "text": "- 5 + 3 = 8\n- Weather: I don\u2019t have live weather access here. Tell me the city/ZIP and date (e.g., \u201ctoday in Seattle\u201d), and I can suggest likely conditions based on climate norms or show you how to check quickly." + "text": "5 + 3 = 8.\n\nFor the weather, tell me the location (city and country or ZIP/postcode) and whether you want current conditions or a forecast (and for which date/time)." } ], "role": "assistant" @@ -416,18 +416,18 @@ http_interactions: "top_p": 1.0, "truncation": "disabled", "usage": { - "input_tokens": 819, + "input_tokens": 599, "input_tokens_details": { "cached_tokens": 0 }, - "output_tokens": 61, + "output_tokens": 45, "output_tokens_details": { "reasoning_tokens": 0 }, - "total_tokens": 880 + "total_tokens": 644 }, "user": null, "metadata": {} } - recorded_at: Wed, 19 Nov 2025 20:46:48 GMT + recorded_at: Wed, 19 Nov 2025 21:00:11 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml index e3113440..cedfbec0 100644 --- a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_multiple_servers.yml @@ -47,7 +47,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:47:01 GMT + - Wed, 19 Nov 2025 21:01:17 GMT Content-Type: - application/json Transfer-Encoding: @@ -61,7 +61,7 @@ http_interactions: X-Ratelimit-Remaining-Requests: - '14999' X-Ratelimit-Remaining-Tokens: - - '39999742' + - '39999552' X-Ratelimit-Reset-Requests: - 4ms X-Ratelimit-Reset-Tokens: @@ -73,18 +73,18 @@ http_interactions: Openai-Project: - PROJECT_ID X-Request-Id: - - req_3a6c3c814288492cb4c9bcffda99d81e + - req_2b65c32939164c9093850716443dd580 Openai-Processing-Ms: - - '12146' + - '24230' X-Envoy-Upstream-Service-Time: - - '12150' + - '24234' Cf-Cache-Status: - DYNAMIC Set-Cookie: - - __cf_bm=j2.cfQDVPxlsLwWrqPmnoLgRx6FjOvdhDAdZeNoiwws-1763585221-1.0.1.1-xMHbVgX_aU0ZBtQgMiIqXH.XN1a9SRt24.IvJaTK_G54jLIwVPqeBROke.LVRN2VAjY0eGnsP9daFP_4kSIvr6jIFCR0jOKkDvFXsM7Usgo; - path=/; expires=Wed, 19-Nov-25 21:17:01 GMT; domain=.api.openai.com; HttpOnly; + - __cf_bm=D2INndyqrWkuru74aJIEUTa.1TFXQXyy3nI7QuxUHh8-1763586077-1.0.1.1-yGRUVGCAwJzGHNx6vEi5mNQyj7UeAJRWtvRnvyhoOrk7UxjWBYjkkLvRI6dVskPVHyZmK0tUVzYZCAofGiAOL6952hwQFQCEWkL1uhKVnA4; + path=/; expires=Wed, 19-Nov-25 21:31:17 GMT; domain=.api.openai.com; HttpOnly; Secure; SameSite=None - - _cfuvid=HfhDIJwloIhF0Ex4_rIqS8SBlLrJguf2kwC9GiP7aHE-1763585221220-0.0.1.1-604800000; + - _cfuvid=ROwX.kC._rGh6fKQOAtJJ9UxQcpAxNl.RKSHPlB.1q0-1763586077510-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload @@ -93,16 +93,16 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a128f247e6715e0-SJC + - 9a12a3c09eacba62-SJC Alt-Svc: - h3=":443"; ma=86400 body: encoding: ASCII-8BIT string: |- { - "id": "resp_01142e73b6db117e00691e2cb911f88198b67e43d7dc3c2291", + "id": "resp_06a0f877661f6c2a00691e3005458c819aa2ea27e6a884bc7f", "object": "response", - "created_at": 1763585209, + "created_at": 1763586053, "status": "completed", "background": false, "billing": { @@ -116,7 +116,7 @@ http_interactions: "model": "gpt-5-2025-08-07", "output": [ { - "id": "mcpl_01142e73b6db117e00691e2cb968608198a41622b3b25afc00", + "id": "mcpl_06a0f877661f6c2a00691e30058154819ab3befe700c8577f2", "type": "mcp_list_tools", "server_label": "weather", "tools": [ @@ -134,7 +134,7 @@ http_interactions: ] }, { - "id": "mcpl_01142e73b6db117e00691e2cb968f081989cfd717a72971239", + "id": "mcpl_06a0f877661f6c2a00691e30058210819a96ef93b760a8df01", "type": "mcp_list_tools", "server_label": "github_copilot", "tools": [ @@ -1858,12 +1858,12 @@ http_interactions: ] }, { - "id": "rs_01142e73b6db117e00691e2cbb47308198b2c8f0e0c04ed9e7", + "id": "rs_06a0f877661f6c2a00691e3008481c819aa9138acd0c11ef38", "type": "reasoning", "summary": [] }, { - "id": "msg_01142e73b6db117e00691e2cc3337c8198a4606717b4ac6447", + "id": "msg_06a0f877661f6c2a00691e301b8544819a8564dee53b5a19f3", "type": "message", "status": "completed", "content": [ @@ -1871,7 +1871,7 @@ http_interactions: "type": "output_text", "annotations": [], "logprobs": [], - "text": "Happy to help! To get started, I need a couple details:\n\n- Weather:\n - Which location (city and country or ZIP/postcode)?\n - Do you want current conditions, today\u2019s forecast, or a multi\u2011day forecast?\n\n- Repository information:\n - Which GitHub repository (owner/name or a URL)?\n - What details do you want (e.g., branches, latest commits, open issues/PRs, releases, tags)?" + "text": "I\u2019m happy to do that\u2014just need a couple details:\n\n- Weather: What location (city/country or lat/long)? Do you want current conditions or a forecast, and in Celsius or Fahrenheit?\n- Repository: Which GitHub repo (owner/name or URL)? What specifics do you want (stars/forks, latest commit, open issues/PRs, branches, tags/releases, languages)?\n\nShare those and I\u2019ll fetch the info." } ], "role": "assistant" @@ -1923,16 +1923,16 @@ http_interactions: "usage": { "input_tokens": 5369, "input_tokens_details": { - "cached_tokens": 5248 + "cached_tokens": 0 }, - "output_tokens": 482, + "output_tokens": 672, "output_tokens_details": { - "reasoning_tokens": 384 + "reasoning_tokens": 576 }, - "total_tokens": 5851 + "total_tokens": 6041 }, "user": null, "metadata": {} } - recorded_at: Wed, 19 Nov 2025 20:47:01 GMT + recorded_at: Wed, 19 Nov 2025 21:01:17 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml index c4e4aa8f..e2a98775 100644 --- a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server.yml @@ -47,7 +47,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:46:21 GMT + - Wed, 19 Nov 2025 21:00:53 GMT Content-Type: - application/json Transfer-Encoding: @@ -61,7 +61,7 @@ http_interactions: X-Ratelimit-Remaining-Requests: - '14999' X-Ratelimit-Remaining-Tokens: - - '39999994' + - '39999672' X-Ratelimit-Reset-Requests: - 4ms X-Ratelimit-Reset-Tokens: @@ -73,18 +73,18 @@ http_interactions: Openai-Project: - PROJECT_ID X-Request-Id: - - req_a0c5f02486ed497abb5d00386934bdd8 + - req_3c03d571859d476382d6b37b08df6a27 Openai-Processing-Ms: - - '8667' + - '13185' X-Envoy-Upstream-Service-Time: - - '8670' + - '13695' Cf-Cache-Status: - DYNAMIC Set-Cookie: - - __cf_bm=hEMCBTQW2kbtAW3WZlboz0.6poiAhEfPOmz0KuCKxBA-1763585181-1.0.1.1-6CSuuuouS3h6lNKBeSKHyL02OMvCAYjGjtPdJVJAAIVrsQ8HAk2vSUyOuXNN17MJ1eEXsBSQFV4QuH._fMIpsYxFwn.DZjjH9MjdBYkZb8o; - path=/; expires=Wed, 19-Nov-25 21:16:21 GMT; domain=.api.openai.com; HttpOnly; + - __cf_bm=kflfqHdJ1B8HHdvMcnWGiSw67yQk4AODpq80TmZIXSU-1763586053-1.0.1.1-H1uVqWqBvl.z9DDg33zvzS3PKtETnWw.A8KN0H.pNadRshogV23OcX1YKLmoInBoCtXzxDWy8EHDVBZq1kg6AeqGojBr08itByDBaFvPwUM; + path=/; expires=Wed, 19-Nov-25 21:30:53 GMT; domain=.api.openai.com; HttpOnly; Secure; SameSite=None - - _cfuvid=kWh83s9g4apTauc3btcfmRN7gISFSYbxm6AUx7mq0_c-1763585181322-0.0.1.1-604800000; + - _cfuvid=AMJuJxcHyxnJNR5ESnrTVjjFtWH5jwUltQDgI2m9PzM-1763586053133-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload @@ -93,16 +93,16 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a128e40ad027aaf-SJC + - 9a12a36a0f2dfb4c-SJC Alt-Svc: - h3=":443"; ma=86400 body: encoding: ASCII-8BIT string: |- { - "id": "resp_0a61b801b444d08000691e2c94a29c8197912046d330286604", + "id": "resp_0c6ed4b8ea47a17200691e2ff805e08194a3b62232cbe32246", "object": "response", - "created_at": 1763585172, + "created_at": 1763586040, "status": "completed", "background": false, "billing": { @@ -116,7 +116,7 @@ http_interactions: "model": "gpt-5-2025-08-07", "output": [ { - "id": "mcpl_0a61b801b444d08000691e2c94e7e48197b1c74f8cfdb06d07", + "id": "mcpl_0c6ed4b8ea47a17200691e2ff8518081949e17ad461a2dc4f7", "type": "mcp_list_tools", "server_label": "weather", "tools": [ @@ -134,12 +134,12 @@ http_interactions: ] }, { - "id": "rs_0a61b801b444d08000691e2c9698588197827178deb303ddb2", + "id": "rs_0c6ed4b8ea47a17200691e2ff9ec648194b438f387e293eaa3", "type": "reasoning", "summary": [] }, { - "id": "msg_0a61b801b444d08000691e2c9c38688197ae98d8ca7b672d05", + "id": "msg_0c6ed4b8ea47a17200691e30044d208194ba722ed82bf17365", "type": "message", "status": "completed", "content": [ @@ -147,7 +147,7 @@ http_interactions: "type": "output_text", "annotations": [], "logprobs": [], - "text": "Sure\u2014what location should I use? Please share a city and country (or ZIP/postcode or coordinates), and your preferred units (C or F)." + "text": "Sure\u2014what location do you want the current weather for? Please share a city and country (or ZIP/postcode), and your preferred units (Celsius or Fahrenheit)." } ], "role": "assistant" @@ -191,14 +191,14 @@ http_interactions: "input_tokens_details": { "cached_tokens": 0 }, - "output_tokens": 229, + "output_tokens": 552, "output_tokens_details": { - "reasoning_tokens": 192 + "reasoning_tokens": 512 }, - "total_tokens": 371 + "total_tokens": 694 }, "user": null, "metadata": {} } - recorded_at: Wed, 19 Nov 2025 20:46:21 GMT + recorded_at: Wed, 19 Nov 2025 21:00:53 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml index 28b1a402..d079b251 100644 --- a/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/common_format/mcp_test/test_agent_common_format_single_server_with_auth.yml @@ -47,7 +47,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:46:12 GMT + - Wed, 19 Nov 2025 21:00:39 GMT Content-Type: - application/json Transfer-Encoding: @@ -61,7 +61,7 @@ http_interactions: X-Ratelimit-Remaining-Requests: - '14999' X-Ratelimit-Remaining-Tokens: - - '39999552' + - '39999374' X-Ratelimit-Reset-Requests: - 4ms X-Ratelimit-Reset-Tokens: @@ -73,18 +73,18 @@ http_interactions: Openai-Project: - PROJECT_ID X-Request-Id: - - req_725ba7e71fda420a94b554e9145875e4 + - req_4fe1b42247f54f5f8d08f449080c2496 Openai-Processing-Ms: - - '18277' + - '27694' X-Envoy-Upstream-Service-Time: - - '18280' + - '27697' Cf-Cache-Status: - DYNAMIC Set-Cookie: - - __cf_bm=9Y5IzrkvtruiIpBJPjA_YtnMLViW09gnhXpCFU5MIr4-1763585172-1.0.1.1-2BtyiLqLt5Tk7jGO03TuGBSmxSEf9zkw7ePAewveycnA4zEyh1aeBbIl2repFN4fTRTSnVIx4O_4R9qCbIsypuWejk9VM9Z.hxB5QueNYsE; - path=/; expires=Wed, 19-Nov-25 21:16:12 GMT; domain=.api.openai.com; HttpOnly; + - __cf_bm=u_ixvNBnE_uGRZuW0BFA_8daa2YtGUUBZM67aDfW8uc-1763586039-1.0.1.1-O72bWYpsWBa.MQ7HvDBvHuK8fz0YIrTkhtSxt87JqzBfbt0LzJFK_hsNznbqDyJ3s74cJOowjnENWg7CcGFlfqSWMNubFfDBB7JROc67IVc; + path=/; expires=Wed, 19-Nov-25 21:30:39 GMT; domain=.api.openai.com; HttpOnly; Secure; SameSite=None - - _cfuvid=XS4xf6HTx0IcH9mL6bPrTEpDK9hIAo3tN57JX3FLGBk-1763585172364-0.0.1.1-604800000; + - _cfuvid=2yzg5PAukBYR91zDzzjZzZdvTx4n341kXd55hyf_nsQ-1763586039191-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload @@ -93,16 +93,16 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a128dcc7dd0fa62-SJC + - 9a12a2bac9e4270a-SJC Alt-Svc: - h3=":443"; ma=86400 body: encoding: ASCII-8BIT string: |- { - "id": "resp_0bbb974cbbc378f500691e2c8216c0819891982946cb6cf0ad", + "id": "resp_0a1a94ff5963180600691e2fdb7c308195be8226e81298424c", "object": "response", - "created_at": 1763585154, + "created_at": 1763586011, "status": "completed", "background": false, "billing": { @@ -116,7 +116,7 @@ http_interactions: "model": "gpt-5-2025-08-07", "output": [ { - "id": "mcpl_0bbb974cbbc378f500691e2c82402c8198bdf001518e0d9a7e", + "id": "mcpl_0a1a94ff5963180600691e2fdbcd648195bf2aa0f4554487ee", "type": "mcp_list_tools", "server_label": "github_copilot", "tools": [ @@ -1840,12 +1840,12 @@ http_interactions: ] }, { - "id": "rs_0bbb974cbbc378f500691e2c841370819897cf8a919e277e6a", + "id": "rs_0a1a94ff5963180600691e2fddabbc81959c66ce6978b890ef", "type": "reasoning", "summary": [] }, { - "id": "msg_0bbb974cbbc378f500691e2c925fe4819892ab87007a8a0f3a", + "id": "msg_0a1a94ff5963180600691e2ff3904c81958f84eb49da0196a1", "type": "message", "status": "completed", "content": [ @@ -1853,7 +1853,7 @@ http_interactions: "type": "output_text", "annotations": [], "logprobs": [], - "text": "Which repository do you want info for? Please provide the owner and repo name (e.g., owner/repo) or paste the GitHub URL.\n\nAlso, what details would you like? I can fetch things like:\n- Description, license, topics, default branch\n- Stars/forks/watchers\n- Latest release/tags\n- Branches and recent commits\n- Open issues/PRs and their statuses\n- Languages and file structure" + "text": "Sure\u2014what repository do you have in mind? Please provide either:\n- The GitHub URL, or\n- owner/repo (for example: vercel/next.js)\n\nAlso, what details would you like? I can fetch any of the following:\n- Overview: description, topics, license, default branch, stars/forks/watchers, last updated\n- Activity: recent commits, contributors\n- Code: branches, tags, languages, README\n- Releases: latest release, changelog\n- Work: open issues/PRs (counts or top items), labels, milestones\n\nIf you\u2019re not sure, I can pull a quick overview by default once you share the repo." } ], "role": "assistant" @@ -1896,16 +1896,16 @@ http_interactions: "usage": { "input_tokens": 5312, "input_tokens_details": { - "cached_tokens": 5248 + "cached_tokens": 0 }, - "output_tokens": 671, + "output_tokens": 849, "output_tokens_details": { - "reasoning_tokens": 576 + "reasoning_tokens": 704 }, - "total_tokens": 5983 + "total_tokens": 6161 }, "user": null, "metadata": {} } - recorded_at: Wed, 19 Nov 2025 20:46:12 GMT + recorded_at: Wed, 19 Nov 2025 21:00:39 GMT recorded_with: VCR 6.3.1 diff --git a/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml b/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml index 088e263d..5d5c7288 100644 --- a/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml +++ b/test/fixtures/vcr_cassettes/integration/open_ai/responses/native_format_test/test_agent_mcp_server.yml @@ -47,7 +47,7 @@ http_interactions: message: OK headers: Date: - - Wed, 19 Nov 2025 20:50:30 GMT + - Wed, 19 Nov 2025 21:01:31 GMT Content-Type: - application/json Transfer-Encoding: @@ -73,18 +73,18 @@ http_interactions: Openai-Project: - PROJECT_ID X-Request-Id: - - req_f930de4b1fef42efbd671c8f50d718d5 + - req_9295f14d4dd84dc099101d59970868b8 Openai-Processing-Ms: - - '6126' + - '4649' X-Envoy-Upstream-Service-Time: - - '6130' + - '4652' Cf-Cache-Status: - DYNAMIC Set-Cookie: - - __cf_bm=CSKnilMCYDGbIJen9mOqPAZ7DhPigg8Pt9lbuzpb4C0-1763585430-1.0.1.1-elHDnP58qXxQEsSrnvucnenaNLCDZx7DPQ.RgpWQl00bHT8O.GMC2dbL7nYk0AcuEUiE0o3B_2_3Fw2X1jP5rCtTjCg9XwrOBh9NALSeSiM; - path=/; expires=Wed, 19-Nov-25 21:20:30 GMT; domain=.api.openai.com; HttpOnly; + - __cf_bm=t6Md2Z8ktst2TLkho7mrzHR.qTEsSKKfrt9BW1zmbyo-1763586091-1.0.1.1-vxuhqec9qHLE6XEsGrnq1uWrMmmrlkW7D_oK9rL8dm9hHrlehfKYXHaiLKu9lK2WcXN1hbUsQNmNeyJT9mNBjCU289KxVecnenZX5YwdTL4; + path=/; expires=Wed, 19-Nov-25 21:31:31 GMT; domain=.api.openai.com; HttpOnly; Secure; SameSite=None - - _cfuvid=UCeoHiVlR_TvsF53s39eQ1irF_vsGIxk5sC3B_i1KBM-1763585430500-0.0.1.1-604800000; + - _cfuvid=23UO7O0FsnZCexgZIpX9RBhJbaDwQqPEq9WBoSNKWD8-1763586091570-0.0.1.1-604800000; path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None Strict-Transport-Security: - max-age=31536000; includeSubDomains; preload @@ -93,16 +93,16 @@ http_interactions: Server: - cloudflare Cf-Ray: - - 9a1294661dd3eb25-SJC + - 9a12a492d923cf2e-SJC Alt-Svc: - h3=":443"; ma=86400 body: encoding: ASCII-8BIT string: |- { - "id": "resp_03973314476bd0d900691e2d905f88819bb815ddd4fd9ab654", + "id": "resp_0a7f52788c12648f00691e3026e910819aa4466498f7e2f09a", "object": "response", - "created_at": 1763585424, + "created_at": 1763586087, "status": "completed", "background": false, "billing": { @@ -116,7 +116,7 @@ http_interactions: "model": "gpt-4.1-2025-04-14", "output": [ { - "id": "mcpl_03973314476bd0d900691e2d90b224819b91340ffbd1a64e78", + "id": "mcpl_0a7f52788c12648f00691e3027380c819a8fa6e670b6533e37", "type": "mcp_list_tools", "server_label": "cloudflare-demo", "tools": [ @@ -134,7 +134,7 @@ http_interactions: ] }, { - "id": "msg_03973314476bd0d900691e2d922164819b80e5e14143fd8a75", + "id": "msg_0a7f52788c12648f00691e30289804819ab5994045994e21da", "type": "message", "status": "completed", "content": [ @@ -142,7 +142,7 @@ http_interactions: "type": "output_text", "annotations": [], "logprobs": [], - "text": "Here are the tools I currently have available:\n\n1. **Image Input Capabilities:** I can analyze and interpret images you upload.\n2. **Web Browsing (not available right now):** Sometimes, I have the ability to access the web for real-time information, but it's currently disabled.\n3. **Plugin for Cloudflare's MCP Demo Day:** \n - I can fetch information about Cloudflare\u2019s Managed Components Platform (MCP) Demo Day using a dedicated plugin.\n\nIf you need information from these areas or want to try the MCP Demo Day plugin, let me know!" + "text": "Here are the tools I have available right now:\n\n1. **Image Input Capabilities:** I can analyze and interpret images you upload.\n2. **Cloudflare MCP Demo Day Information:** I have access to a tool (`mcp_cloudflare-demo`) that allows me to fetch information specifically related to Cloudflare\u2019s MCP (Magic Cloud Platform) Demo Day.\n\nIf you have a specific question or need information using one of these tools, just let me know!" } ], "role": "assistant" @@ -186,14 +186,14 @@ http_interactions: "input_tokens_details": { "cached_tokens": 0 }, - "output_tokens": 119, + "output_tokens": 93, "output_tokens_details": { "reasoning_tokens": 0 }, - "total_tokens": 194 + "total_tokens": 168 }, "user": null, "metadata": {} } - recorded_at: Wed, 19 Nov 2025 20:50:30 GMT + recorded_at: Wed, 19 Nov 2025 21:01:31 GMT recorded_with: VCR 6.3.1 diff --git a/test/integration/open_ai/responses/common_format/mcp_test.rb b/test/integration/open_ai/responses/common_format/mcp_test.rb index ad6d4cb6..84b4f762 100644 --- a/test/integration/open_ai/responses/common_format/mcp_test.rb +++ b/test/integration/open_ai/responses/common_format/mcp_test.rb @@ -27,7 +27,7 @@ class TestAgent < ActiveAgent::Base def common_format_single_server prompt( input: "Get the current weather", - mcp_servers: [ + mcps: [ { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" } ] ) @@ -49,7 +49,7 @@ def common_format_single_server def common_format_single_server_with_auth prompt( input: "Get repository information", - mcp_servers: [ + mcps: [ { name: "github_copilot", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } ] ) @@ -76,7 +76,7 @@ def common_format_single_server_with_auth def common_format_multiple_servers prompt( input: "Get the weather and repository information", - mcp_servers: [ + mcps: [ { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" }, { name: "github_copilot", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } ] @@ -125,7 +125,7 @@ def common_format_mixed_tools_and_mcp } } ], - mcp_servers: [ + mcps: [ { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" } ] ) From 700132ab083ca8873625d617a8be715a9b2f86e4 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 13:33:28 -0800 Subject: [PATCH 14/17] Fix bug with native Anthropic MCPs --- .../providers/anthropic/transforms.rb | 5 ++- test/providers/anthropic/transforms_test.rb | 42 ++++++++++++++++++- 2 files changed, 44 insertions(+), 3 deletions(-) diff --git a/lib/active_agent/providers/anthropic/transforms.rb b/lib/active_agent/providers/anthropic/transforms.rb index a2b2aead..747ed2e0 100644 --- a/lib/active_agent/providers/anthropic/transforms.rb +++ b/lib/active_agent/providers/anthropic/transforms.rb @@ -81,8 +81,9 @@ def normalize_mcp_servers(mcp_servers) mcp_servers.map do |server| server_hash = server.is_a?(Hash) ? server.deep_symbolize_keys : server - # If already in Anthropic format (has type: "url" and authorization_token), return as-is - if server_hash[:type] == "url" && !server_hash[:authorization] + # If already in Anthropic native format (has type: "url"), return as-is + # Check for absence of common format 'authorization' field OR presence of native 'authorization_token' + if server_hash[:type] == "url" && (server_hash[:authorization_token] || !server_hash[:authorization]) next server_hash end diff --git a/test/providers/anthropic/transforms_test.rb b/test/providers/anthropic/transforms_test.rb index 737127fd..3c223a7b 100644 --- a/test/providers/anthropic/transforms_test.rb +++ b/test/providers/anthropic/transforms_test.rb @@ -587,7 +587,7 @@ def transforms assert_nil result[0][:authorization_token] end - test "normalize_mcp_servers preserves Anthropic format" do + test "normalize_mcp_servers preserves Anthropic format without auth" do mcp_servers = [ { type: "url", @@ -601,6 +601,46 @@ def transforms assert_equal 1, result.size assert_equal "url", result[0][:type] assert_equal "stripe", result[0][:name] + assert_equal "https://mcp.stripe.com", result[0][:url] + end + + test "normalize_mcp_servers preserves Anthropic format with authorization_token" do + mcp_servers = [ + { + type: "url", + name: "stripe", + url: "https://mcp.stripe.com", + authorization_token: "sk_test_123" + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal 1, result.size + assert_equal "url", result[0][:type] + assert_equal "stripe", result[0][:name] + assert_equal "https://mcp.stripe.com", result[0][:url] + assert_equal "sk_test_123", result[0][:authorization_token] + end + + test "normalize_mcp_servers converts common format with authorization to native" do + mcp_servers = [ + { + type: "url", + name: "test", + url: "https://test.com", + authorization: "token123" # Common format field, should be converted + } + ] + + result = transforms.normalize_mcp_servers(mcp_servers) + + assert_equal 1, result.size + assert_equal "url", result[0][:type] + assert_equal "test", result[0][:name] + assert_equal "https://test.com", result[0][:url] + assert_equal "token123", result[0][:authorization_token] + assert_nil result[0][:authorization] # Should not have common format field end test "normalize_mcp_servers handles multiple servers" do From 7255037d348030806b1ba40a8d9367af65923a7b Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Wed, 19 Nov 2025 13:44:58 -0800 Subject: [PATCH 15/17] Update CHANGELOG for MCPs --- CHANGELOG.md | 82 +++++++++++++++++-------- test/docs/actions/mcps_examples_test.rb | 8 +-- 2 files changed, 62 insertions(+), 28 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 71d7740a..a7bd267a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,64 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). +## [Unreleased] + +### Added + +**Universal Tools Format** +```ruby +# Single format works across all providers (Anthropic, OpenAI, OpenRouter, Ollama, Mock) +tools: [{ + name: "get_weather", + description: "Get current weather", + parameters: { + type: "object", + properties: { + location: { type: "string", description: "City and state" } + }, + required: ["location"] + } +}] + +# Tool choice normalization +tool_choice: "auto" # Let model decide +tool_choice: "required" # Force tool use +tool_choice: { name: "get_weather" } # Force specific tool +``` + +Automatic conversion to provider-specific formats. Old formats still work (backward compatible). + +**Model Context Protocol (MCP) Support** +```ruby +# Universal MCP format works across providers (Anthropic, OpenAI) +class MyAgent < ActiveAgent::Base + generate_with :anthropic, model: "claude-haiku-4-5" + + def research + prompt( + message: "Research AI developments", + mcps: [{ + name: "github", + url: "https://api.githubcopilot.com/mcp/", + authorization: ENV["GITHUB_MCP_TOKEN"] + }] + ) + end +end +``` + +- Common format: `{name: "server", url: "https://...", authorization: "token"}` +- Auto-converts to provider native formats +- Anthropic: Beta API support, up to 20 servers per request +- OpenAI: Responses API with pre-built connectors (Dropbox, Google Drive, etc.) +- Backwards compatible: accepts both `mcps` and `mcp_servers` parameters +- Comprehensive documentation with tested examples +- Full VCR test coverage with real MCP endpoints + +### Changed + +- Shared `ToolChoiceClearing` concern eliminates duplication across providers + ## [1.0.0] - 2025-11-21 Major refactor with breaking changes. Complete provider rewrite. New modular architecture. @@ -111,29 +169,6 @@ Template paths: ### Added -**Universal Tools Format** -```ruby -# Single format works across all providers (Anthropic, OpenAI, OpenRouter, Ollama, Mock) -tools: [{ - name: "get_weather", - description: "Get current weather", - parameters: { - type: "object", - properties: { - location: { type: "string", description: "City and state" } - }, - required: ["location"] - } -}] - -# Tool choice normalization -tool_choice: "auto" # Let model decide -tool_choice: "required" # Force tool use -tool_choice: { name: "get_weather" } # Force specific tool -``` - -Automatic conversion to provider-specific formats. Old formats still work (backward compatible). - **Mock Provider for Testing** ```ruby class MyAgent < ActiveAgent::Base @@ -219,7 +254,6 @@ response.usage.service_tier # Anthropic - Retry logic moved to provider SDKs (automatic exponential backoff) - Migrated to official SDKs: `openai` gem and `anthropic` gem - Type-safe options with per-provider definitions -- Shared `ToolChoiceClearing` concern eliminates duplication across providers **Configuration** - Options configurable at class level, instance level, or per-call diff --git a/test/docs/actions/mcps_examples_test.rb b/test/docs/actions/mcps_examples_test.rb index 2a4d959b..fd49ea7b 100644 --- a/test/docs/actions/mcps_examples_test.rb +++ b/test/docs/actions/mcps_examples_test.rb @@ -10,7 +10,7 @@ class WeatherAgent < ActiveAgent::Base def forecast prompt( - message: "What's the weather like?", + "What's the weather like?", mcps: [ { name: "weather", url: "https://demo-day.mcp.cloudflare.com/sse" } ] ) end @@ -35,7 +35,7 @@ class DataAgent < ActiveAgent::Base def analyze prompt( - message: "Analyze the latest data", + "Analyze the latest data", mcps: [ { name: "cloudflare-demo", url: "https://demo-day.mcp.cloudflare.com/sse" } ] ) end @@ -60,7 +60,7 @@ class IntegratedAgent < ActiveAgent::Base def research prompt( - input: "Research the latest AI developments", + "Research the latest AI developments", mcps: [ { name: "cloudflare", url: "https://demo-day.mcp.cloudflare.com/sse" }, { name: "github", url: "https://api.githubcopilot.com/mcp/", authorization: ENV["GITHUB_MCP_TOKEN"] } @@ -88,7 +88,7 @@ class HybridAgent < ActiveAgent::Base def analyze_data prompt( - input: "Calculate and fetch data", + "Calculate and fetch data", tools: [ { name: "calculate", description: "Perform calculations", From 45c36bf539c06c41643fc6dc6a31819528867759 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Fri, 21 Nov 2025 18:10:01 -0800 Subject: [PATCH 16/17] Fix issues extracting the correct message content from Anthropic MCP usage when in json_object mode --- .../providers/anthropic_provider.rb | 25 +- .../providers/common/messages/_types.rb | 67 ++-- .../providers/common/messages/assistant.rb | 24 +- test/providers/common/messages/types_test.rb | 331 ++++++++++++++++++ 4 files changed, 407 insertions(+), 40 deletions(-) create mode 100644 test/providers/common/messages/types_test.rb diff --git a/lib/active_agent/providers/anthropic_provider.rb b/lib/active_agent/providers/anthropic_provider.rb index 1bdb0009..3e193104 100644 --- a/lib/active_agent/providers/anthropic_provider.rb +++ b/lib/active_agent/providers/anthropic_provider.rb @@ -17,6 +17,9 @@ module Providers # # @see BaseProvider class AnthropicProvider < BaseProvider + # Lead-in message for JSON response format emulation + JSON_RESPONSE_FORMAT_LEAD_IN = "Here is the JSON requested:\n{" + # @todo Add support for Anthropic::BedrockClient and Anthropic::VertexClient # @return [Anthropic::Client] def client @@ -72,7 +75,7 @@ def prepare_prompt_request_response_format self.message_stack.push({ role: "assistant", - content: "Here is the JSON requested:\n{" + content: JSON_RESPONSE_FORMAT_LEAD_IN }) end @@ -227,7 +230,7 @@ def process_prompt_finished(api_response = nil) # # Handles JSON response format simulation by prepending `{` to the response - # content after removing the assistant lead-in message. + # content if the last message in the request is the JSON lead-in prompt. # # @see BaseProvider#process_prompt_finished_extract_messages # @param api_response [Hash] converted response hash @@ -235,10 +238,20 @@ def process_prompt_finished(api_response = nil) def process_prompt_finished_extract_messages(api_response) return unless api_response - # Handle JSON response format simulation - if request.response_format&.dig(:type) == "json_object" - request.pop_message! - api_response[:content][0][:text] = "{#{api_response[:content][0][:text]}" + # Get the last message (may be either Hash or gem object) + last_message = request.messages.last + last_role = last_message.is_a?(Hash) ? last_message[:role] : last_message&.role + last_content = last_message.is_a?(Hash) ? last_message[:content] : last_message&.content + + # Check if the last message in request is the JSON lead-in prompt + if last_role.to_sym == :assistant && last_content == JSON_RESPONSE_FORMAT_LEAD_IN + # Remove the lead-in message from the request + request.messages.pop + + # Prepend "{" to the response's first content text + if api_response[:content]&.first&.dig(:text) + api_response[:content][0][:text] = "{#{api_response[:content][0][:text]}" + end end [ api_response ] diff --git a/lib/active_agent/providers/common/messages/_types.rb b/lib/active_agent/providers/common/messages/_types.rb index 724fdde4..5c93623a 100644 --- a/lib/active_agent/providers/common/messages/_types.rb +++ b/lib/active_agent/providers/common/messages/_types.rb @@ -51,12 +51,6 @@ def cast_message(value) when "assistant" # Filter to only known attributes for Assistant filtered_hash = hash.slice(:role, :content, :name) - - # Compress content array to string if needed (Anthropic format) - if filtered_hash[:content].is_a?(Array) - filtered_hash[:content] = compress_content_array(filtered_hash[:content]) - end - Common::Messages::Assistant.new(**filtered_hash) when "tool" # Filter to only known attributes for Tool @@ -94,29 +88,6 @@ def serialize_message(value) raise ArgumentError, "Cannot serialize #{value.class}" end end - - # Compresses Anthropic-style content array into a string. - # - # Anthropic messages can have content as an array of blocks like: - # [{type: "text", text: "..."}, {type: "tool_use", ...}] - # This extracts and joins text blocks into a single string. - # - # @param content_array [Array] - # @return [String] - def compress_content_array(content_array) - content_array.map do |block| - case block[:type]&.to_s - when "text" - block[:text] - when "tool_use" - # Tool use blocks don't have readable text content - nil - else - # Unknown block type, try to extract text if present - block[:text] - end - end.compact.join("\n") - end end # Type for Messages array @@ -124,7 +95,9 @@ class MessagesType < ActiveModel::Type::Value def cast(value) case value when Array - value.map { |v| message_type.cast(v) }.compact + messages = value.map { |v| message_type.cast(v) }.compact + # Split messages with array content into separate messages + messages.flat_map { |msg| split_content_blocks(msg) } when nil [] else @@ -152,6 +125,40 @@ def deserialize(value) def message_type @message_type ||= MessageType.new end + + # Splits an assistant message with array content into separate messages + # for each content block. + # + # @param message [Common::Messages::Base] + # @return [Array] + def split_content_blocks(message) + # Only split assistant messages with array content + return [ message ] unless message.is_a?(Common::Messages::Assistant) && message.content.is_a?(Array) + + message.content.map do |block| + case block[:type]&.to_s + when "text" + # Create a message for text blocks + Common::Messages::Assistant.new(role: "assistant", content: block[:text], name: message.name) + when "tool_use" + # Create a message with tool use info as string representation + tool_info = "[Tool Use: #{block[:name]}]\nID: #{block[:id]}\nInput: #{JSON.pretty_generate(block[:input])}" + Common::Messages::Assistant.new(role: "assistant", content: tool_info, name: message.name) + when "mcp_tool_use" + # Create a message with MCP tool use info + tool_info = "[MCP Tool Use: #{block[:name]}]\nID: #{block[:id]}\nServer: #{block[:server_name]}\nInput: #{JSON.pretty_generate(block[:input] || {})}" + Common::Messages::Assistant.new(role: "assistant", content: tool_info, name: message.name) + when "mcp_tool_result" + # Create a message with MCP tool result + result_info = "[MCP Tool Result]\n#{block[:content]}" + Common::Messages::Assistant.new(role: "assistant", content: result_info, name: message.name) + else + # For unknown block types, try to extract text + content = block[:text] || block.to_s + Common::Messages::Assistant.new(role: "assistant", content:, name: message.name) + end + end.compact + end end end end diff --git a/lib/active_agent/providers/common/messages/assistant.rb b/lib/active_agent/providers/common/messages/assistant.rb index e09aa422..2099b182 100644 --- a/lib/active_agent/providers/common/messages/assistant.rb +++ b/lib/active_agent/providers/common/messages/assistant.rb @@ -9,7 +9,7 @@ module Messages # Represents messages sent by the AI assistant in a conversation. class Assistant < Base attribute :role, :string, as: "assistant" - attribute :content, :string + attribute :content # Accept both string and array (provider-native formats) attribute :name, :string validates :content, presence: true @@ -24,9 +24,16 @@ class Assistant < Base # @param normalize_names [Symbol, nil] key normalization method (e.g., :underscore) # @return [Hash, Array, nil] parsed JSON structure or nil if parsing fails def parsed_json(symbolize_names: true, normalize_names: :underscore) - start_char = [ content.index("{"), content.index("[") ].compact.min - end_char = [ content.rindex("}"), content.rindex("]") ].compact.max - content_stripped = content[start_char..end_char] if start_char && end_char + # Handle array content (from content blocks) by searching through each block + content_str = if content.is_a?(Array) + content.map { |block| block.is_a?(Hash) ? block[:text] : block.to_s }.join("\n") + else + content.to_s + end + + start_char = [ content_str.index("{"), content_str.index("[") ].compact.min + end_char = [ content_str.rindex("}"), content_str.rindex("]") ].compact.max + content_stripped = content_str[start_char..end_char] if start_char && end_char return unless content_stripped content_parsed = JSON.parse(content_stripped) @@ -48,6 +55,15 @@ def parsed_json(symbolize_names: true, normalize_names: :underscore) nil end + # Returns content as a string, handling both string and array formats + def text + if content.is_a?(Array) + content.map { |block| block.is_a?(Hash) ? block[:text] : block.to_s }.join("\n") + else + content.to_s + end + end + alias_method :json_object, :parsed_json alias_method :parse_json, :parsed_json end diff --git a/test/providers/common/messages/types_test.rb b/test/providers/common/messages/types_test.rb new file mode 100644 index 00000000..495b2b6f --- /dev/null +++ b/test/providers/common/messages/types_test.rb @@ -0,0 +1,331 @@ +# frozen_string_literal: true + +require "test_helper" +require "active_agent/providers/common/messages/_types" + +module ActiveAgent + module Providers + module Common + module Messages + class TypesTest < ActiveSupport::TestCase + test "MessageType casts strings and hashes to appropriate message types" do + message_type = create_message_type + + # String → User message + user_result = message_type.cast("Hello") + assert_instance_of ActiveAgent::Providers::Common::Messages::User, user_result + assert_equal "Hello", user_result.content + + # Hash with user role → User message + user_hash = message_type.cast({ role: "user", content: "Hi" }) + assert_instance_of ActiveAgent::Providers::Common::Messages::User, user_hash + + # Hash with assistant role → Assistant message + assistant_result = message_type.cast({ role: "assistant", content: "Hello" }) + assert_instance_of ActiveAgent::Providers::Common::Messages::Assistant, assistant_result + end + + test "MessageType drops system messages" do + message_type = create_message_type + result = message_type.cast({ role: "system", content: "System prompt" }) + assert_nil result + end + + test "MessagesType casts array of Hash messages to Message objects" do + messages_type = create_messages_type + messages = [ + { role: "user", content: "Hi" }, + { role: "assistant", content: "Hello" } + ] + result = messages_type.cast(messages) + + assert_equal 2, result.length + assert_instance_of ActiveAgent::Providers::Common::Messages::User, result[0] + assert_instance_of ActiveAgent::Providers::Common::Messages::Assistant, result[1] + end + + test "MessagesType casts nil to empty array" do + messages_type = create_messages_type + result = messages_type.cast(nil) + + assert_equal [], result + end + + test "MessagesType splits assistant message with array content into separate messages" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + content: [ + { type: "text", text: "Hello" }, + { type: "text", text: "World" } + ] + } + ] + result = messages_type.cast(messages) + + assert_equal 2, result.length + assert_all_instances_of(result, ActiveAgent::Providers::Common::Messages::Assistant) + assert_equal "Hello", result[0].content + assert_equal "World", result[1].content + end + + test "MessagesType splits tool_use blocks into separate messages" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + content: [ + { type: "text", text: "I'll help with that" }, + { + type: "tool_use", + id: "tool_123", + name: "search", + input: { query: "test" } + } + ] + } + ] + result = messages_type.cast(messages) + + assert_equal 2, result.length + assert_all_instances_of(result, ActiveAgent::Providers::Common::Messages::Assistant) + + # First message is text + assert_equal "I'll help with that", result[0].content + + # Second message contains tool info + assert_includes result[1].content, "[Tool Use: search]" + assert_includes result[1].content, "ID: tool_123" + assert_includes result[1].content, "Input:" + end + + test "MessagesType splits mcp_tool_use blocks into separate messages" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + content: [ + { type: "text", text: "Using MCP" }, + { + type: "mcp_tool_use", + id: "mcp_123", + name: "get_file", + server_name: "file_server", + input: { path: "/home/user/file.txt" } + } + ] + } + ] + result = messages_type.cast(messages) + + assert_equal 2, result.length + assert_all_instances_of(result, ActiveAgent::Providers::Common::Messages::Assistant) + + # MCP tool message + mcp_message = result[1].content + assert_includes mcp_message, "[MCP Tool Use: get_file]" + assert_includes mcp_message, "ID: mcp_123" + assert_includes mcp_message, "Server: file_server" + assert_includes mcp_message, "Input:" + end + + test "MessagesType splits mcp_tool_result blocks into separate messages" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + content: [ + { + type: "mcp_tool_result", + id: "result_123", + name: "get_file", + content: "File contents here" + } + ] + } + ] + result = messages_type.cast(messages) + + assert_equal 1, result.length + result_message = result[0].content + assert_includes result_message, "[MCP Tool Result]" + assert_includes result_message, "File contents here" + end + + test "MessagesType handles mixed content types in single message" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + content: [ + { type: "text", text: "Text 1" }, + { + type: "tool_use", + id: "tool_1", + name: "search", + input: { q: "test" } + }, + { type: "text", text: "Text 2" } + ] + } + ] + result = messages_type.cast(messages) + + assert_equal 3, result.length + assert_all_instances_of(result, ActiveAgent::Providers::Common::Messages::Assistant) + assert_equal "Text 1", result[0].content + assert_includes result[1].content, "[Tool Use: search]" + assert_equal "Text 2", result[2].content + end + + test "MessagesType handles empty and nil inputs in tool blocks" do + messages_type = create_messages_type + + # Empty input in tool_use block + empty_input_result = messages_type.cast([ + { + role: "assistant", + content: [ + { + type: "tool_use", + id: "tool_1", + name: "get_time", + input: {} + } + ] + } + ]) + assert_equal 1, empty_input_result.length + assert_includes empty_input_result[0].content, "Input: {}" + + # Nil input in mcp_tool_use block + nil_input_result = messages_type.cast([ + { + role: "assistant", + content: [ + { + type: "mcp_tool_use", + id: "mcp_1", + name: "ping", + server_name: "server", + input: nil + } + ] + } + ]) + assert_equal 1, nil_input_result.length + assert_includes nil_input_result[0].content, "Input: {}" + end + + test "MessagesType preserves message name through split" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + name: "gpt-4", + content: [ + { type: "text", text: "Text 1" }, + { type: "text", text: "Text 2" } + ] + } + ] + result = messages_type.cast(messages) + + assert_equal 2, result.length + result.each { |msg| assert_equal "gpt-4", msg.name } + end + + test "MessagesType does not split non-assistant messages with array content" do + messages_type = create_messages_type + messages = [ + { + role: "user", + content: [ { type: "text", text: "User message" } ] + } + ] + result = messages_type.cast(messages) + + assert_equal 1, result.length + assert_instance_of ActiveAgent::Providers::Common::Messages::User, result[0] + end + + test "MessagesType does not split assistant messages with string content" do + messages_type = create_messages_type + messages = [ + { + role: "assistant", + content: "Simple string content" + } + ] + result = messages_type.cast(messages) + + assert_equal 1, result.length + assert_instance_of ActiveAgent::Providers::Common::Messages::Assistant, result[0] + assert_equal "Simple string content", result[0].content + end + + test "MessagesType compacts nil messages from system roles" do + messages_type = create_messages_type + messages = [ + { role: "system", content: "System prompt" }, + { role: "user", content: "User message" }, + { role: "system", content: "Another system prompt" }, + { role: "assistant", content: "Assistant response" } + ] + result = messages_type.cast(messages) + + assert_equal 2, result.length + assert_instance_of ActiveAgent::Providers::Common::Messages::User, result[0] + assert_instance_of ActiveAgent::Providers::Common::Messages::Assistant, result[1] + end + + test "Assistant message parsed_json handles array content" do + content_array = [ + { type: "text", text: '{"name": "John", "age": 30}' }, + { type: "text", text: "Some other text" } + ] + assistant_message = ActiveAgent::Providers::Common::Messages::Assistant.new(content: content_array) + + result = assistant_message.parsed_json + + assert_not_nil result + assert_equal "John", result[:name] + assert_equal 30, result[:age] + end + + test "Assistant message text method handles string and array content" do + # String content + string_message = ActiveAgent::Providers::Common::Messages::Assistant.new(content: "Hello World") + assert_equal "Hello World", string_message.text + + # Array content + array_message = ActiveAgent::Providers::Common::Messages::Assistant.new( + content: [ + { type: "text", text: "Hello" }, + { type: "text", text: "World" } + ] + ) + assert_equal "Hello\nWorld", array_message.text + end + + private + + def create_message_type + # Access MessageType through the Types module + ActiveAgent::Providers::Common::Messages::Types::MessageType.new + end + + def create_messages_type + # Access MessagesType through the Types module + ActiveAgent::Providers::Common::Messages::Types::MessagesType.new + end + + def assert_all_instances_of(array, klass) + array.each { |item| assert_instance_of klass, item } + end + end + end + end + end +end From 0b9d681613f9abd9aee6110f18b7995026489b45 Mon Sep 17 00:00:00 2001 From: Zane Wolfgang Pickett Date: Fri, 21 Nov 2025 20:16:40 -0800 Subject: [PATCH 17/17] Automatically retry with anthropic json_object emulation --- .../providers/anthropic_provider.rb | 52 +++++++++++++++---- 1 file changed, 41 insertions(+), 11 deletions(-) diff --git a/lib/active_agent/providers/anthropic_provider.rb b/lib/active_agent/providers/anthropic_provider.rb index 3e193104..6f2498bf 100644 --- a/lib/active_agent/providers/anthropic_provider.rb +++ b/lib/active_agent/providers/anthropic_provider.rb @@ -20,6 +20,14 @@ class AnthropicProvider < BaseProvider # Lead-in message for JSON response format emulation JSON_RESPONSE_FORMAT_LEAD_IN = "Here is the JSON requested:\n{" + attr_internal :json_format_retry_count + + def initialize(kwargs = {}) + super + + self.json_format_retry_count = kwargs[:max_retries] || ::Anthropic::Client::DEFAULT_MAX_RETRIES + end + # @todo Add support for Anthropic::BedrockClient and Anthropic::VertexClient # @return [Anthropic::Client] def client @@ -49,18 +57,18 @@ def extract_used_function_names message_stack.pluck(:content).flatten.select { _1[:type] == "tool_use" }.pluck(:name) end - # Returns true if tool_choice forces any tool use (type == :any). + # Checks if tool_choice requires the model to call any tool. # - # @return [Boolean] + # @return [Boolean] true if tool_choice type is :any def tool_choice_forces_required? return false unless request.tool_choice.respond_to?(:type) request.tool_choice.type == :any end - # Returns [true, name] if tool_choice forces a specific tool (type == :tool). + # Checks if tool_choice requires a specific tool to be called. # - # @return [Array] + # @return [Array] [true, tool_name] if forcing a specific tool, [false, nil] otherwise def tool_choice_forces_specific? return [ false, nil ] unless request.tool_choice.respond_to?(:type) return [ false, nil ] unless request.tool_choice.type == :tool @@ -79,6 +87,12 @@ def prepare_prompt_request_response_format }) end + # Selects between Anthropic's stable and beta message APIs. + # + # Uses beta API when explicitly requested via anthropic_beta option or when + # using MCP servers, which require beta features. Falls back to stable API + # for standard message creation. + # # @see BaseProvider#api_prompt_executer # @return [Anthropic::Messages, Anthropic::Resources::Beta::Messages] def api_prompt_executer @@ -92,7 +106,7 @@ def api_prompt_executer # @see BaseProvider#api_response_normalize # @param api_response [Anthropic::Models::Message] - # @return [Hash] normalized response hash + # @return [Hash] def api_response_normalize(api_response) return api_response unless api_response @@ -217,23 +231,39 @@ def process_tool_call_function(api_function_call) end end - # Converts API response message to hash for message_stack. - # Converts Anthropic gem response object to hash for storage. + # Processes completed API response and handles JSON format retries. # + # When response_format is json_object and the response fails JSON validation, + # recursively retries the request to obtain well-formed JSON. + # + # @see BaseProvider#process_prompt_finished # @param api_response [Anthropic::Models::Message] # @return [Common::PromptResponse, nil] def process_prompt_finished(api_response = nil) # Convert gem object to hash so that raw_response[:usage] works api_response_hash = api_response ? Anthropic::Transforms.gem_to_hash(api_response) : nil - super(api_response_hash) + + common_response = super(api_response_hash) + + # If we failed to get the expected well formed JSON Object Response, recursively try again + if request.response_format&.dig(:type) == "json_object" && common_response.message.parsed_json.nil? && json_format_retry_count > 0 + self.json_format_retry_count -= 1 + + resolve_prompt + else + common_response + end end + # Reconstructs JSON responses that were split due to Anthropic format constraints. # - # Handles JSON response format simulation by prepending `{` to the response - # content if the last message in the request is the JSON lead-in prompt. + # Anthropic's API doesn't natively support json_object response format, so we + # simulate it by having the assistant echo a JSON lead-in ("Here is the JSON requested:\n{"), + # then send the response back for completion. This method detects and reverses + # that workaround by stripping the lead-in message and prepending "{" to the response. # # @see BaseProvider#process_prompt_finished_extract_messages - # @param api_response [Hash] converted response hash + # @param api_response [Hash] API response with content blocks # @return [Array, nil] def process_prompt_finished_extract_messages(api_response) return unless api_response