# FRIDAY - AI Test Case Generator
<p align="center">
<img src="docs/images/banner.svg" alt="Auto Test Case Generator Banner" width="1000">
</p>
An AI-powered test case generator that leverages Google Vertex AI and LangChain to automatically create test cases from Jira/GitHub issues and Confluence documentation.
## 🚀 Features
- Generate test cases using Google Vertex AI or OpenAI
- Extract requirements from Jira tickets or GitHub issues
- Integrate context from Confluence pages
- Process data using LangChain's prompt engineering
- Store and search documents using ChromaDB vectorization
- Export test cases in JSON or Markdown format
- Create embeddings using web crawler for additional context
## 📋 Prerequisites
- Python 3.12+
- Google Cloud Platform account with Vertex AI enabled or OpenAI
- Jira/GitHub and Confluence access credentials
## 🔄 Sequence Diagram
```mermaid
%%{init: {
'theme': 'base',
'themeVariables': {
'primaryColor': '#1a1a1a',
'primaryTextColor': '#fff',
'primaryBorderColor': '#4285f4',
'lineColor': '#4285f4',
'secondaryColor': '#2d2d2d',
'tertiaryColor': '#2d2d2d',
'actorBkg': '#4285f4',
'actorTextColor': '#fff',
'actorLineColor': '#4285f4',
'signalColor': '#6c757d',
'signalTextColor': '#fff',
'labelBoxBkgColor': '#2d2d2d',
'labelBoxBorderColor': '#4285f4',
'labelTextColor': '#fff',
'loopTextColor': '#fff',
'noteBorderColor': '#43a047',
'noteBkgColor': '#43a047',
'noteTextColor': '#fff',
'activationBorderColor': '#4285f4',
'activationBkgColor': '#2d2d2d',
'sequenceNumberColor': '#fff'
}
}}%%
sequenceDiagram
box rgba(66, 133, 244, 0.1) External Components
participant User
end
box rgba(66, 133, 244, 0.1) Core System
participant Main
participant IssueConnector
participant JiraConnector
participant GitHubConnector
participant ConfluenceConnector
participant TestCaseGenerator
participant PromptBuilder
end
Note over User,PromptBuilder: Test Case Generation Flow
User->>+Main: Run main.py with issue-key/number<br/>and confluence-id
alt Jira Issue
rect rgba(67, 160, 71, 0.1)
Main->>+IssueConnector: Get issue details
IssueConnector->>+JiraConnector: Fetch Jira issue
JiraConnector-->>-IssueConnector: Return issue details
IssueConnector-->>-Main: Return issue details
Main->>+IssueConnector: Extract acceptance criteria
IssueConnector->>JiraConnector: Extract from Jira
JiraConnector-->>IssueConnector: Return criteria
IssueConnector-->>-Main: Return acceptance criteria
end
else GitHub Issue
rect rgba(67, 160, 71, 0.1)
Main->>+IssueConnector: Get issue details
IssueConnector->>+GitHubConnector: Fetch GitHub issue
GitHubConnector-->>-IssueConnector: Return issue details
IssueConnector-->>-Main: Return issue details
Main->>+IssueConnector: Extract acceptance criteria
IssueConnector->>GitHubConnector: Extract from GitHub
GitHubConnector-->>IssueConnector: Return criteria
IssueConnector-->>-Main: Return acceptance criteria
end
end
rect rgba(255, 152, 0, 0.1)
Main->>+ConfluenceConnector: Fetch Confluence<br/>page content
ConfluenceConnector-->>-Main: Return page content
end
rect rgba(66, 133, 244, 0.1)
Main->>+PromptBuilder: Build prompt with details
PromptBuilder-->>-Main: Return prompt
Main->>+TestCaseGenerator: Generate test cases
TestCaseGenerator-->>-Main: Return test cases
end
Main->>-User: Save test cases to<br/>output file
Note over User,PromptBuilder: Process Complete
```
## ⚡️ Quick Start
1. Install via Homebrew:
```bash
brew tap dipjyotimetia/friday
brew install friday
```
2. Run setup:
```bash
friday setup
```
3. Generate test cases:
```bash
# From Jira
friday generate --jira-key PROJ-123 --confluence-id 12345 -o test_cases.md
# From GitHub
friday generate --gh-issue 456 --gh-repo owner/repo --confluence-id 12345 -o test_cases.md
```
## 📖 Usage Options
### Web Crawler
```bash
# Crawl single domain
friday crawl https://example.com --provider vertex --persist-dir ./my_data/chroma --max-pages 5
# Crawl multiple domains
friday crawl https://example.com --provider vertex --persist-dir ./my_data/chroma --max-pages 10 --same-domain false
```
### Command Reference
```bash
friday --help # Show all commands
friday version # Display version
friday generate --help # Show generation options
```
### Parameters
- `--jira-key`: Jira issue key
- `--confluence-id`: Confluence page ID (optional)
- `--gh-issue`: GitHub issue number
- `--gh-repo`: GitHub repository (format: owner/repo)
- `--output`: Output file path (default: test_cases.json)
## 🔧 GitHub Action
```yaml
name: Friday Test Generator
on:
pull_request:
types: [opened, synchronize]
workflow_dispatch:
inputs:
confluence_id:
description: 'Confluence Page ID'
required: true
jira_id:
description: 'Jira Issue ID'
required: false
jobs:
generate-tests:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Generate Test Cases
uses: dipjyotimetia/friday@main
with:
github_token: ${{ secrets.GITHUB_TOKEN }}
confluence_url: ${{ secrets.CONFLUENCE_URL }}
confluence_user: ${{ secrets.CONFLUENCE_USERNAME }}
confluence_token: ${{ secrets.CONFLUENCE_API_TOKEN }}
jira_url: ${{ secrets.JIRA_URL }}
jira_user: ${{ secrets.JIRA_USERNAME }}
jira_token: ${{ secrets.JIRA_API_TOKEN }}
confluence_id: ${{ inputs.confluence_id || '12345' }}
jira_id: ${{ inputs.jira_id || github.event.pull_request.number }}
google_cloud_project: ${{ secrets.GOOGLE_CLOUD_PROJECT }}
google_cloud_region: ${{ secrets.GOOGLE_CLOUD_REGION }}
model_provider: 'vertex'
persist_dir: './data/chroma'
- name: Upload Test Cases
uses: actions/upload-artifact@v4
with:
name: test-cases
path: test_cases.md
```
## 💻 Development
1. Clone and setup:
```bash
git clone https://github.com/dipjyotimetia/friday.git
cd friday
chmod +x prerequisites.sh
./prerequisites.sh
```
2. Configure environment:
```bash
cp .env.example .env
# Add your credentials to .env
```
3. Run Tests:
```bash
# Run tests
poetry run pytest tests/ -v
```
4. Format Code:
```bash
# Format code
poetry run ruff format
```
6. Deploy to google cloud
```bash
chmod +x deploy.sh
PROJECT_ID="your-project" REGION="us-west1" ./deploy.sh
```
MIT License
Copyright (c) 2025 Dipjyoti Metia
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
Raw data
{
"_id": null,
"home_page": null,
"name": "friday-cli",
"maintainer": null,
"docs_url": null,
"requires_python": "<4.0,>=3.12",
"maintainer_email": null,
"keywords": "atlassian, jira, google, github, vertexai, genai, chroma, openai, deepseek, cli",
"author": "Dipjyoti Metia",
"author_email": "dipjyotimetia@gmail.com",
"download_url": "https://files.pythonhosted.org/packages/b0/56/d73b3b937ba020f69a1b403cf3ffe58b580d3a31de5ff7728ce8ca0c3a2e/friday_cli-0.1.43.tar.gz",
"platform": null,
"description": "# FRIDAY - AI Test Case Generator\n\n<p align=\"center\">\n <img src=\"docs/images/banner.svg\" alt=\"Auto Test Case Generator Banner\" width=\"1000\">\n</p>\n\nAn AI-powered test case generator that leverages Google Vertex AI and LangChain to automatically create test cases from Jira/GitHub issues and Confluence documentation.\n\n## \ud83d\ude80 Features\n\n- Generate test cases using Google Vertex AI or OpenAI\n- Extract requirements from Jira tickets or GitHub issues \n- Integrate context from Confluence pages\n- Process data using LangChain's prompt engineering\n- Store and search documents using ChromaDB vectorization\n- Export test cases in JSON or Markdown format\n- Create embeddings using web crawler for additional context\n\n## \ud83d\udccb Prerequisites\n\n- Python 3.12+\n- Google Cloud Platform account with Vertex AI enabled or OpenAI\n- Jira/GitHub and Confluence access credentials\n\n## \ud83d\udd04 Sequence Diagram\n\n```mermaid\n%%{init: {\n 'theme': 'base',\n 'themeVariables': {\n 'primaryColor': '#1a1a1a',\n 'primaryTextColor': '#fff',\n 'primaryBorderColor': '#4285f4',\n 'lineColor': '#4285f4',\n 'secondaryColor': '#2d2d2d',\n 'tertiaryColor': '#2d2d2d',\n 'actorBkg': '#4285f4',\n 'actorTextColor': '#fff',\n 'actorLineColor': '#4285f4',\n 'signalColor': '#6c757d',\n 'signalTextColor': '#fff',\n 'labelBoxBkgColor': '#2d2d2d',\n 'labelBoxBorderColor': '#4285f4',\n 'labelTextColor': '#fff',\n 'loopTextColor': '#fff',\n 'noteBorderColor': '#43a047',\n 'noteBkgColor': '#43a047',\n 'noteTextColor': '#fff',\n 'activationBorderColor': '#4285f4',\n 'activationBkgColor': '#2d2d2d',\n 'sequenceNumberColor': '#fff'\n }\n}}%%\n\nsequenceDiagram\n box rgba(66, 133, 244, 0.1) External Components\n participant User\n end\n \n box rgba(66, 133, 244, 0.1) Core System\n participant Main\n participant IssueConnector\n participant JiraConnector\n participant GitHubConnector\n participant ConfluenceConnector\n participant TestCaseGenerator\n participant PromptBuilder\n end\n\n Note over User,PromptBuilder: Test Case Generation Flow\n\n User->>+Main: Run main.py with issue-key/number<br/>and confluence-id\n \n alt Jira Issue\n rect rgba(67, 160, 71, 0.1)\n Main->>+IssueConnector: Get issue details\n IssueConnector->>+JiraConnector: Fetch Jira issue\n JiraConnector-->>-IssueConnector: Return issue details\n IssueConnector-->>-Main: Return issue details\n Main->>+IssueConnector: Extract acceptance criteria\n IssueConnector->>JiraConnector: Extract from Jira\n JiraConnector-->>IssueConnector: Return criteria\n IssueConnector-->>-Main: Return acceptance criteria\n end\n else GitHub Issue\n rect rgba(67, 160, 71, 0.1)\n Main->>+IssueConnector: Get issue details\n IssueConnector->>+GitHubConnector: Fetch GitHub issue\n GitHubConnector-->>-IssueConnector: Return issue details\n IssueConnector-->>-Main: Return issue details\n Main->>+IssueConnector: Extract acceptance criteria\n IssueConnector->>GitHubConnector: Extract from GitHub\n GitHubConnector-->>IssueConnector: Return criteria\n IssueConnector-->>-Main: Return acceptance criteria\n end\n end\n \n rect rgba(255, 152, 0, 0.1)\n Main->>+ConfluenceConnector: Fetch Confluence<br/>page content\n ConfluenceConnector-->>-Main: Return page content\n end\n \n rect rgba(66, 133, 244, 0.1)\n Main->>+PromptBuilder: Build prompt with details\n PromptBuilder-->>-Main: Return prompt\n Main->>+TestCaseGenerator: Generate test cases\n TestCaseGenerator-->>-Main: Return test cases\n end\n \n Main->>-User: Save test cases to<br/>output file\n\n Note over User,PromptBuilder: Process Complete\n```\n\n## \u26a1\ufe0f Quick Start\n\n1. Install via Homebrew:\n```bash\nbrew tap dipjyotimetia/friday\nbrew install friday\n```\n\n2. Run setup:\n```bash \nfriday setup\n```\n\n3. Generate test cases:\n```bash\n# From Jira\nfriday generate --jira-key PROJ-123 --confluence-id 12345 -o test_cases.md\n\n# From GitHub\nfriday generate --gh-issue 456 --gh-repo owner/repo --confluence-id 12345 -o test_cases.md\n```\n\n## \ud83d\udcd6 Usage Options\n\n### Web Crawler\n```bash\n# Crawl single domain\nfriday crawl https://example.com --provider vertex --persist-dir ./my_data/chroma --max-pages 5\n\n# Crawl multiple domains\nfriday crawl https://example.com --provider vertex --persist-dir ./my_data/chroma --max-pages 10 --same-domain false\n```\n\n### Command Reference\n```bash\nfriday --help # Show all commands\nfriday version # Display version\nfriday generate --help # Show generation options\n```\n\n### Parameters\n- `--jira-key`: Jira issue key\n- `--confluence-id`: Confluence page ID (optional)\n- `--gh-issue`: GitHub issue number\n- `--gh-repo`: GitHub repository (format: owner/repo)\n- `--output`: Output file path (default: test_cases.json)\n\n## \ud83d\udd27 GitHub Action\n\n```yaml\nname: Friday Test Generator\n\non:\n pull_request:\n types: [opened, synchronize]\n workflow_dispatch:\n inputs:\n confluence_id:\n description: 'Confluence Page ID'\n required: true\n jira_id:\n description: 'Jira Issue ID'\n required: false\n\njobs:\n generate-tests:\n runs-on: ubuntu-latest\n steps:\n - uses: actions/checkout@v4\n \n - name: Generate Test Cases\n uses: dipjyotimetia/friday@main\n with:\n github_token: ${{ secrets.GITHUB_TOKEN }}\n confluence_url: ${{ secrets.CONFLUENCE_URL }}\n confluence_user: ${{ secrets.CONFLUENCE_USERNAME }}\n confluence_token: ${{ secrets.CONFLUENCE_API_TOKEN }}\n jira_url: ${{ secrets.JIRA_URL }}\n jira_user: ${{ secrets.JIRA_USERNAME }}\n jira_token: ${{ secrets.JIRA_API_TOKEN }}\n confluence_id: ${{ inputs.confluence_id || '12345' }}\n jira_id: ${{ inputs.jira_id || github.event.pull_request.number }}\n google_cloud_project: ${{ secrets.GOOGLE_CLOUD_PROJECT }}\n google_cloud_region: ${{ secrets.GOOGLE_CLOUD_REGION }}\n model_provider: 'vertex'\n persist_dir: './data/chroma'\n\n - name: Upload Test Cases\n uses: actions/upload-artifact@v4\n with:\n name: test-cases\n path: test_cases.md\n```\n\n## \ud83d\udcbb Development\n\n1. Clone and setup:\n```bash\ngit clone https://github.com/dipjyotimetia/friday.git\ncd friday\nchmod +x prerequisites.sh\n./prerequisites.sh\n```\n\n2. Configure environment:\n```bash\ncp .env.example .env\n# Add your credentials to .env\n```\n\n3. Run Tests:\n```bash\n# Run tests\npoetry run pytest tests/ -v\n```\n\n4. Format Code:\n```bash\n# Format code\npoetry run ruff format\n```\n\n6. Deploy to google cloud\n```bash\nchmod +x deploy.sh\nPROJECT_ID=\"your-project\" REGION=\"us-west1\" ./deploy.sh\n```\nMIT License\n\nCopyright (c) 2025 Dipjyoti Metia\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "AI-powered test case generator CLI",
"version": "0.1.43",
"project_urls": {
"Bug Tracker": "https://github.com/dipjyotimetia/friday/issues",
"Homepage": "https://github.com/dipjyotimetia/friday"
},
"split_keywords": [
"atlassian",
" jira",
" google",
" github",
" vertexai",
" genai",
" chroma",
" openai",
" deepseek",
" cli"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "731b30674872799011461232f878828711b655bd26dcaf95b4c097c992e44feb",
"md5": "c60cecf529b8803fb72b56664ad1837c",
"sha256": "e7ef8e46d0cb644f4eb9533c8a8632dbd132f5e21cddaacfb400a2e2bef26abb"
},
"downloads": -1,
"filename": "friday_cli-0.1.43-py3-none-any.whl",
"has_sig": false,
"md5_digest": "c60cecf529b8803fb72b56664ad1837c",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": "<4.0,>=3.12",
"size": 25138,
"upload_time": "2025-01-29T12:43:27",
"upload_time_iso_8601": "2025-01-29T12:43:27.280157Z",
"url": "https://files.pythonhosted.org/packages/73/1b/30674872799011461232f878828711b655bd26dcaf95b4c097c992e44feb/friday_cli-0.1.43-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "b056d73b3b937ba020f69a1b403cf3ffe58b580d3a31de5ff7728ce8ca0c3a2e",
"md5": "0d38f0bc120757e3fc532411a54dceae",
"sha256": "5fb0068ab771ac272d76fa0909f926b1ee385da48c5aa358c5601674f5757d14"
},
"downloads": -1,
"filename": "friday_cli-0.1.43.tar.gz",
"has_sig": false,
"md5_digest": "0d38f0bc120757e3fc532411a54dceae",
"packagetype": "sdist",
"python_version": "source",
"requires_python": "<4.0,>=3.12",
"size": 19198,
"upload_time": "2025-01-29T12:43:29",
"upload_time_iso_8601": "2025-01-29T12:43:29.166275Z",
"url": "https://files.pythonhosted.org/packages/b0/56/d73b3b937ba020f69a1b403cf3ffe58b580d3a31de5ff7728ce8ca0c3a2e/friday_cli-0.1.43.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2025-01-29 12:43:29",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "dipjyotimetia",
"github_project": "friday",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"requirements": [
{
"name": "aiohappyeyeballs",
"specs": [
[
"==",
"2.4.4"
]
]
},
{
"name": "aiohttp",
"specs": [
[
"==",
"3.11.11"
]
]
},
{
"name": "aiosignal",
"specs": [
[
"==",
"1.3.2"
]
]
},
{
"name": "annotated-types",
"specs": [
[
"==",
"0.7.0"
]
]
},
{
"name": "anyio",
"specs": [
[
"==",
"4.8.0"
]
]
},
{
"name": "asgiref",
"specs": [
[
"==",
"3.8.1"
]
]
},
{
"name": "atlassian-python-api",
"specs": [
[
"==",
"3.41.19"
]
]
},
{
"name": "attrs",
"specs": [
[
"==",
"25.1.0"
]
]
},
{
"name": "automat",
"specs": [
[
"==",
"24.8.1"
]
]
},
{
"name": "backoff",
"specs": [
[
"==",
"2.2.1"
]
]
},
{
"name": "bcrypt",
"specs": [
[
"==",
"4.2.1"
]
]
},
{
"name": "beautifulsoup4",
"specs": [
[
"==",
"4.12.3"
]
]
},
{
"name": "build",
"specs": [
[
"==",
"1.2.2.post1"
]
]
},
{
"name": "cachetools",
"specs": [
[
"==",
"5.5.1"
]
]
},
{
"name": "certifi",
"specs": [
[
"==",
"2024.12.14"
]
]
},
{
"name": "cffi",
"specs": [
[
"==",
"1.17.1"
]
]
},
{
"name": "charset-normalizer",
"specs": [
[
"==",
"3.4.1"
]
]
},
{
"name": "chroma-hnswlib",
"specs": [
[
"==",
"0.7.6"
]
]
},
{
"name": "chromadb",
"specs": [
[
"==",
"0.5.23"
]
]
},
{
"name": "click",
"specs": [
[
"==",
"8.1.8"
]
]
},
{
"name": "colorama",
"specs": [
[
"==",
"0.4.6"
]
]
},
{
"name": "coloredlogs",
"specs": [
[
"==",
"15.0.1"
]
]
},
{
"name": "constantly",
"specs": [
[
"==",
"23.10.4"
]
]
},
{
"name": "cryptography",
"specs": [
[
"==",
"44.0.0"
]
]
},
{
"name": "cssselect",
"specs": [
[
"==",
"1.2.0"
]
]
},
{
"name": "dataclasses-json",
"specs": [
[
"==",
"0.6.7"
]
]
},
{
"name": "defusedxml",
"specs": [
[
"==",
"0.7.1"
]
]
},
{
"name": "deprecated",
"specs": [
[
"==",
"1.2.18"
]
]
},
{
"name": "distro",
"specs": [
[
"==",
"1.9.0"
]
]
},
{
"name": "docstring-parser",
"specs": [
[
"==",
"0.16"
]
]
},
{
"name": "durationpy",
"specs": [
[
"==",
"0.9"
]
]
},
{
"name": "fastapi",
"specs": [
[
"==",
"0.115.7"
]
]
},
{
"name": "filelock",
"specs": [
[
"==",
"3.17.0"
]
]
},
{
"name": "filetype",
"specs": [
[
"==",
"1.2.0"
]
]
},
{
"name": "flatbuffers",
"specs": [
[
"==",
"25.1.24"
]
]
},
{
"name": "frozenlist",
"specs": [
[
"==",
"1.5.0"
]
]
},
{
"name": "fsspec",
"specs": [
[
"==",
"2024.12.0"
]
]
},
{
"name": "google-ai-generativelanguage",
"specs": [
[
"==",
"0.6.15"
]
]
},
{
"name": "google-api-core",
"specs": [
[
"==",
"2.24.1"
]
]
},
{
"name": "google-api-python-client",
"specs": [
[
"==",
"2.160.0"
]
]
},
{
"name": "google-auth-httplib2",
"specs": [
[
"==",
"0.2.0"
]
]
},
{
"name": "google-auth",
"specs": [
[
"==",
"2.38.0"
]
]
},
{
"name": "google-cloud-aiplatform",
"specs": [
[
"==",
"1.79.0"
]
]
},
{
"name": "google-cloud-bigquery",
"specs": [
[
"==",
"3.29.0"
]
]
},
{
"name": "google-cloud-core",
"specs": [
[
"==",
"2.4.1"
]
]
},
{
"name": "google-cloud-resource-manager",
"specs": [
[
"==",
"1.14.0"
]
]
},
{
"name": "google-cloud-storage",
"specs": [
[
"==",
"2.19.0"
]
]
},
{
"name": "google-crc32c",
"specs": [
[
"==",
"1.6.0"
]
]
},
{
"name": "google-genai",
"specs": [
[
"==",
"0.7.0"
]
]
},
{
"name": "google-generativeai",
"specs": [
[
"==",
"0.8.4"
]
]
},
{
"name": "google-resumable-media",
"specs": [
[
"==",
"2.7.2"
]
]
},
{
"name": "googleapis-common-protos",
"specs": [
[
"==",
"1.66.0"
]
]
},
{
"name": "greenlet",
"specs": [
[
"==",
"3.1.1"
]
]
},
{
"name": "grpc-google-iam-v1",
"specs": [
[
"==",
"0.14.0"
]
]
},
{
"name": "grpcio-status",
"specs": [
[
"==",
"1.70.0"
]
]
},
{
"name": "grpcio",
"specs": [
[
"==",
"1.70.0"
]
]
},
{
"name": "h11",
"specs": [
[
"==",
"0.14.0"
]
]
},
{
"name": "httpcore",
"specs": [
[
"==",
"1.0.7"
]
]
},
{
"name": "httplib2",
"specs": [
[
"==",
"0.22.0"
]
]
},
{
"name": "httptools",
"specs": [
[
"==",
"0.6.4"
]
]
},
{
"name": "httpx-sse",
"specs": [
[
"==",
"0.4.0"
]
]
},
{
"name": "httpx",
"specs": [
[
"==",
"0.28.1"
]
]
},
{
"name": "huggingface-hub",
"specs": [
[
"==",
"0.28.0"
]
]
},
{
"name": "humanfriendly",
"specs": [
[
"==",
"10.0"
]
]
},
{
"name": "hyperlink",
"specs": [
[
"==",
"21.0.0"
]
]
},
{
"name": "idna",
"specs": [
[
"==",
"3.10"
]
]
},
{
"name": "importlib-metadata",
"specs": [
[
"==",
"8.5.0"
]
]
},
{
"name": "importlib-resources",
"specs": [
[
"==",
"6.5.2"
]
]
},
{
"name": "incremental",
"specs": [
[
"==",
"24.7.2"
]
]
},
{
"name": "itemadapter",
"specs": [
[
"==",
"0.10.0"
]
]
},
{
"name": "itemloaders",
"specs": [
[
"==",
"1.3.2"
]
]
},
{
"name": "jiter",
"specs": [
[
"==",
"0.8.2"
]
]
},
{
"name": "jmespath",
"specs": [
[
"==",
"1.0.1"
]
]
},
{
"name": "jsonpatch",
"specs": [
[
"==",
"1.33"
]
]
},
{
"name": "jsonpointer",
"specs": [
[
"==",
"3.0.0"
]
]
},
{
"name": "kubernetes",
"specs": [
[
"==",
"32.0.0"
]
]
},
{
"name": "langchain-chroma",
"specs": [
[
"==",
"0.2.1"
]
]
},
{
"name": "langchain-community",
"specs": [
[
"==",
"0.3.16"
]
]
},
{
"name": "langchain-core",
"specs": [
[
"==",
"0.3.32"
]
]
},
{
"name": "langchain-deepseek",
"specs": [
[
"==",
"0.0.1"
]
]
},
{
"name": "langchain-google-genai",
"specs": [
[
"==",
"2.0.9"
]
]
},
{
"name": "langchain-google-vertexai",
"specs": [
[
"==",
"2.0.12"
]
]
},
{
"name": "langchain-openai",
"specs": [
[
"==",
"0.3.2"
]
]
},
{
"name": "langchain-text-splitters",
"specs": [
[
"==",
"0.3.5"
]
]
},
{
"name": "langchain",
"specs": [
[
"==",
"0.3.16"
]
]
},
{
"name": "langsmith",
"specs": [
[
"==",
"0.3.2"
]
]
},
{
"name": "lxml",
"specs": [
[
"==",
"5.3.0"
]
]
},
{
"name": "markdown-it-py",
"specs": [
[
"==",
"3.0.0"
]
]
},
{
"name": "marshmallow",
"specs": [
[
"==",
"3.26.0"
]
]
},
{
"name": "mdurl",
"specs": [
[
"==",
"0.1.2"
]
]
},
{
"name": "mmh3",
"specs": [
[
"==",
"5.1.0"
]
]
},
{
"name": "monotonic",
"specs": [
[
"==",
"1.6"
]
]
},
{
"name": "mpmath",
"specs": [
[
"==",
"1.3.0"
]
]
},
{
"name": "multidict",
"specs": [
[
"==",
"6.1.0"
]
]
},
{
"name": "mypy-extensions",
"specs": [
[
"==",
"1.0.0"
]
]
},
{
"name": "numpy",
"specs": [
[
"==",
"1.26.4"
]
]
},
{
"name": "oauthlib",
"specs": [
[
"==",
"3.2.2"
]
]
},
{
"name": "onnxruntime",
"specs": [
[
"==",
"1.20.1"
]
]
},
{
"name": "openai",
"specs": [
[
"==",
"1.60.2"
]
]
},
{
"name": "opentelemetry-api",
"specs": [
[
"==",
"1.29.0"
]
]
},
{
"name": "opentelemetry-exporter-otlp-proto-common",
"specs": [
[
"==",
"1.29.0"
]
]
},
{
"name": "opentelemetry-exporter-otlp-proto-grpc",
"specs": [
[
"==",
"1.29.0"
]
]
},
{
"name": "opentelemetry-instrumentation-asgi",
"specs": [
[
"==",
"0.50b0"
]
]
},
{
"name": "opentelemetry-instrumentation-fastapi",
"specs": [
[
"==",
"0.50b0"
]
]
},
{
"name": "opentelemetry-instrumentation",
"specs": [
[
"==",
"0.50b0"
]
]
},
{
"name": "opentelemetry-proto",
"specs": [
[
"==",
"1.29.0"
]
]
},
{
"name": "opentelemetry-sdk",
"specs": [
[
"==",
"1.29.0"
]
]
},
{
"name": "opentelemetry-semantic-conventions",
"specs": [
[
"==",
"0.50b0"
]
]
},
{
"name": "opentelemetry-util-http",
"specs": [
[
"==",
"0.50b0"
]
]
},
{
"name": "orjson",
"specs": [
[
"==",
"3.10.15"
]
]
},
{
"name": "overrides",
"specs": [
[
"==",
"7.7.0"
]
]
},
{
"name": "packaging",
"specs": [
[
"==",
"24.2"
]
]
},
{
"name": "parsel",
"specs": [
[
"==",
"1.10.0"
]
]
},
{
"name": "posthog",
"specs": [
[
"==",
"3.11.0"
]
]
},
{
"name": "propcache",
"specs": [
[
"==",
"0.2.1"
]
]
},
{
"name": "protego",
"specs": [
[
"==",
"0.4.0"
]
]
},
{
"name": "proto-plus",
"specs": [
[
"==",
"1.26.0"
]
]
},
{
"name": "protobuf",
"specs": [
[
"==",
"5.29.3"
]
]
},
{
"name": "pyasn1-modules",
"specs": [
[
"==",
"0.4.1"
]
]
},
{
"name": "pyasn1",
"specs": [
[
"==",
"0.6.1"
]
]
},
{
"name": "pycparser",
"specs": [
[
"==",
"2.22"
]
]
},
{
"name": "pydantic-core",
"specs": [
[
"==",
"2.27.2"
]
]
},
{
"name": "pydantic-settings",
"specs": [
[
"==",
"2.7.1"
]
]
},
{
"name": "pydantic",
"specs": [
[
"==",
"2.10.6"
]
]
},
{
"name": "pydispatcher",
"specs": [
[
"==",
"2.0.7"
]
]
},
{
"name": "pygithub",
"specs": [
[
"==",
"2.5.0"
]
]
},
{
"name": "pygments",
"specs": [
[
"==",
"2.19.1"
]
]
},
{
"name": "pyjwt",
"specs": [
[
"==",
"2.10.1"
]
]
},
{
"name": "pynacl",
"specs": [
[
"==",
"1.5.0"
]
]
},
{
"name": "pyopenssl",
"specs": [
[
"==",
"25.0.0"
]
]
},
{
"name": "pyparsing",
"specs": [
[
"==",
"3.2.1"
]
]
},
{
"name": "pypika",
"specs": [
[
"==",
"0.48.9"
]
]
},
{
"name": "pyproject-hooks",
"specs": [
[
"==",
"1.2.0"
]
]
},
{
"name": "pypydispatcher",
"specs": [
[
"==",
"2.1.2"
]
]
},
{
"name": "pyreadline3",
"specs": [
[
"==",
"3.5.4"
]
]
},
{
"name": "python-dateutil",
"specs": [
[
"==",
"2.9.0.post0"
]
]
},
{
"name": "python-dotenv",
"specs": [
[
"==",
"1.0.1"
]
]
},
{
"name": "python-json-logger",
"specs": [
[
"==",
"3.2.1"
]
]
},
{
"name": "pyyaml",
"specs": [
[
"==",
"6.0.2"
]
]
},
{
"name": "queuelib",
"specs": [
[
"==",
"1.7.0"
]
]
},
{
"name": "regex",
"specs": [
[
"==",
"2024.11.6"
]
]
},
{
"name": "requests-file",
"specs": [
[
"==",
"2.1.0"
]
]
},
{
"name": "requests-oauthlib",
"specs": [
[
"==",
"2.0.0"
]
]
},
{
"name": "requests-toolbelt",
"specs": [
[
"==",
"1.0.0"
]
]
},
{
"name": "requests",
"specs": [
[
"==",
"2.32.3"
]
]
},
{
"name": "retrying",
"specs": [
[
"==",
"1.3.4"
]
]
},
{
"name": "rich",
"specs": [
[
"==",
"13.9.4"
]
]
},
{
"name": "rsa",
"specs": [
[
"==",
"4.9"
]
]
},
{
"name": "scrapy",
"specs": [
[
"==",
"2.12.0"
]
]
},
{
"name": "service-identity",
"specs": [
[
"==",
"24.2.0"
]
]
},
{
"name": "setuptools",
"specs": [
[
"==",
"75.8.0"
]
]
},
{
"name": "shapely",
"specs": [
[
"==",
"2.0.6"
]
]
},
{
"name": "shellingham",
"specs": [
[
"==",
"1.5.4"
]
]
},
{
"name": "six",
"specs": [
[
"==",
"1.17.0"
]
]
},
{
"name": "sniffio",
"specs": [
[
"==",
"1.3.1"
]
]
},
{
"name": "soupsieve",
"specs": [
[
"==",
"2.6"
]
]
},
{
"name": "sqlalchemy",
"specs": [
[
"==",
"2.0.37"
]
]
},
{
"name": "starlette",
"specs": [
[
"==",
"0.45.3"
]
]
},
{
"name": "sympy",
"specs": [
[
"==",
"1.13.3"
]
]
},
{
"name": "tenacity",
"specs": [
[
"==",
"9.0.0"
]
]
},
{
"name": "tiktoken",
"specs": [
[
"==",
"0.8.0"
]
]
},
{
"name": "tldextract",
"specs": [
[
"==",
"5.1.3"
]
]
},
{
"name": "tokenizers",
"specs": [
[
"==",
"0.20.3"
]
]
},
{
"name": "tqdm",
"specs": [
[
"==",
"4.67.1"
]
]
},
{
"name": "twisted",
"specs": [
[
"==",
"24.11.0"
]
]
},
{
"name": "typer",
"specs": [
[
"==",
"0.15.1"
]
]
},
{
"name": "typing-extensions",
"specs": [
[
"==",
"4.12.2"
]
]
},
{
"name": "typing-inspect",
"specs": [
[
"==",
"0.9.0"
]
]
},
{
"name": "uritemplate",
"specs": [
[
"==",
"4.1.1"
]
]
},
{
"name": "urllib3",
"specs": [
[
"==",
"2.3.0"
]
]
},
{
"name": "uvicorn",
"specs": [
[
"==",
"0.34.0"
]
]
},
{
"name": "uvloop",
"specs": [
[
"==",
"0.21.0"
]
]
},
{
"name": "w3lib",
"specs": [
[
"==",
"2.3.1"
]
]
},
{
"name": "watchfiles",
"specs": [
[
"==",
"1.0.4"
]
]
},
{
"name": "websocket-client",
"specs": [
[
"==",
"1.8.0"
]
]
},
{
"name": "websockets",
"specs": [
[
"==",
"14.2"
]
]
},
{
"name": "wrapt",
"specs": [
[
"==",
"1.17.2"
]
]
},
{
"name": "yarl",
"specs": [
[
"==",
"1.18.3"
]
]
},
{
"name": "zipp",
"specs": [
[
"==",
"3.21.0"
]
]
},
{
"name": "zope-interface",
"specs": [
[
"==",
"7.2"
]
]
},
{
"name": "zstandard",
"specs": [
[
"==",
"0.23.0"
]
]
}
],
"lcname": "friday-cli"
}