Skip to content

Instantly share code, notes, and snippets.

@cvladan
Last active May 31, 2023 02:16
#!/bin/bash
# Homepage: https://gist.github.com/cvladan/8f0de5e61d773664d634cd70f3e07fe6
# Homepage: selivan.github.io/socks
# https://github.com/selivan/selivan.github.io/blob/master/socks.txt
#
# Author: Pavel Selivanov, Vladan Colovic
# Contributors: Vlad Safronov (Oracle Linux 7.5, Centos 7), Vladan Colovic (Ubuntu)
#
# Usage:
# export PORT=someport; export PASSWORD=somepass; export USER=someuser
# curl https://gist.githubusercontent.com/cvladan/8f0de5e61d773664d634cd70f3e07fe6/raw/danted_setup.sh | sudo --preserve-env bash
#
function get_external_address() {
local addr=$( timeout 3 dig +short myip.opendns.com @resolver1.opendns.com || \
timeout 3 curl -s http://ipecho.net/plain || \
timeout 3 curl -s http://ident.me/ || \
timeout 3 curl -s http://whatismyip.akamai.com/ )
[ $? -ne 0 ] && addr="<this server IP address>"
echo "$addr"
}
# args: file user password
function generate_password_file() {
# -1 generate md5-based password hash
echo "$2:$( openssl passwd -1 "$3" )" > "$1"
}
# args: file; generates: file.db
function generate_password_dbfile() {
awk -F: '{print $1; print $2}' < "$1" | db_load -T -t hash "${1}.db"
}
# args: file pwdfile
function generate_pam() {
# nodelay: don't cause a delay on auth failure. Anti-DDOS
cat > "$1" << EOF
auth required pam_pwdfile.so nodelay pwdfile=$2
account required pam_permit.so
EOF
}
# args: file pwdfile
function generate_pam_userdb() {
# Note that the path to the database file should be specified without the .db suffix
cat > "$1" << EOF
auth required pam_userdb.so db=$2 crypt=crypt
account required pam_permit.so
EOF
}
# args: file interface port
function generate_config_v11() {
cat > "$1" << EOF
internal: $2 port=$3
external: $2
method: pam
user.privileged: root
user.notprivileged: nobody
client pass {
from: 0.0.0.0/0 to: 0.0.0.0/0
log: error
}
# deny proxied to loopback
block {
from: 0.0.0.0/0 to: 127.0.0.0/8
log: error
}
pass {
from: 0.0.0.0/0 to: 0.0.0.0/0
log: error
}
EOF
}
# args: file interface port
function generate_config_v14() {
cat > "$1" <<EOF
# https://www.inet.no/dante/doc/1.4.x/config/ipv6.html
internal.protocol: ipv4 ipv6
internal: $2 port=$3
external.protocol: ipv4 ipv6
external: $2
socksmethod: pam.any
user.privileged: root
user.notprivileged: nobody
client pass {
from: 0.0.0.0/0 to: 0.0.0.0/0
log: error
}
client pass {
from: ::/0 to: ::/0
log: error
}
# deny proxied to loopback
socks block {
from: 0.0.0.0/0 to: 127.0.0.0/8
log: error
}
socks block {
from: ::/0 to: ::1/128
log: error
}
socks pass {
from: 0.0.0.0/0 to: 0.0.0.0/0
log: error
}
EOF
}
# args: file interface port
function generate_systemd_file() {
cat > "$1" <<EOF
# /etc/systemd/system/sockd.service
[Unit]
Description=Dante Socks5 Daemon
After=network.target
[Service]
Type=forking
PIDFile=/var/run/sockd.pid
ExecStart=/usr/sbin/sockd -D -f /etc/sockd.conf
ExecReload=/bin/kill -HUP \${MAINPID}
KillMode=process
Restart=on-failure
[Install]
WantedBy=multi-user.target
Alias=danted.service
EOF
}
# args: port
function open_ufw_port() {
# Open port in firewall if required
if which ufw > /dev/null; then
ufw allow "$PORT"/tcp
fi
}
# args: port
function open_firewalld_port() {
# Open port in firewall if required
if which firewall-cmd > /dev/null; then
firewall-cmd --zone=public --permanent --add-port="$1"/tcp
firewall-cmd --reload
fi
}
IFACE=$(ip route get 8.8.8.8 | head -1 | cut -d' ' -f5)
[ -z "$USER" ] && export USER=user
[ -z "$PORT" ] && export PORT=8080
[ -z "$PASSWORD" ] && export PASSWORD=$( cat /dev/urandom | tr --delete --complement 'a-z0-9' | head --bytes=10 )
[ -e /etc/lsb-release ] && source /etc/lsb-release
[ -e /etc/os-release ] && source /etc/os-release
# Ubuntu 16.06 Xenial
# Ubuntu 18.04 Bionic
# Ubuntu 20.04 Focal
# Ubuntu 22.04 Jammy
# Ubuntu 23.04 Lunar
if [ "$DISTRIB_ID" = "Ubuntu" ]; then
apt update
apt install -y dante-server libpam-pwdfile openssl
generate_password_file /etc/danted.passwd "$USER" "$PASSWORD"
generate_pam /etc/pam.d/sockd /etc/danted.passwd
generate_config_v14 /etc/danted.conf "$IFACE" "$PORT"
open_ufw_port "$PORT"
systemctl restart danted.service
echo ""
echo "Your socks proxy configuration:"
echo "Address: $( get_external_address )"
echo "Port: $PORT"
echo "User: $USER"
echo "Password: $PASSWORD"
echo ""
echo "On another host, test with:"
echo " curl -x socks5://$USER:$PASSWORD@$( get_external_address ):$PORT ipinfo.io"
echo ""
# CentOS 7 and Oracle Linux 7.5
elif [ "$ID $VERSION_ID" = "ol 7.5" -o "$ID $VERSION_ID" = "centos 7" ]; then
DANTE_TGZ="tgz-prod.dante-1.4.2-rhel72-amd64-64bit-gcc.tar.gz"
curl --progress-bar -O https://www.inet.no/dante/sslfiles/dante-1.4.2/"$DANTE_TGZ"
tar -C / -xzf "$DANTE_TGZ"
yum install -q -y openssl which bind-utils
generate_password_file /etc/danted.passwd "$USER" "$PASSWORD"
generate_password_dbfile /etc/danted.passwd
generate_pam_userdb /etc/pam.d/sockd /etc/danted.passwd
generate_config_v14 /etc/sockd.conf "$IFACE" "$PORT"
open_firewalld_port "$PORT"
generate_systemd_file /etc/systemd/system/sockd.service
systemctl daemon-reload
systemctl enable sockd.service
systemctl restart sockd.service
echo "Your socks proxy configuration:"
echo "Address: $( get_external_address )"
echo "Port: $PORT"
echo "User: $USER"
echo "Password: $PASSWORD"
else
echo "Sorry, this distribution ($DISTRIB_ID $DISTRIB_CODENAME) is not supported"
echo ""
echo "Supported distributions:"
echo "- Ubuntu 16.04 Xenial"
echo "- Ubuntu 18.04 Bionic"
echo "- Ubuntu 20.04 Focal"
echo "- Ubuntu 22.04 Jammy"
echo "- Ubuntu 23.04 Lunar"
echo "- Oracle Linux 7.5"
echo "- Centos 7"
exit 1
fi
@cvladan
Copy link
Author

cvladan commented May 31, 2023

Pay attention. By default, this uses ufw for Ubuntu, but on Oracle Cloud, you should use firewall-cmd.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment