mirror of
https://github.com/Zie619/n8n-workflows.git
synced 2025-11-25 19:37:52 +08:00
Merge pull request #38 from wildcard/feature/devcontainer-integration
feat: Add devcontainer configuration for Claude Code development
This commit is contained in:
82
.devcontainer/Dockerfile
Normal file
82
.devcontainer/Dockerfile
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
FROM node:20
|
||||||
|
|
||||||
|
ARG TZ
|
||||||
|
ENV TZ="$TZ"
|
||||||
|
|
||||||
|
# Install basic development tools, Python, and iptables/ipset
|
||||||
|
RUN apt update && apt install -y less \
|
||||||
|
git \
|
||||||
|
procps \
|
||||||
|
sudo \
|
||||||
|
fzf \
|
||||||
|
zsh \
|
||||||
|
man-db \
|
||||||
|
unzip \
|
||||||
|
gnupg2 \
|
||||||
|
gh \
|
||||||
|
iptables \
|
||||||
|
ipset \
|
||||||
|
iproute2 \
|
||||||
|
dnsutils \
|
||||||
|
aggregate \
|
||||||
|
jq \
|
||||||
|
python3 \
|
||||||
|
python3-pip \
|
||||||
|
python3-venv \
|
||||||
|
python3-dev
|
||||||
|
|
||||||
|
# Ensure default node user has access to /usr/local/share
|
||||||
|
RUN mkdir -p /usr/local/share/npm-global && \
|
||||||
|
chown -R node:node /usr/local/share
|
||||||
|
|
||||||
|
ARG USERNAME=node
|
||||||
|
|
||||||
|
# Persist bash history.
|
||||||
|
RUN SNIPPET="export PROMPT_COMMAND='history -a' && export HISTFILE=/commandhistory/.bash_history" \
|
||||||
|
&& mkdir /commandhistory \
|
||||||
|
&& touch /commandhistory/.bash_history \
|
||||||
|
&& chown -R $USERNAME /commandhistory
|
||||||
|
|
||||||
|
# Set `DEVCONTAINER` environment variable to help with orientation
|
||||||
|
ENV DEVCONTAINER=true
|
||||||
|
|
||||||
|
# Create workspace and config directories and set permissions
|
||||||
|
RUN mkdir -p /workspace /home/node/.claude && \
|
||||||
|
chown -R node:node /workspace /home/node/.claude
|
||||||
|
|
||||||
|
WORKDIR /workspace
|
||||||
|
|
||||||
|
RUN ARCH=$(dpkg --print-architecture) && \
|
||||||
|
wget "https://github.com/dandavison/delta/releases/download/0.18.2/git-delta_0.18.2_${ARCH}.deb" && \
|
||||||
|
sudo dpkg -i "git-delta_0.18.2_${ARCH}.deb" && \
|
||||||
|
rm "git-delta_0.18.2_${ARCH}.deb"
|
||||||
|
|
||||||
|
# Set up non-root user
|
||||||
|
USER node
|
||||||
|
|
||||||
|
# Install global packages
|
||||||
|
ENV NPM_CONFIG_PREFIX=/usr/local/share/npm-global
|
||||||
|
ENV PATH=$PATH:/usr/local/share/npm-global/bin
|
||||||
|
|
||||||
|
# Set the default shell to zsh rather than sh
|
||||||
|
ENV SHELL=/bin/zsh
|
||||||
|
|
||||||
|
# Default powerline10k theme
|
||||||
|
RUN sh -c "$(wget -O- https://github.com/deluan/zsh-in-docker/releases/download/v1.2.0/zsh-in-docker.sh)" -- \
|
||||||
|
-p git \
|
||||||
|
-p fzf \
|
||||||
|
-a "source /usr/share/doc/fzf/examples/key-bindings.zsh" \
|
||||||
|
-a "source /usr/share/doc/fzf/examples/completion.zsh" \
|
||||||
|
-a "export PROMPT_COMMAND='history -a' && export HISTFILE=/commandhistory/.bash_history" \
|
||||||
|
-x
|
||||||
|
|
||||||
|
# Install Claude
|
||||||
|
RUN npm install -g @anthropic-ai/claude-code
|
||||||
|
|
||||||
|
# Copy and set up firewall script
|
||||||
|
COPY init-firewall.sh /usr/local/bin/
|
||||||
|
USER root
|
||||||
|
RUN chmod +x /usr/local/bin/init-firewall.sh && \
|
||||||
|
echo "node ALL=(root) NOPASSWD: /usr/local/bin/init-firewall.sh" > /etc/sudoers.d/node-firewall && \
|
||||||
|
chmod 0440 /etc/sudoers.d/node-firewall
|
||||||
|
USER node
|
||||||
62
.devcontainer/devcontainer.json
Normal file
62
.devcontainer/devcontainer.json
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
{
|
||||||
|
"name": "n8n Workflows - Claude Code",
|
||||||
|
"build": {
|
||||||
|
"dockerfile": "Dockerfile",
|
||||||
|
"args": {
|
||||||
|
"TZ": "${localEnv:TZ:America/Los_Angeles}"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"runArgs": [
|
||||||
|
"--cap-add=NET_ADMIN",
|
||||||
|
"--cap-add=NET_RAW"
|
||||||
|
],
|
||||||
|
"customizations": {
|
||||||
|
"vscode": {
|
||||||
|
"extensions": [
|
||||||
|
"dbaeumer.vscode-eslint",
|
||||||
|
"esbenp.prettier-vscode",
|
||||||
|
"eamodio.gitlens",
|
||||||
|
"ms-python.python",
|
||||||
|
"ms-python.flake8",
|
||||||
|
"ms-python.black-formatter",
|
||||||
|
"tamasfe.even-better-toml"
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
||||||
|
"editor.codeActionsOnSave": {
|
||||||
|
"source.fixAll.eslint": "explicit"
|
||||||
|
},
|
||||||
|
"python.defaultInterpreterPath": "/usr/local/bin/python3",
|
||||||
|
"python.linting.enabled": true,
|
||||||
|
"python.linting.flake8Enabled": true,
|
||||||
|
"[python]": {
|
||||||
|
"editor.defaultFormatter": "ms-python.black-formatter"
|
||||||
|
},
|
||||||
|
"terminal.integrated.defaultProfile.linux": "zsh",
|
||||||
|
"terminal.integrated.profiles.linux": {
|
||||||
|
"bash": {
|
||||||
|
"path": "bash",
|
||||||
|
"icon": "terminal-bash"
|
||||||
|
},
|
||||||
|
"zsh": {
|
||||||
|
"path": "zsh"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"remoteUser": "node",
|
||||||
|
"mounts": [
|
||||||
|
"source=claude-code-bashhistory,target=/commandhistory,type=volume",
|
||||||
|
"source=claude-code-config,target=/home/node/.claude,type=volume"
|
||||||
|
],
|
||||||
|
"remoteEnv": {
|
||||||
|
"NODE_OPTIONS": "--max-old-space-size=4096",
|
||||||
|
"CLAUDE_CONFIG_DIR": "/home/node/.claude",
|
||||||
|
"POWERLEVEL9K_DISABLE_GITSTATUS": "true"
|
||||||
|
},
|
||||||
|
"workspaceMount": "source=${localWorkspaceFolder},target=/workspace,type=bind,consistency=delegated",
|
||||||
|
"workspaceFolder": "/workspace",
|
||||||
|
"postCreateCommand": "sudo /usr/local/bin/init-firewall.sh"
|
||||||
|
}
|
||||||
118
.devcontainer/init-firewall.sh
Normal file
118
.devcontainer/init-firewall.sh
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail # Exit on error, undefined vars, and pipeline failures
|
||||||
|
IFS=$'\n\t' # Stricter word splitting
|
||||||
|
|
||||||
|
# Flush existing rules and delete existing ipsets
|
||||||
|
iptables -F
|
||||||
|
iptables -X
|
||||||
|
iptables -t nat -F
|
||||||
|
iptables -t nat -X
|
||||||
|
iptables -t mangle -F
|
||||||
|
iptables -t mangle -X
|
||||||
|
ipset destroy allowed-domains 2>/dev/null || true
|
||||||
|
|
||||||
|
# First allow DNS and localhost before any restrictions
|
||||||
|
# Allow outbound DNS
|
||||||
|
iptables -A OUTPUT -p udp --dport 53 -j ACCEPT
|
||||||
|
# Allow inbound DNS responses
|
||||||
|
iptables -A INPUT -p udp --sport 53 -j ACCEPT
|
||||||
|
# Allow outbound SSH
|
||||||
|
iptables -A OUTPUT -p tcp --dport 22 -j ACCEPT
|
||||||
|
# Allow inbound SSH responses
|
||||||
|
iptables -A INPUT -p tcp --sport 22 -m state --state ESTABLISHED -j ACCEPT
|
||||||
|
# Allow localhost
|
||||||
|
iptables -A INPUT -i lo -j ACCEPT
|
||||||
|
iptables -A OUTPUT -o lo -j ACCEPT
|
||||||
|
|
||||||
|
# Create ipset with CIDR support
|
||||||
|
ipset create allowed-domains hash:net
|
||||||
|
|
||||||
|
# Fetch GitHub meta information and aggregate + add their IP ranges
|
||||||
|
echo "Fetching GitHub IP ranges..."
|
||||||
|
gh_ranges=$(curl -s https://api.github.com/meta)
|
||||||
|
if [ -z "$gh_ranges" ]; then
|
||||||
|
echo "ERROR: Failed to fetch GitHub IP ranges"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! echo "$gh_ranges" | jq -e '.web and .api and .git' >/dev/null; then
|
||||||
|
echo "ERROR: GitHub API response missing required fields"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Processing GitHub IPs..."
|
||||||
|
while read -r cidr; do
|
||||||
|
if [[ ! "$cidr" =~ ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}/[0-9]{1,2}$ ]]; then
|
||||||
|
echo "ERROR: Invalid CIDR range from GitHub meta: $cidr"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "Adding GitHub range $cidr"
|
||||||
|
ipset add allowed-domains "$cidr"
|
||||||
|
done < <(echo "$gh_ranges" | jq -r '(.web + .api + .git)[]' | aggregate -q)
|
||||||
|
|
||||||
|
# Resolve and add other allowed domains
|
||||||
|
for domain in \
|
||||||
|
"registry.npmjs.org" \
|
||||||
|
"api.anthropic.com" \
|
||||||
|
"sentry.io" \
|
||||||
|
"statsig.anthropic.com" \
|
||||||
|
"statsig.com"; do
|
||||||
|
echo "Resolving $domain..."
|
||||||
|
ips=$(dig +short A "$domain")
|
||||||
|
if [ -z "$ips" ]; then
|
||||||
|
echo "ERROR: Failed to resolve $domain"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
while read -r ip; do
|
||||||
|
if [[ ! "$ip" =~ ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$ ]]; then
|
||||||
|
echo "ERROR: Invalid IP from DNS for $domain: $ip"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "Adding $ip for $domain"
|
||||||
|
ipset add allowed-domains "$ip"
|
||||||
|
done < <(echo "$ips")
|
||||||
|
done
|
||||||
|
|
||||||
|
# Get host IP from default route
|
||||||
|
HOST_IP=$(ip route | grep default | cut -d" " -f3)
|
||||||
|
if [ -z "$HOST_IP" ]; then
|
||||||
|
echo "ERROR: Failed to detect host IP"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
HOST_NETWORK=$(echo "$HOST_IP" | sed "s/\.[0-9]*$/.0\/24/")
|
||||||
|
echo "Host network detected as: $HOST_NETWORK"
|
||||||
|
|
||||||
|
# Set up remaining iptables rules
|
||||||
|
iptables -A INPUT -s "$HOST_NETWORK" -j ACCEPT
|
||||||
|
iptables -A OUTPUT -d "$HOST_NETWORK" -j ACCEPT
|
||||||
|
|
||||||
|
# Set default policies to DROP first
|
||||||
|
iptables -P INPUT DROP
|
||||||
|
iptables -P FORWARD DROP
|
||||||
|
iptables -P OUTPUT DROP
|
||||||
|
|
||||||
|
# First allow established connections for already approved traffic
|
||||||
|
iptables -A INPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
|
||||||
|
iptables -A OUTPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
|
||||||
|
|
||||||
|
# Then allow only specific outbound traffic to allowed domains
|
||||||
|
iptables -A OUTPUT -m set --match-set allowed-domains dst -j ACCEPT
|
||||||
|
|
||||||
|
echo "Firewall configuration complete"
|
||||||
|
echo "Verifying firewall rules..."
|
||||||
|
if curl --connect-timeout 5 https://example.com >/dev/null 2>&1; then
|
||||||
|
echo "ERROR: Firewall verification failed - was able to reach https://example.com"
|
||||||
|
exit 1
|
||||||
|
else
|
||||||
|
echo "Firewall verification passed - unable to reach https://example.com as expected"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Verify GitHub API access
|
||||||
|
if ! curl --connect-timeout 5 https://api.github.com/zen >/dev/null 2>&1; then
|
||||||
|
echo "ERROR: Firewall verification failed - unable to reach https://api.github.com"
|
||||||
|
exit 1
|
||||||
|
else
|
||||||
|
echo "Firewall verification passed - able to reach https://api.github.com as expected"
|
||||||
|
fi
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -88,3 +88,6 @@ package-lock.json
|
|||||||
|
|
||||||
# versions
|
# versions
|
||||||
.python-version
|
.python-version
|
||||||
|
|
||||||
|
# Claude Code local settings (created during development)
|
||||||
|
.claude/settings.local.json
|
||||||
Reference in New Issue
Block a user