This guide walks you through setting up Open SWE end-to-end: local development, GitHub App creation, LangSmith configuration, webhooks, and production deployment.
The steps are ordered to avoid forward references. Each step only depends on things you've already completed.
- Python 3.11 – 3.13 (3.14 is not yet supported due to dependency constraints)
- uv package manager
- LangGraph CLI
- ngrok (for local development — exposes webhook endpoints to the internet)
git clone https://github.com/langchain-ai/open-swe.git
cd open-swe
uv venv
source .venv/bin/activate
uv sync --all-extrasYou'll need the ngrok URL in subsequent steps when configuring webhooks, so start it first.
ngrok http 2024 --url https://some-url-you-configure.ngrok.devYou don't need to pass the --url flag, however doing so will use the same subdomain each time you startup the server. Without this, you'll need to update the webhook URL in GitHub, Slack and Linear every time you restart your server for local development.
Copy the HTTPS URL you set, or if you didn't pass --url, the one ngrok gives you. You'll paste this into the webhook settings in steps 3 and 5.
Keep this terminal open — ngrok needs to stay running during local development. Use a second terminal for the rest of the steps.
Open SWE authenticates as a GitHub App to clone repos, push branches, and open PRs.
Before creating the app you need to decide on an OAuth provider ID — this is a short string you'll use in both GitHub and LangSmith to link the two. Pick something memorable, for example:
github-oauth-provider
Write this down. You'll use it in the callback URL below and again in step 4 when configuring LangSmith.
- Go to GitHub Settings → Developer settings → GitHub Apps → New GitHub App
- Fill in:
- App name:
open-swe(or your preferred name) - Homepage URL: This can be any valid URL — it's only shown on the GitHub Marketplace page (which you won't be using). Use something like
https://github.com/langchain-ai/open-swe - Callback URL:
https://smith.langchain.com/host-oauth-callback/<your-provider-id>— replace<your-provider-id>with the ID you chose in step 3a (e.g.https://smith.langchain.com/host-oauth-callback/github-oauth-provider) - Request user authorization (OAuth) during installation: ✅ Enable this
- Webhook URL:
https://<your-ngrok-url>/webhooks/github— use the ngrok URL from step 2 - Webhook secret: generate one and save it — you'll need it later as
GITHUB_WEBHOOK_SECRET:openssl rand -hex 32
- App name:
- Set permissions:
- Repository permissions:
- Contents: Read & write
- Pull requests: Read & write
- Issues: Read & write
- Metadata: Read-only
- Repository permissions:
- Under Subscribe to events, enable:
Issue commentPull request reviewPull request review comment
- Click Create GitHub App
After creating the app:
- App ID — shown at the top of the app's settings page. Save this as
GITHUB_APP_ID. - Private key — scroll down to Private keys → click Generate a private key. A
.pemfile will download. Save its contents asGITHUB_APP_PRIVATE_KEY.
- From your app's settings page, click Install App in the sidebar
- Select your org or personal account
- Choose which repositories Open SWE should have access to
- Click Install
- After installation, look at the URL in your browser — it will look like:
or for an org:
https://github.com/settings/installations/12345678The number at the end (https://github.com/organizations/YOUR-ORG/settings/installations/1234567812345678) is your Installation ID. Save this asGITHUB_APP_INSTALLATION_ID.
Note: The installation page may prompt you to authenticate with LangSmith. If you haven't set up LangSmith yet (step 4), that's fine — you can still grab the Installation ID from the URL and complete the OAuth setup later.
Open SWE uses LangSmith for:
- Tracing: all agent runs are logged for debugging and observability
- Sandboxes: each task runs in an isolated LangSmith cloud sandbox
- Create a LangSmith account if you don't have one
- Go to Settings → API Keys → Create API Key
- Save it as
LANGSMITH_API_KEY_PROD
This lets each user authenticate with their own GitHub account. Without it, all operations use the GitHub App's installation token (a shared bot identity).
What this affects:
- With per-user OAuth: PRs and commits show the triggering user's identity; each user's GitHub permissions are respected
- Without it (bot-token-only mode): all PRs and commits appear as the GitHub App bot; the app's installation-level permissions are used for everything
To set up per-user OAuth:
- In LangSmith, go to Settings → OAuth Providers → Add Provider
- Set the Provider ID to the same string you chose in step 3a (e.g.
github-oauth-provider) - Enter the Client ID and Client Secret from your GitHub App (found on the GitHub App settings page under OAuth credentials)
- Save. You'll reference this Provider ID as
GITHUB_OAUTH_PROVIDER_IDin your environment variables.
LangSmith sandboxes provide the isolated execution environment for each agent run. You can create a template using the same Docker image we use internally by visiting the sandbox page in LangSmith, and setting the following fields:
Name: you can set this to whatever name you'd like, e.g.open-sweContainer Image:bracelangchain/deepagents-sandbox:v1this contains the Docker file in this repoCPU:500mMemory:4096MiEphemeral Storage:15Gi
If you don't set these, you can use a Python based docker image in the template.
Open SWE can be triggered from GitHub, Linear, and/or Slack. Configure whichever surfaces your team uses — you don't need all of them.
GitHub triggering works automatically once your GitHub App is set up (step 3). Users can:
- Tag
@openswein issue titles or bodies to start a task - Tag
@openswein issue comments for follow-up instructions - Tag
@openswein PR review comments to have it address review feedback
To control which GitHub users can trigger the agent, add them to the GITHUB_USER_EMAIL_MAP in agent/utils/github_user_email_map.py:
GITHUB_USER_EMAIL_MAP = {
"their-github-username": "their-email@example.com",
}You should also add the GitHub organization which should be allowed to be triggered from in GitHub:
agent/webapp.py
ALLOWED_GITHUB_ORGS = "langchain-ai,anthropics"Open SWE listens for Linear comments that mention @openswe.
Create a webhook:
- In Linear, go to Settings → API → Webhooks → New webhook
- Fill in:
- Label:
open-swe - URL:
https://<your-ngrok-url>/webhooks/linear— use the ngrok URL from step 2 - Secret: generate with
openssl rand -hex 32— save this asLINEAR_WEBHOOK_SECRET
- Label:
- Under Data change events, enable Comments → Create only
- Click Create webhook
Get your API key:
- Go to Settings → API → Personal API keys → New API key
- Name it
open-swe, select All access, and copy the key - Save it as
LINEAR_API_KEY
Configure team-to-repo mapping:
Open SWE routes Linear issues to GitHub repos based on the Linear team and project. Edit the mapping in agent/utils/linear_team_repo_map.py:
LINEAR_TEAM_TO_REPO = {
"My Team": {"owner": "my-org", "name": "my-repo"},
"Engineering": {
"projects": {
"backend": {"owner": "my-org", "name": "backend"},
"frontend": {"owner": "my-org", "name": "frontend"},
},
"default": {"owner": "my-org", "name": "monorepo"},
},
}Create a Slack App:
- Go to api.slack.com/apps → Create New App → From a manifest
- Copy the manifest below, replacing the two placeholder URLs:
- Replace
<your-provider-id>with the OAuth provider ID from step 3a - Replace
<your-ngrok-url>with the ngrok URL from step 2
- Replace
Slack App Manifest
{
"display_information": {
"name": "Open SWE",
"description": "Enables Open SWE to interact with your workspace",
"background_color": "#000000"
},
"features": {
"app_home": {
"home_tab_enabled": false,
"messages_tab_enabled": true,
"messages_tab_read_only_enabled": false
},
"bot_user": {
"display_name": "Open SWE",
"always_online": true
}
},
"oauth_config": {
"redirect_urls": [
"https://smith.langchain.com/host-oauth-callback/<your-provider-id>"
],
"scopes": {
"bot": [
"reactions:write",
"app_mentions:read",
"channels:history",
"channels:read",
"chat:write",
"groups:history",
"groups:read",
"im:history",
"im:read",
"im:write",
"mpim:history",
"mpim:read",
"team:read",
"users:read",
"users:read.email"
]
}
},
"settings": {
"event_subscriptions": {
"request_url": "https://<your-ngrok-url>/webhooks/slack",
"bot_events": [
"app_mention",
"message.im",
"message.mpim"
]
},
"org_deploy_enabled": false,
"socket_mode_enabled": false,
"token_rotation_enabled": false
}
}- Install the app to your workspace and copy the Bot User OAuth Token (
xoxb-...)
Credentials you'll need:
SLACK_BOT_TOKEN: the Bot User OAuth Token (xoxb-...)SLACK_SIGNING_SECRET: found under Basic Information → App CredentialsSLACK_BOT_USER_ID: the bot's user ID (find it in Slack by clicking the bot's profile)SLACK_BOT_USERNAME: the bot's display name (e.g.open-swe)
Configure default repo:
Slack messages are routed to a default repo unless the user specifies one with repo:owner/name:
SLACK_REPO_OWNER="my-org" # Default GitHub org
SLACK_REPO_NAME="my-repo" # Default GitHub repoCreate a .env file in the project root. Below is the full list — only fill in the sections relevant to the triggers you configured.
# === LangSmith ===
LANGSMITH_API_KEY_PROD="" # From step 4a
LANGCHAIN_TRACING_V2="true"
LANGCHAIN_PROJECT="" # LangSmith project name for traces
# === LLM ===
ANTHROPIC_API_KEY="" # Anthropic API key (default provider)
# === GitHub App (required) ===
GITHUB_APP_ID="" # From step 3c
GITHUB_APP_PRIVATE_KEY="-----BEGIN RSA PRIVATE KEY-----
...
-----END RSA PRIVATE KEY-----
"
GITHUB_APP_INSTALLATION_ID="" # From step 3d
# === GitHub Webhook (required) ===
GITHUB_WEBHOOK_SECRET="" # The secret you generated in step 3b
# === GitHub OAuth via LangSmith (optional) ===
# Without these, all operations use the GitHub App's bot token.
# With these, each user authenticates with their own GitHub account.
GITHUB_OAUTH_PROVIDER_ID="" # The provider ID from steps 3a / 4b
# === Org Allowlist (optional) ===
# Comma-separated list of GitHub orgs the agent is allowed to operate on.
# Leave empty to allow all orgs.
ALLOWED_GITHUB_ORGS="" # e.g. "my-org,my-other-org"
# === Linear (if using Linear trigger) ===
LINEAR_API_KEY="" # From step 5
LINEAR_WEBHOOK_SECRET="" # From step 5
# === Slack (if using Slack trigger) ===
SLACK_BOT_TOKEN="" # From step 5
SLACK_BOT_USER_ID=""
SLACK_BOT_USERNAME=""
SLACK_SIGNING_SECRET=""
SLACK_REPO_OWNER="" # Default org for Slack-triggered tasks
SLACK_REPO_NAME="" # Default repo for Slack-triggered tasks
# === Sandbox (optional) ===
DEFAULT_SANDBOX_TEMPLATE_NAME="" # Custom sandbox template name (default: deepagents-cli)
DEFAULT_SANDBOX_TEMPLATE_IMAGE="" # Custom Docker image (default: python:3)
# === Token Encryption ===
TOKEN_ENCRYPTION_KEY="" # Generate with: openssl rand -base64 32Make sure ngrok is still running from step 2, then start the LangGraph server in a second terminal:
uv run langgraph dev --no-browserThe server runs on http://localhost:2024 with these endpoints:
| Endpoint | Purpose |
|---|---|
POST /webhooks/github |
GitHub issue/PR/comment webhooks |
POST /webhooks/linear |
Linear comment webhooks |
GET /webhooks/linear |
Linear webhook verification |
POST /webhooks/slack |
Slack event webhooks |
GET /webhooks/slack |
Slack webhook verification |
GET /health |
Health check |
- Go to any issue in a repository where the app is installed
- Create or comment on an issue with:
@openswe what files are in this repo? - You should see:
- A 👀 reaction on your comment within a few seconds
- A new run in your LangSmith project
- The agent replies with a comment on the issue
- Go to any Linear issue in a team you configured in
LINEAR_TEAM_TO_REPO - Add a comment:
@openswe what files are in this repo? - You should see:
- A 👀 reaction on your comment within a few seconds
- A new run in your LangSmith project
- The agent replies with a comment on the issue
- In any channel where the bot is invited, start a thread
- Mention the bot:
@open-swe what's in the repo? - You should see:
- An 👀 reaction on your message
- A reply in the thread with the agent's response
For production, deploy the agent on LangGraph Cloud instead of running locally:
- Push your code to a GitHub repository
- Connect the repo to LangGraph Cloud
- Set all environment variables from step 6 in the deployment config
- Update your webhook URLs (Linear, Slack, GitHub App) to point to your production URL (replace the ngrok URL)
The langgraph.json at the project root already defines the graph entry point and HTTP app:
{
"graphs": {
"agent": "agent.server:get_agent"
},
"http": {
"app": "agent.webapp:app"
}
}- Verify ngrok is running and the URL matches what's configured in GitHub/Linear/Slack
- Check the ngrok web inspector at
http://localhost:4040for incoming requests - Ensure you enabled the correct event types (Comments → Create for Linear,
app_mentionfor Slack, Issues + Issue comment for GitHub) - Webhook secrets are required — if
GITHUB_WEBHOOK_SECRET,LINEAR_WEBHOOK_SECRET, orSLACK_SIGNING_SECRETis not set, all requests to that endpoint will be rejected with 401
- Verify
GITHUB_APP_ID,GITHUB_APP_PRIVATE_KEY, andGITHUB_APP_INSTALLATION_IDare set correctly - Ensure the GitHub App is installed on the target repositories
- Check that the private key includes the full
-----BEGIN RSA PRIVATE KEY-----and-----END RSA PRIVATE KEY-----lines
- Verify
LANGSMITH_API_KEY_PRODis set and valid - Check LangSmith sandbox quotas in your workspace settings
- If you see
Failed to check template '', ensure eitherDEFAULT_SANDBOX_TEMPLATE_NAMEis set or that your LangSmith API key has permissions to create sandbox templates - If you get a 403 Forbidden error on the sandbox templates endpoint, your LangSmith workspace may not have sandbox access enabled — contact LangSmith support
- For GitHub: ensure the comment or issue contains
@openswe(case-insensitive), and the commenter's GitHub username is inGITHUB_USER_EMAIL_MAP - For Linear: ensure the comment contains
@openswe(case-insensitive) - For Slack: ensure the bot is invited to the channel and the message is an
@mention - Check server logs for webhook processing errors
- Ensure
TOKEN_ENCRYPTION_KEYis set (generate withopenssl rand -base64 32) - The key must be a valid 32-byte Fernet-compatible base64 string