Git Product home page Git Product logo

frp's Introduction

frp

Build Status GitHub release Go Report Card GitHub Releases Stats

README | 中文文档

Gold Sponsors

What is frp?

frp is a fast reverse proxy that allows you to expose a local server located behind a NAT or firewall to the Internet. It currently supports TCP and UDP, as well as HTTP and HTTPS protocols, enabling requests to be forwarded to internal services via domain name.

frp also offers a P2P connect mode.

Table of Contents

Development Status

frp is currently under development. You can try the latest release version in the master branch, or use the dev branch to access the version currently in development.

We are currently working on version 2 and attempting to perform some code refactoring and improvements. However, please note that it will not be compatible with version 1.

We will transition from version 0 to version 1 at the appropriate time and will only accept bug fixes and improvements, rather than big feature requests.

About V2

The complexity and difficulty of the v2 version are much higher than anticipated. I can only work on its development during fragmented time periods, and the constant interruptions disrupt productivity significantly. Given this situation, we will continue to optimize and iterate on the current version until we have more free time to proceed with the major version overhaul.

The concept behind v2 is based on my years of experience and reflection in the cloud-native domain, particularly in K8s and ServiceMesh. Its core is a modernized four-layer and seven-layer proxy, similar to envoy. This proxy itself is highly scalable, not only capable of implementing the functionality of intranet penetration but also applicable to various other domains. Building upon this highly scalable core, we aim to implement all the capabilities of frp v1 while also addressing the functionalities that were previously unachievable or difficult to implement in an elegant manner. Furthermore, we will maintain efficient development and iteration capabilities.

In addition, I envision frp itself becoming a highly extensible system and platform, similar to how we can provide a range of extension capabilities based on K8s. In K8s, we can customize development according to enterprise needs, utilizing features such as CRD, controller mode, webhook, CSI, and CNI. In frp v1, we introduced the concept of server plugins, which implemented some basic extensibility. However, it relies on a simple HTTP protocol and requires users to start independent processes and manage them on their own. This approach is far from flexible and convenient, and real-world demands vary greatly. It is unrealistic to expect a non-profit open-source project maintained by a few individuals to meet everyone's needs.

Finally, we acknowledge that the current design of modules such as configuration management, permission verification, certificate management, and API management is not modern enough. While we may carry out some optimizations in the v1 version, ensuring compatibility remains a challenging issue that requires a considerable amount of effort to address.

We sincerely appreciate your support for frp.

Architecture

architecture

Example Usage

To begin, download the latest program for your operating system and architecture from the Release page.

Next, place the frps binary and server configuration file on Server A, which has a public IP address.

Finally, place the frpc binary and client configuration file on Server B, which is located on a LAN that cannot be directly accessed from the public internet.

Some antiviruses improperly mark frpc as malware and delete it. This is due to frp being a networking tool capable of creating reverse proxies. Antiviruses sometimes flag reverse proxies due to their ability to bypass firewall port restrictions. If you are using antivirus, then you may need to whitelist/exclude frpc in your antivirus settings to avoid accidental quarantine/deletion. See issue 3637 for more details.

Access your computer in a LAN network via SSH

  1. Modify frps.toml on server A by setting the bindPort for frp clients to connect to:
# frps.toml
bindPort = 7000
  1. Start frps on server A:

./frps -c ./frps.toml

  1. Modify frpc.toml on server B and set the serverAddr field to the public IP address of your frps server:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "ssh"
type = "tcp"
localIP = "127.0.0.1"
localPort = 22
remotePort = 6000

Note that the localPort (listened on the client) and remotePort (exposed on the server) are used for traffic going in and out of the frp system, while the serverPort is used for communication between frps and frpc.

  1. Start frpc on server B:

./frpc -c ./frpc.toml

  1. To access server B from another machine through server A via SSH (assuming the username is test), use the following command:

ssh -oPort=6000 [email protected]

Multiple SSH services sharing the same port

This example implements multiple SSH services exposed through the same port using a proxy of type tcpmux. Similarly, as long as the client supports the HTTP Connect proxy connection method, port reuse can be achieved in this way.

  1. Deploy frps on a machine with a public IP and modify the frps.toml file. Here is a simplified configuration:
bindPort = 7000
tcpmuxHTTPConnectPort = 5002
  1. Deploy frpc on the internal machine A with the following configuration:
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "ssh1"
type = "tcpmux"
multiplexer = "httpconnect"
customDomains = ["machine-a.example.com"]
localIP = "127.0.0.1"
localPort = 22
  1. Deploy another frpc on the internal machine B with the following configuration:
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "ssh2"
type = "tcpmux"
multiplexer = "httpconnect"
customDomains = ["machine-b.example.com"]
localIP = "127.0.0.1"
localPort = 22
  1. To access internal machine A using SSH ProxyCommand, assuming the username is "test":

ssh -o 'proxycommand socat - PROXY:x.x.x.x:%h:%p,proxyport=5002' [email protected]

  1. To access internal machine B, the only difference is the domain name, assuming the username is "test":

ssh -o 'proxycommand socat - PROXY:x.x.x.x:%h:%p,proxyport=5002' [email protected]

Accessing Internal Web Services with Custom Domains in LAN

Sometimes we need to expose a local web service behind a NAT network to others for testing purposes with our own domain name.

Unfortunately, we cannot resolve a domain name to a local IP. However, we can use frp to expose an HTTP(S) service.

  1. Modify frps.toml and set the HTTP port for vhost to 8080:
# frps.toml
bindPort = 7000
vhostHTTPPort = 8080

If you want to configure an https proxy, you need to set up the vhostHTTPSPort.

  1. Start frps:

./frps -c ./frps.toml

  1. Modify frpc.toml and set serverAddr to the IP address of the remote frps server. Specify the localPort of your web service:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "web"
type = "http"
localPort = 80
customDomains = ["www.example.com"]
  1. Start frpc:

./frpc -c ./frpc.toml

  1. Map the A record of www.example.com to either the public IP of the remote frps server or a CNAME record pointing to your original domain.

  2. Visit your local web service using url http://www.example.com:8080.

Forward DNS query requests

  1. Modify frps.toml:
# frps.toml
bindPort = 7000
  1. Start frps:

./frps -c ./frps.toml

  1. Modify frpc.toml and set serverAddr to the IP address of the remote frps server. Forward DNS query requests to the Google Public DNS server 8.8.8.8:53:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "dns"
type = "udp"
localIP = "8.8.8.8"
localPort = 53
remotePort = 6000
  1. Start frpc:

./frpc -c ./frpc.toml

  1. Test DNS resolution using the dig command:

dig @x.x.x.x -p 6000 www.google.com

Forward Unix Domain Socket

Expose a Unix domain socket (e.g. the Docker daemon socket) as TCP.

Configure frps as above.

  1. Start frpc with the following configuration:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "unix_domain_socket"
type = "tcp"
remotePort = 6000
[proxies.plugin]
type = "unix_domain_socket"
unixPath = "/var/run/docker.sock"
  1. Test the configuration by getting the docker version using curl:

curl http://x.x.x.x:6000/version

Expose a simple HTTP file server

Expose a simple HTTP file server to access files stored in the LAN from the public Internet.

Configure frps as described above, then:

  1. Start frpc with the following configuration:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "test_static_file"
type = "tcp"
remotePort = 6000
[proxies.plugin]
type = "static_file"
localPath = "/tmp/files"
stripPrefix = "static"
httpUser = "abc"
httpPassword = "abc"
  1. Visit http://x.x.x.x:6000/static/ from your browser and specify correct username and password to view files in /tmp/files on the frpc machine.

Enable HTTPS for a local HTTP(S) service

You may substitute https2https for the plugin, and point the localAddr to a HTTPS endpoint.

  1. Start frpc with the following configuration:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "test_https2http"
type = "https"
customDomains = ["test.example.com"]

[proxies.plugin]
type = "https2http"
localAddr = "127.0.0.1:80"
crtPath = "./server.crt"
keyPath = "./server.key"
hostHeaderRewrite = "127.0.0.1"
requestHeaders.set.x-from-where = "frp"
  1. Visit https://test.example.com.

Expose your service privately

To mitigate risks associated with exposing certain services directly to the public network, STCP (Secret TCP) mode requires a preshared key to be used for access to the service from other clients.

Configure frps same as above.

  1. Start frpc on machine B with the following config. This example is for exposing the SSH service (port 22), and note the secretKey field for the preshared key, and that the remotePort field is removed here:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "secret_ssh"
type = "stcp"
secretKey = "abcdefg"
localIP = "127.0.0.1"
localPort = 22
  1. Start another frpc (typically on another machine C) with the following config to access the SSH service with a security key (secretKey field):
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[visitors]]
name = "secret_ssh_visitor"
type = "stcp"
serverName = "secret_ssh"
secretKey = "abcdefg"
bindAddr = "127.0.0.1"
bindPort = 6000
  1. On machine C, connect to SSH on machine B, using this command:

ssh -oPort=6000 127.0.0.1

P2P Mode

xtcp is designed to transmit large amounts of data directly between clients. A frps server is still needed, as P2P here only refers to the actual data transmission.

Note that it may not work with all types of NAT devices. You might want to fallback to stcp if xtcp doesn't work.

  1. Start frpc on machine B, and expose the SSH port. Note that the remotePort field is removed:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000
# set up a new stun server if the default one is not available.
# natHoleStunServer = "xxx"

[[proxies]]
name = "p2p_ssh"
type = "xtcp"
secretKey = "abcdefg"
localIP = "127.0.0.1"
localPort = 22
  1. Start another frpc (typically on another machine C) with the configuration to connect to SSH using P2P mode:
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000
# set up a new stun server if the default one is not available.
# natHoleStunServer = "xxx"

[[visitors]]
name = "p2p_ssh_visitor"
type = "xtcp"
serverName = "p2p_ssh"
secretKey = "abcdefg"
bindAddr = "127.0.0.1"
bindPort = 6000
# when automatic tunnel persistence is required, set it to true
keepTunnelOpen = false
  1. On machine C, connect to SSH on machine B, using this command:

ssh -oPort=6000 127.0.0.1

Features

Configuration Files

Since v0.52.0, we support TOML, YAML, and JSON for configuration. Please note that INI is deprecated and will be removed in future releases. New features will only be available in TOML, YAML, or JSON. Users wanting these new features should switch their configuration format accordingly.

Read the full example configuration files to find out even more features not described here.

Examples use TOML format, but you can still use YAML or JSON.

These configuration files is for reference only. Please do not use this configuration directly to run the program as it may have various issues.

Full configuration file for frps (Server)

Full configuration file for frpc (Client)

Using Environment Variables

Environment variables can be referenced in the configuration file, using Go's standard format:

# frpc.toml
serverAddr = "{{ .Envs.FRP_SERVER_ADDR }}"
serverPort = 7000

[[proxies]]
name = "ssh"
type = "tcp"
localIP = "127.0.0.1"
localPort = 22
remotePort = "{{ .Envs.FRP_SSH_REMOTE_PORT }}"

With the config above, variables can be passed into frpc program like this:

export FRP_SERVER_ADDR=x.x.x.x
export FRP_SSH_REMOTE_PORT=6000
./frpc -c ./frpc.toml

frpc will render configuration file template using OS environment variables. Remember to prefix your reference with .Envs.

Split Configures Into Different Files

You can split multiple proxy configs into different files and include them in the main file.

# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000
includes = ["./confd/*.toml"]
# ./confd/test.toml

[[proxies]]
name = "ssh"
type = "tcp"
localIP = "127.0.0.1"
localPort = 22
remotePort = 6000

Server Dashboard

Check frp's status and proxies' statistics information by Dashboard.

Configure a port for dashboard to enable this feature:

# The default value is 127.0.0.1. Change it to 0.0.0.0 when you want to access it from a public network.
webServer.addr = "0.0.0.0"
webServer.port = 7500
# dashboard's username and password are both optional
webServer.user = "admin"
webServer.password = "admin"

Then visit http://[serverAddr]:7500 to see the dashboard, with username and password both being admin.

Additionally, you can use HTTPS port by using your domains wildcard or normal SSL certificate:

webServer.port = 7500
# dashboard's username and password are both optional
webServer.user = "admin"
webServer.password = "admin"
webServer.tls.certFile = "server.crt"
webServer.tls.keyFile = "server.key"

Then visit https://[serverAddr]:7500 to see the dashboard in secure HTTPS connection, with username and password both being admin.

dashboard

Client Admin UI

The Client Admin UI helps you check and manage frpc's configuration.

Configure an address for admin UI to enable this feature:

webServer.addr = "127.0.0.1"
webServer.port = 7400
webServer.user = "admin"
webServer.password = "admin"

Then visit http://127.0.0.1:7400 to see admin UI, with username and password both being admin.

Monitor

When web server is enabled, frps will save monitor data in cache for 7 days. It will be cleared after process restart.

Prometheus is also supported.

Prometheus

Enable dashboard first, then configure enablePrometheus = true in frps.toml.

http://{dashboard_addr}/metrics will provide prometheus monitor data.

Authenticating the Client

There are 2 authentication methods to authenticate frpc with frps.

You can decide which one to use by configuring auth.method in frpc.toml and frps.toml, the default one is token.

Configuring auth.additionalScopes = ["HeartBeats"] will use the configured authentication method to add and validate authentication on every heartbeat between frpc and frps.

Configuring auth.additionalScopes = ["NewWorkConns"] will do the same for every new work connection between frpc and frps.

Token Authentication

When specifying auth.method = "token" in frpc.toml and frps.toml - token based authentication will be used.

Make sure to specify the same auth.token in frps.toml and frpc.toml for frpc to pass frps validation

OIDC Authentication

When specifying auth.method = "oidc" in frpc.toml and frps.toml - OIDC based authentication will be used.

OIDC stands for OpenID Connect, and the flow used is called Client Credentials Grant.

To use this authentication type - configure frpc.toml and frps.toml as follows:

# frps.toml
auth.method = "oidc"
auth.oidc.issuer = "https://example-oidc-issuer.com/"
auth.oidc.audience = "https://oidc-audience.com/.default"
# frpc.toml
auth.method = "oidc"
auth.oidc.clientID = "98692467-37de-409a-9fac-bb2585826f18" # Replace with OIDC client ID
auth.oidc.clientSecret = "oidc_secret"
auth.oidc.audience = "https://oidc-audience.com/.default"
auth.oidc.tokenEndpointURL = "https://example-oidc-endpoint.com/oauth2/v2.0/token"

Encryption and Compression

The features are off by default. You can turn on encryption and/or compression:

# frpc.toml

[[proxies]]
name = "ssh"
type = "tcp"
localPort = 22
remotePort = 6000
transport.useEncryption = true
transport.useCompression = true

TLS

Since v0.50.0, the default value of transport.tls.enable and transport.tls.disableCustomTLSFirstByte has been changed to true, and tls is enabled by default.

For port multiplexing, frp sends a first byte 0x17 to dial a TLS connection. This only takes effect when you set transport.tls.disableCustomTLSFirstByte to false.

To enforce frps to only accept TLS connections - configure transport.tls.force = true in frps.toml. This is optional.

frpc TLS settings:

transport.tls.enable = true
transport.tls.certFile = "certificate.crt"
transport.tls.keyFile = "certificate.key"
transport.tls.trustedCaFile = "ca.crt"

frps TLS settings:

transport.tls.force = true
transport.tls.certFile = "certificate.crt"
transport.tls.keyFile = "certificate.key"
transport.tls.trustedCaFile = "ca.crt"

You will need a root CA cert and at least one SSL/TLS certificate. It can be self-signed or regular (such as Let's Encrypt or another SSL/TLS certificate provider).

If you using frp via IP address and not hostname, make sure to set the appropriate IP address in the Subject Alternative Name (SAN) area when generating SSL/TLS Certificates.

Given an example:

  • Prepare openssl config file. It exists at /etc/pki/tls/openssl.cnf in Linux System and /System/Library/OpenSSL/openssl.cnf in MacOS, and you can copy it to current path, like cp /etc/pki/tls/openssl.cnf ./my-openssl.cnf. If not, you can build it by yourself, like:
cat > my-openssl.cnf << EOF
[ ca ]
default_ca = CA_default
[ CA_default ]
x509_extensions = usr_cert
[ req ]
default_bits        = 2048
default_md          = sha256
default_keyfile     = privkey.pem
distinguished_name  = req_distinguished_name
attributes          = req_attributes
x509_extensions     = v3_ca
string_mask         = utf8only
[ req_distinguished_name ]
[ req_attributes ]
[ usr_cert ]
basicConstraints       = CA:FALSE
nsComment              = "OpenSSL Generated Certificate"
subjectKeyIdentifier   = hash
authorityKeyIdentifier = keyid,issuer
[ v3_ca ]
subjectKeyIdentifier   = hash
authorityKeyIdentifier = keyid:always,issuer
basicConstraints       = CA:true
EOF
  • build ca certificates:
openssl genrsa -out ca.key 2048
openssl req -x509 -new -nodes -key ca.key -subj "/CN=example.ca.com" -days 5000 -out ca.crt
  • build frps certificates:
openssl genrsa -out server.key 2048

openssl req -new -sha256 -key server.key \
    -subj "/C=XX/ST=DEFAULT/L=DEFAULT/O=DEFAULT/CN=server.com" \
    -reqexts SAN \
    -config <(cat my-openssl.cnf <(printf "\n[SAN]\nsubjectAltName=DNS:localhost,IP:127.0.0.1,DNS:example.server.com")) \
    -out server.csr

openssl x509 -req -days 365 -sha256 \
	-in server.csr -CA ca.crt -CAkey ca.key -CAcreateserial \
	-extfile <(printf "subjectAltName=DNS:localhost,IP:127.0.0.1,DNS:example.server.com") \
	-out server.crt
  • build frpc certificates:
openssl genrsa -out client.key 2048
openssl req -new -sha256 -key client.key \
    -subj "/C=XX/ST=DEFAULT/L=DEFAULT/O=DEFAULT/CN=client.com" \
    -reqexts SAN \
    -config <(cat my-openssl.cnf <(printf "\n[SAN]\nsubjectAltName=DNS:client.com,DNS:example.client.com")) \
    -out client.csr

openssl x509 -req -days 365 -sha256 \
    -in client.csr -CA ca.crt -CAkey ca.key -CAcreateserial \
	-extfile <(printf "subjectAltName=DNS:client.com,DNS:example.client.com") \
	-out client.crt

Hot-Reloading frpc configuration

The webServer fields are required for enabling HTTP API:

# frpc.toml
webServer.addr = "127.0.0.1"
webServer.port = 7400

Then run command frpc reload -c ./frpc.toml and wait for about 10 seconds to let frpc create or update or remove proxies.

Note that global client parameters won't be modified except 'start'.

You can run command frpc verify -c ./frpc.toml before reloading to check if there are config errors.

Get proxy status from client

Use frpc status -c ./frpc.toml to get status of all proxies. The webServer fields are required for enabling HTTP API.

Only allowing certain ports on the server

allowPorts in frps.toml is used to avoid abuse of ports:

# frps.toml
allowPorts = [
  { start = 2000, end = 3000 },
  { single = 3001 },
  { single = 3003 },
  { start = 4000, end = 50000 }
]

Port Reuse

vhostHTTPPort and vhostHTTPSPort in frps can use same port with bindPort. frps will detect the connection's protocol and handle it correspondingly.

What you need to pay attention to is that if you want to configure vhostHTTPSPort and bindPort to the same port, you need to first set transport.tls.disableCustomTLSFirstByte to false.

We would like to try to allow multiple proxies bind a same remote port with different protocols in the future.

Bandwidth Limit

For Each Proxy

# frpc.toml

[[proxies]]
name = "ssh"
type = "tcp"
localPort = 22
remotePort = 6000
transport.bandwidthLimit = "1MB"

Set transport.bandwidthLimit in each proxy's configure to enable this feature. Supported units are MB and KB.

Set transport.bandwidthLimitMode to client or server to limit bandwidth on the client or server side. Default is client.

TCP Stream Multiplexing

frp supports tcp stream multiplexing since v0.10.0 like HTTP2 Multiplexing, in which case all logic connections to the same frpc are multiplexed into the same TCP connection.

You can disable this feature by modify frps.toml and frpc.toml:

# frps.toml and frpc.toml, must be same
transport.tcpMux = false

Support KCP Protocol

KCP is a fast and reliable protocol that can achieve the transmission effect of a reduction of the average latency by 30% to 40% and reduction of the maximum delay by a factor of three, at the cost of 10% to 20% more bandwidth wasted than TCP.

KCP mode uses UDP as the underlying transport. Using KCP in frp:

  1. Enable KCP in frps:
# frps.toml
bindPort = 7000
# Specify a UDP port for KCP.
kcpBindPort = 7000

The kcpBindPort number can be the same number as bindPort, since bindPort field specifies a TCP port.

  1. Configure frpc.toml to use KCP to connect to frps:
# frpc.toml
serverAddr = "x.x.x.x"
# Same as the 'kcpBindPort' in frps.toml
serverPort = 7000
transport.protocol = "kcp"

Support QUIC Protocol

QUIC is a new multiplexed transport built on top of UDP.

Using QUIC in frp:

  1. Enable QUIC in frps:
# frps.toml
bindPort = 7000
# Specify a UDP port for QUIC.
quicBindPort = 7000

The quicBindPort number can be the same number as bindPort, since bindPort field specifies a TCP port.

  1. Configure frpc.toml to use QUIC to connect to frps:
# frpc.toml
serverAddr = "x.x.x.x"
# Same as the 'quicBindPort' in frps.toml
serverPort = 7000
transport.protocol = "quic"

Connection Pooling

By default, frps creates a new frpc connection to the backend service upon a user request. With connection pooling, frps keeps a certain number of pre-established connections, reducing the time needed to establish a connection.

This feature is suitable for a large number of short connections.

  1. Configure the limit of pool count each proxy can use in frps.toml:
# frps.toml
transport.maxPoolCount = 5
  1. Enable and specify the number of connection pool:
# frpc.toml
transport.poolCount = 1

Load balancing

Load balancing is supported by group.

This feature is only available for types tcp, http, tcpmux now.

# frpc.toml

[[proxies]]
name = "test1"
type = "tcp"
localPort = 8080
remotePort = 80
loadBalancer.group = "web"
loadBalancer.groupKey = "123"

[[proxies]]
name = "test2"
type = "tcp"
localPort = 8081
remotePort = 80
loadBalancer.group = "web"
loadBalancer.groupKey = "123"

loadBalancer.groupKey is used for authentication.

Connections to port 80 will be dispatched to proxies in the same group randomly.

For type tcp, remotePort in the same group should be the same.

For type http, customDomains, subdomain, locations should be the same.

Service Health Check

Health check feature can help you achieve high availability with load balancing.

Add healthCheck.type = "tcp" or healthCheck.type = "http" to enable health check.

With health check type tcp, the service port will be pinged (TCPing):

# frpc.toml

[[proxies]]
name = "test1"
type = "tcp"
localPort = 22
remotePort = 6000
# Enable TCP health check
healthCheck.type = "tcp"
# TCPing timeout seconds
healthCheck.timeoutSeconds = 3
# If health check failed 3 times in a row, the proxy will be removed from frps
healthCheck.maxFailed = 3
# A health check every 10 seconds
healthCheck.intervalSeconds = 10

With health check type http, an HTTP request will be sent to the service and an HTTP 2xx OK response is expected:

# frpc.toml

[[proxies]]
name = "web"
type = "http"
localIP = "127.0.0.1"
localPort = 80
customDomains = ["test.example.com"]
# Enable HTTP health check
healthCheck.type = "http"
# frpc will send a GET request to '/status'
# and expect an HTTP 2xx OK response
healthCheck.path = "/status"
healthCheck.timeoutSeconds = 3
healthCheck.maxFailed = 3
healthCheck.intervalSeconds = 10

Rewriting the HTTP Host Header

By default frp does not modify the tunneled HTTP requests at all as it's a byte-for-byte copy.

However, speaking of web servers and HTTP requests, your web server might rely on the Host HTTP header to determine the website to be accessed. frp can rewrite the Host header when forwarding the HTTP requests, with the hostHeaderRewrite field:

# frpc.toml

[[proxies]]
name = "web"
type = "http"
localPort = 80
customDomains = ["test.example.com"]
hostHeaderRewrite = "dev.example.com"

The HTTP request will have the Host header rewritten to Host: dev.example.com when it reaches the actual web server, although the request from the browser probably has Host: test.example.com.

Setting other HTTP Headers

Similar to Host, You can override other HTTP request and response headers with proxy type http.

# frpc.toml

[[proxies]]
name = "web"
type = "http"
localPort = 80
customDomains = ["test.example.com"]
hostHeaderRewrite = "dev.example.com"
requestHeaders.set.x-from-where = "frp"
responseHeaders.set.foo = "bar"

In this example, it will set header x-from-where: frp in the HTTP request and foo: bar in the HTTP response.

Get Real IP

HTTP X-Forwarded-For

This feature is for http proxies or proxies with the https2http and https2https plugins enabled.

You can get user's real IP from HTTP request headers X-Forwarded-For.

Proxy Protocol

frp supports Proxy Protocol to send user's real IP to local services. It support all types except UDP.

Here is an example for https service:

# frpc.toml

[[proxies]]
name = "web"
type = "https"
localPort = 443
customDomains = ["test.example.com"]

# now v1 and v2 are supported
transport.proxyProtocolVersion = "v2"

You can enable Proxy Protocol support in nginx to expose user's real IP in HTTP header X-Real-IP, and then read X-Real-IP header in your web service for the real IP.

Require HTTP Basic Auth (Password) for Web Services

Anyone who can guess your tunnel URL can access your local web server unless you protect it with a password.

This enforces HTTP Basic Auth on all requests with the username and password specified in frpc's configure file.

It can only be enabled when proxy type is http.

# frpc.toml

[[proxies]]
name = "web"
type = "http"
localPort = 80
customDomains = ["test.example.com"]
httpUser = "abc"
httpPassword = "abc"

Visit http://test.example.com in the browser and now you are prompted to enter the username and password.

Custom Subdomain Names

It is convenient to use subdomain configure for http and https types when many people share one frps server.

# frps.toml
subDomainHost = "frps.com"

Resolve *.frps.com to the frps server's IP. This is usually called a Wildcard DNS record.

# frpc.toml

[[proxies]]
name = "web"
type = "http"
localPort = 80
subdomain = "test"

Now you can visit your web service on test.frps.com.

Note that if subdomainHost is not empty, customDomains should not be the subdomain of subdomainHost.

URL Routing

frp supports forwarding HTTP requests to different backend web services by url routing.

locations specifies the prefix of URL used for routing. frps first searches for the most specific prefix location given by literal strings regardless of the listed order.

# frpc.toml

[[proxies]]
name = "web01"
type = "http"
localPort = 80
customDomains = ["web.example.com"]
locations = ["/"]

[[proxies]]
name = "web02"
type = "http"
localPort = 81
customDomains = ["web.example.com"]
locations = ["/news", "/about"]

HTTP requests with URL prefix /news or /about will be forwarded to web02 and other requests to web01.

TCP Port Multiplexing

frp supports receiving TCP sockets directed to different proxies on a single port on frps, similar to vhostHTTPPort and vhostHTTPSPort.

The only supported TCP port multiplexing method available at the moment is httpconnect - HTTP CONNECT tunnel.

When setting tcpmuxHTTPConnectPort to anything other than 0 in frps, frps will listen on this port for HTTP CONNECT requests.

The host of the HTTP CONNECT request will be used to match the proxy in frps. Proxy hosts can be configured in frpc by configuring customDomains and / or subdomain under tcpmux proxies, when multiplexer = "httpconnect".

For example:

# frps.toml
bindPort = 7000
tcpmuxHTTPConnectPort = 1337
# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000

[[proxies]]
name = "proxy1"
type = "tcpmux"
multiplexer = "httpconnect"
customDomains = ["test1"]
localPort = 80

[[proxies]]
name = "proxy2"
type = "tcpmux"
multiplexer = "httpconnect"
customDomains = ["test2"]
localPort = 8080

In the above configuration - frps can be contacted on port 1337 with a HTTP CONNECT header such as:

CONNECT test1 HTTP/1.1\r\n\r\n

and the connection will be routed to proxy1.

Connecting to frps via PROXY

frpc can connect to frps through proxy if you set OS environment variable HTTP_PROXY, or if transport.proxyURL is set in frpc.toml file.

It only works when protocol is tcp.

# frpc.toml
serverAddr = "x.x.x.x"
serverPort = 7000
transport.proxyURL = "http://user:[email protected]:8080"

Port range mapping

Added in v0.56.0

We can use the range syntax of Go template combined with the built-in parseNumberRangePair function to achieve port range mapping.

The following example, when run, will create 8 proxies named test-6000, test-6001 ... test-6007, each mapping the remote port to the local port.

{{- range $_, $v := parseNumberRangePair "6000-6006,6007" "6000-6006,6007" }}
[[proxies]]
name = "tcp-{{ $v.First }}"
type = "tcp"
localPort = {{ $v.First }}
remotePort = {{ $v.Second }}
{{- end }}

Client Plugins

frpc only forwards requests to local TCP or UDP ports by default.

Plugins are used for providing rich features. There are built-in plugins such as unix_domain_socket, http_proxy, socks5, static_file, http2https, https2http, https2https and you can see example usage.

Using plugin http_proxy:

# frpc.toml

[[proxies]]
name = "http_proxy"
type = "tcp"
remotePort = 6000
[proxies.plugin]
type = "http_proxy"
httpUser = "abc"
httpPassword = "abc"

httpUser and httpPassword are configuration parameters used in http_proxy plugin.

Server Manage Plugins

Read the document.

Find more plugins in gofrp/plugin.

SSH Tunnel Gateway

added in v0.53.0

frp supports listening to an SSH port on the frps side and achieves TCP protocol proxying through the SSH -R protocol, without relying on frpc.

# frps.toml
sshTunnelGateway.bindPort = 2200

When running ./frps -c frps.toml, a private key file named .autogen_ssh_key will be automatically created in the current working directory. This generated private key file will be used by the SSH server in frps.

Executing the command

ssh -R :80:127.0.0.1:8080 v0@{frp address} -p 2200 tcp --proxy_name "test-tcp" --remote_port 9090

sets up a proxy on frps that forwards the local 8080 service to the port 9090.

frp (via SSH) (Ctrl+C to quit)

User:
ProxyName: test-tcp
Type: tcp
RemoteAddress: :9090

This is equivalent to:

frpc tcp --proxy_name "test-tcp" --local_ip 127.0.0.1 --local_port 8080 --remote_port 9090

Please refer to this document for more information.

Releated Projects

  • gofrp/plugin - A repository for frp plugins that contains a variety of plugins implemented based on the frp extension mechanism, meeting the customization needs of different scenarios.
  • gofrp/tiny-frpc - A lightweight version of the frp client (around 3.5MB at minimum) implemented using the ssh protocol, supporting some of the most commonly used features, suitable for devices with limited resources.

Contributing

Interested in getting involved? We would like to help you!

  • Take a look at our issues list and consider sending a Pull Request to dev branch.
  • If you want to add a new feature, please create an issue first to describe the new feature, as well as the implementation approach. Once a proposal is accepted, create an implementation of the new features and submit it as a pull request.
  • Sorry for my poor English. Improvements for this document are welcome, even some typo fixes.
  • If you have great ideas, send an email to [email protected].

Note: We prefer you to give your advise in issues, so others with a same question can search it quickly and we don't need to answer them repeatedly.

Donation

If frp helps you a lot, you can support us by:

GitHub Sponsors

Support us by Github Sponsors.

You can have your company's logo placed on README file of this project.

PayPal

Donate money by PayPal to my account [email protected].

frp's People

Contributors

akx avatar arugal avatar becods avatar bgkavinga avatar bingtianbaihua avatar blizard863 avatar cuishuang avatar ddscentral avatar ericlarssen avatar fakeboboliu avatar fatedier avatar fsyxhua avatar fzhyzamt avatar glzjin avatar hal-wang avatar hurricanezwf avatar im-zhou avatar jiajunhuang avatar koho avatar lonwern avatar maodanp avatar moul avatar muesli avatar pkgplus avatar vashstorm avatar velovix avatar wlynxg avatar wuqinqiang avatar yuyulei avatar zalberth avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

frp's Issues

启动frpc时frps报错并停止运行

报错信息如下

panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xb code=0x1 addr=0x0 pc=0x80df603]

goroutine 8 [running]:
frp/models/server.(*ProxyServer).Start(0x189254a0, 0x1898c060, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/models/server/server.go:103 +0x3d3
main.doLogin(0x1898c0f0, 0x1898c060, 0x1, 0x0, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:231 +0x64b
main.controlWorker(0x1898c060)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:71 +0x471
created by main.ProcessControlConn
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:38 +0x11b

goroutine 1 [chan receive]:
frp/utils/conn.(*Listener).Accept(0x1898a460, 0x84c695c, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:75 +0x4a
main.ProcessControlConn(0x1898a460)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:33 +0x2f
main.main()
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/main.go:162 +0x1217

goroutine 5 [IO wait]:
net.runtime_pollWait(0xb7618500, 0x72, 0x1897f180)
    /usr/lib/golang/src/runtime/netpoll.go:157 +0x55
net.(*pollDesc).Wait(0x18967438, 0x72, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:73 +0x35
net.(*pollDesc).WaitRead(0x18967438, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:78 +0x33
net.(*netFD).accept(0x18967400, 0x0, 0xb7618618, 0x1897f190)
    /usr/lib/golang/src/net/fd_unix.go:408 +0x20b
net.(*TCPListener).AcceptTCP(0x18827a70, 0x189674c0, 0x0, 0x0)
    /usr/lib/golang/src/net/tcpsock_posix.go:254 +0x47
frp/utils/conn.Listen.func1(0x18827a58)
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:53 +0x34
created by frp/utils/conn.Listen
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:68 +0x2b8

goroutine 6 [IO wait]:
net.runtime_pollWait(0xb7618488, 0x72, 0x1897f0c0)
    /usr/lib/golang/src/runtime/netpoll.go:157 +0x55
net.(*pollDesc).Wait(0x189675b8, 0x72, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:73 +0x35
net.(*pollDesc).WaitRead(0x189675b8, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:78 +0x33
net.(*netFD).accept(0x18967580, 0x0, 0xb7618618, 0x1897f0d0)
    /usr/lib/golang/src/net/fd_unix.go:408 +0x20b
net.(*TCPListener).AcceptTCP(0x18827ac8, 0x8093300, 0x0, 0x0)
    /usr/lib/golang/src/net/tcpsock_posix.go:254 +0x47
net/http.tcpKeepAliveListener.Accept(0x18827ac8, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:2135 +0x3d
net/http.(*Server).Serve(0x18967540, 0xb76185f8, 0x18827ac8, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:1887 +0x98
net/http.(*Server).ListenAndServe(0x18967540, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:1877 +0xe4
net/http.ListenAndServe(0x1897f060, 0xc, 0xb76185c0, 0x18900f50, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:1967 +0x7a
github.com/gin-gonic/gin.(*Engine).Run(0x18900f50, 0x18827a98, 0x1, 0x1, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/Godeps/_workspace/src/github.com/gin-gonic/gin/gin.go:226 +0x15d
created by frp/models/server.RunDashboardServer
    /root/src/github.com/fatedier/frp/src/frp/models/server/dashboard.go:33 +0x28f

goroutine 7 [IO wait]:
net.runtime_pollWait(0xb7618410, 0x72, 0x1880e080)
    /usr/lib/golang/src/runtime/netpoll.go:157 +0x55
net.(*pollDesc).Wait(0x189676b8, 0x72, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:73 +0x35
net.(*pollDesc).WaitRead(0x189676b8, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:78 +0x33
net.(*netFD).Read(0x18967680, 0x1898e000, 0x1000, 0x1000, 0x0, 0xb7613030, 0x1880e080)
    /usr/lib/golang/src/net/fd_unix.go:232 +0x19a
net.(*conn).Read(0x18827ad0, 0x1898e000, 0x1000, 0x1000, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/net/net.go:172 +0xb9
bufio.(*Reader).fill(0x1898c030)
    /usr/lib/golang/src/bufio/bufio.go:97 +0x172
bufio.(*Reader).ReadSlice(0x1898c030, 0xa, 0x0, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/bufio/bufio.go:328 +0x1b4
bufio.(*Reader).ReadBytes(0x1898c030, 0xa, 0x0, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/bufio/bufio.go:406 +0x8c
bufio.(*Reader).ReadString(0x1898c030, 0x18824f0a, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/bufio/bufio.go:446 +0x46
frp/utils/conn.(*Conn).ReadLine(0x1898c000, 0x0, 0x0, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:132 +0x45
main.controlWorker(0x1898c000)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:57 +0x75
created by main.ProcessControlConn
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:38 +0x11b
zonek@ubuntu-512mb-sfo1-01:~/bin/frp$ ls^C
zonek@ubuntu-512mb-sfo1-01:~/bin/frp$ sudo ./frps_linux_386 -c frps.ini 
panic: runtime error: invalid memory address or nil pointer dereference
[signal 0xb code=0x1 addr=0x0 pc=0x80df603]

goroutine 7 [running]:
frp/models/server.(*ProxyServer).Start(0x189234a0, 0x18815fb0, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/models/server/server.go:103 +0x3d3
main.doLogin(0x1898c0c0, 0x18815fb0, 0x1, 0x0, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:231 +0x64b
main.controlWorker(0x18815fb0)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:71 +0x471
created by main.ProcessControlConn
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:38 +0x11b

goroutine 1 [chan receive]:
frp/utils/conn.(*Listener).Accept(0x1898a380, 0x84c695c, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:75 +0x4a
main.ProcessControlConn(0x1898a380)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:33 +0x2f
main.main()
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/main.go:162 +0x1217

goroutine 5 [IO wait]:
net.runtime_pollWait(0xb75ce508, 0x72, 0x1897f1b0)
    /usr/lib/golang/src/runtime/netpoll.go:157 +0x55
net.(*pollDesc).Wait(0x189673f8, 0x72, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:73 +0x35
net.(*pollDesc).WaitRead(0x189673f8, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:78 +0x33
net.(*netFD).accept(0x189673c0, 0x0, 0xb75ce620, 0x1897f1c0)
    /usr/lib/golang/src/net/fd_unix.go:408 +0x20b
net.(*TCPListener).AcceptTCP(0x18823a70, 0x18967480, 0x0, 0x0)
    /usr/lib/golang/src/net/tcpsock_posix.go:254 +0x47
frp/utils/conn.Listen.func1(0x18823a58)
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:53 +0x34
created by frp/utils/conn.Listen
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:68 +0x2b8

goroutine 6 [IO wait]:
net.runtime_pollWait(0xb75ce490, 0x72, 0x1897f0f0)
    /usr/lib/golang/src/runtime/netpoll.go:157 +0x55
net.(*pollDesc).Wait(0x18967578, 0x72, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:73 +0x35
net.(*pollDesc).WaitRead(0x18967578, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:78 +0x33
net.(*netFD).accept(0x18967540, 0x0, 0xb75ce620, 0x1897f100)
    /usr/lib/golang/src/net/fd_unix.go:408 +0x20b
net.(*TCPListener).AcceptTCP(0x18823ac8, 0x8093300, 0x0, 0x0)
    /usr/lib/golang/src/net/tcpsock_posix.go:254 +0x47
net/http.tcpKeepAliveListener.Accept(0x18823ac8, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:2135 +0x3d
net/http.(*Server).Serve(0x18967500, 0xb75ce600, 0x18823ac8, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:1887 +0x98
net/http.(*Server).ListenAndServe(0x18967500, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:1877 +0xe4
net/http.ListenAndServe(0x1897f090, 0xc, 0xb75ce5c8, 0x188fee70, 0x0, 0x0)
    /usr/lib/golang/src/net/http/server.go:1967 +0x7a
github.com/gin-gonic/gin.(*Engine).Run(0x188fee70, 0x18823a98, 0x1, 0x1, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/Godeps/_workspace/src/github.com/gin-gonic/gin/gin.go:226 +0x15d
created by frp/models/server.RunDashboardServer
    /root/src/github.com/fatedier/frp/src/frp/models/server/dashboard.go:33 +0x28f

goroutine 8 [IO wait]:
net.runtime_pollWait(0xb75ce3a0, 0x72, 0x1880e0b0)
    /usr/lib/golang/src/runtime/netpoll.go:157 +0x55
net.(*pollDesc).Wait(0x18967738, 0x72, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:73 +0x35
net.(*pollDesc).WaitRead(0x18967738, 0x0, 0x0)
    /usr/lib/golang/src/net/fd_poll_runtime.go:78 +0x33
net.(*netFD).Read(0x18967700, 0x1898f000, 0x1000, 0x1000, 0x0, 0xb75c9030, 0x1880e0b0)
    /usr/lib/golang/src/net/fd_unix.go:232 +0x19a
net.(*conn).Read(0x18823ae0, 0x1898f000, 0x1000, 0x1000, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/net/net.go:172 +0xb9
bufio.(*Reader).fill(0x1898c060)
    /usr/lib/golang/src/bufio/bufio.go:97 +0x172
bufio.(*Reader).ReadSlice(0x1898c060, 0xa, 0x0, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/bufio/bufio.go:328 +0x1b4
bufio.(*Reader).ReadBytes(0x1898c060, 0xa, 0x0, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/bufio/bufio.go:406 +0x8c
bufio.(*Reader).ReadString(0x1898c060, 0x1882170a, 0x0, 0x0, 0x0, 0x0)
    /usr/lib/golang/src/bufio/bufio.go:446 +0x46
frp/utils/conn.(*Conn).ReadLine(0x1898c030, 0x0, 0x0, 0x0, 0x0)
    /root/src/github.com/fatedier/frp/src/frp/utils/conn/conn.go:132 +0x45
main.controlWorker(0x1898c030)
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:57 +0x75
created by main.ProcessControlConn
    /root/src/github.com/fatedier/frp/src/frp/cmd/frps/control.go:38 +0x11b

重大BUG,严重的性能问题,内存泄露

用了一段时间,并通过验证,确认 FRP 有严重的性能问题,和 ngrok 一样,用一段时间后,服务器的性能会严重下降,最终导致 服务 crash 掉。我的服务器上就装了 nginx,gogs(go语言编写的一个git服务端),然后如果开了 frp ,不到三天就会服务器反应缓慢,frp 直接 crash 掉。我担心是 gogs 影响的,然后就单独开了 gogs ,结果跑了好几天,一致很稳定,所以确认是 frp 的代码问题,希望作者尽快解决看看,还是很喜欢 frp 的,但是如果这个问题不解决,就不敢用了。

frps --reload 选项没有作用

我从github上的release上下载frp_0.8.0_linux_amd64.tar.gz,发现--reload没有效果,具体是我输入命令之后会出现Usage的提示,具体如下

root@root# ./frps -c ./frps.ini --reload
Usage: 
    frps [-c config_file] [-L log_file] [--log-level=<log_level>] [--addr=<bind_addr>]
    frps --reload
    frps -h | --help
    frps -v | --version

多客户端privilege_mode=true好像没有效果

我在服务端启动了privilege_mode模式,在两台服务器上运行两个客户端,两个客户端的配置文件中privilege_token设置成和服务端的一样,remote_port不一样,但是出现了只能一台设备连接到服务器,另一台日志打印如下:
2016/08/22 19:30:52 [control.go:181][E] ProxyName [privilege_web], start proxy error, ProxyName [privilege_web], this proxy is already working now
2016/08/22 19:30:52 [control.go:39][E] ProxyName [privilege_web], connect to server failed!
提示被占用了,这个是bug么。

https 似乎无法正常工作

首先感谢开发者。
经测试ssh和80端口的http协议都正常工作。可是443端口的https协议似乎有问题。

客户端

# frpc.ini
[web02]
type = https
local_ip = 127.0.0.1
local_port = 443

[privilege_web]
privilege_mode = true
type = https
local_ip = 127.0.0.1
local_port = 443
use_gzip = true
custom_domains = 192.168.1.3
#web03.yourdomain.com
host_header_rewrite = 192.168.1.3
#example.com

服务端 frps.ini

vhost_http_port = 80
vhost_https_port = 443

[web02]
type = https
auth_token = 123
custom_domains = 144.172.70.47
#web02.yourdomain.com

尝试https连接似乎客户端和服务端都没有感知:
frpc.log

2016/08/22 15:05:47 [control.go:185][I] ProxyName [privilege_web], connect to server [144.172.70.47:7000] success!
2016/08/22 15:05:47 [control.go:193][I] Start to send heartbeat to frps
2016/08/22 15:05:47 [control.go:185][I] ProxyName [ssh], connect to server [144.172.70.47:7000] success!
2016/08/22 15:05:47 [control.go:193][I] Start to send heartbeat to frps
2016/08/22 15:05:47 [control.go:185][I] ProxyName [web01], connect to server [144.172.70.47:7000] success!
2016/08/22 15:05:47 [control.go:193][I] Start to send heartbeat to frps
2016/08/22 15:05:47 [control.go:185][I] ProxyName [privilege_ssh], connect to server [144.172.70.47:7000] success!
2016/08/22 15:05:47 [control.go:193][I] Start to send heartbeat to frps
2016/08/22 15:05:47 [control.go:185][I] ProxyName [web02], connect to server [144.172.70.47:7000] success!
2016/08/22 15:05:47 [control.go:193][I] Start to send heartbeat to frps

frps.log

2016/08/22 01:05:46 [main.go:182][I] Start frps success
2016/08/22 01:05:46 [main.go:184][I] PrivilegeMode is enabled, you should pay more attention to security issues
2016/08/22 01:05:52 [control.go:308][I] ProxyName [privilege_web], start proxy success
2016/08/22 01:05:52 [control.go:310][I] ProxyName [privilege_web], created by PrivilegeMode
2016/08/22 01:05:52 [control.go:308][I] ProxyName [ssh], start proxy success
2016/08/22 01:05:52 [control.go:308][I] ProxyName [web01], start proxy success
2016/08/22 01:05:52 [control.go:308][I] ProxyName [web02], start proxy success
2016/08/22 01:05:52 [control.go:308][I] ProxyName [privilege_ssh], start proxy success
2016/08/22 01:05:52 [control.go:310][I] ProxyName [privilege_ssh], created by PrivilegeMode

不知道domain那里我是不是写错了,或者问题出在我本地的路由器,有端口转发。可是http并未出问题。
谢谢

无法开机启动

添加到开机自启后 日志显示connection refused,但是直接执行却是正常的。

访问内网web不通过域名

你好,首先感谢提供这样的工具。
我之前是通过ssh隧道来把我内网的web应用映射出去,现在换成你这个工具,但是我没有域名,只想通过ip,即没有配置custom_domains,可以实现吗?或者是现在已经支持了,如何配置?

如何配合nginx使用?

远程服务器里有web服务,使用nginx作为服务前端,frp要如何配合nginx使用?
以下配置测试了行不通。需要如何配置
nginx配置:
server {
listen 80;
server_name demo.*;
location / {
proxy_pass http://127.0.0.1:7000;
}
}
frps.ini 配置
[common]
bind_port = 7000
vhost_http_port = 8008
[web01]
type = http
custom_domains = demo.exampledomain.com

frpc.ini 配置
[common]
server_addr = 162...*
[web01]
type = http
local_ip = 127.0.0.1
local_port = 80

请问支持多内网服务器吗

比如我有一台公网服务器,有多个内网服务器, 是否支持根据不同的域名/子域名访问不同的内网服务器?

映射远程端口到本地

可以实现远程端口到本地到本地吗?本地电脑在内网且IP不固定,没法在本地安装服务端。
谢谢提供这么好的工具。

增加免服务端配置模式

前几天,在邮件中提过,这里再次写上来,补充些想法。
Ngrok 是免服务端配置的,指新增加一个c端配置,s端不用重改ini 且重启服务才能生效。

免服务端配置,起码适用于分享s端的盆友。
如 Ngrok 已知国内的 免费服务:
http://www.qydev.com/ 北京阿里云
http://www.ittun.com/ 深圳阿里云
http://www.natapp.cn/ 深圳阿里云(已商业化)
http://www.ngrok.cc/ 香港
http://www.gotunnel.org/ 香港

有些都是在校朋友,如,阿里云有在校生优惠计划,9.92元/月, 愿意分享的会公开给大家服务。
希望frp 借鉴 Ngrok, 增加仅c端配置模式。
另外,结合前面一个issues,或许加个流量限制参数,本来免费提供大家s端的,限制极端使用的用户超流量。

go get 安装失败

你好,
我想用下面的命令下载程序编译
go get github.com/fatedier/frp
但是提示错误
can't load package: package github.com/fatedier/frp: no buildable Go source files in /home/webadmin/golang_work/src/github.com/fatedier/frp
之前用这种方式安装的,挺方便的,现在是不支持这种安装方式了吗?

Does't open Gzip option

I have update frpc and frps to latest version,and add "use_gzip = true" into frpc.ini,but when i look by dashboard,it show Gzip false,why? Whether i need to config frps.ini file?

我刚把 frpc 和 frps 升级到最新的版本,并且在 frpc.ini 中,加上了 “use_gzip = true” 的配置信息,但是我通过 dashboard 查看的时候,还是显示的没开启 Gzip。是不是还需要配置 frps.ini 文件,或者会是其它什么原因了?

### frpc.ini

[common]
......
server_port = 7000
log_file = /jffs/frp/frpc.log
log_level = info
log_max_days = 3

[nas]
......
pool_count = 10
use_encryption = false
use_gzip = true

join connections error

2016/08/04 16:54:04 [control.go:181] [I] Start to send heartbeat to frps
2016/08/04 16:54:21 [conn.go:182] [W] join connections error, read tcp 192.168.1.190:41386->192.168.1.190:18080: read tcp 192.168.1.190:44203->121.40.217.143:47000: use of closed network connection

[Question] Does the tool perform NAT hole punching?

Hi,
It's unclear to me if this tool provides NAT hole punching or not? I'm running a host behind a NAT. The admin of the NAT does not want to allow my host to be visible on the internet so he's only allowing outgoing connections from my host and replies. No incoming connections are allowed. Will this tool allow me to expose a service running on my host to the internet? Thanks.

frp http能否绑定多个端口

vhost_http_port = 8080
在配置中这样就绑定了8080端口
但是在我的应用中 我希望不同的web配置绑定不同的端口

增加一个c/c++版本的协议接口吧

方便的话,提供一个c或c++版本的反向代理功能库
方便大家在嵌入式系统上移植终端。
现在openwrt的开发环境不支持go语言。

是否能够增加流量统计

c发给s的流量, 还有发给访问者的流量(按C端区分),仅每天总量,目的是发现流量异常c端

-bash: ./frps: Permission denied

I was trying frp server on my vps (bandwagon host), login as root. After modifying frps.ini,I execute
./frps -c ./frps.ini
and it returns
-bash: ./frps: Permission denied
Am I missed sth? I would appreciate if you could give tips.
Thanks !

ssh没有正常工作

按照示例配置启动frps和frpc之后,通过 ssh 访问内网机器,连接总是超时。
[root@iZ4i1kgqxZ ~]# ssh -oPort=6000 [email protected] ssh: connect to host 192.168.31.138 port 6000: Connection timed out

使用特权模式,开启use_encryption,frps产生错误退出

服务器端开启特权模式

# frps.ini
[common]
bind_addr = 0.0.0.0
bind_port = 7500
vhost_http_port = 801
vhost_https_port = 443
dashboard_port = 7000
log_file = ./frps.log
log_level = info
log_max_days = 3
privilege_token is correct
privilege_mode = true
privilege_token = oxcm!Jkn/jQ(S?of
privilege_allow_ports = 2000-3000,3001,3003,4000-50000
max_pool_count = 100

客户端

# frpc.ini
[privilege_web1]
privilege_mode = true
type = http
local_ip = 127.0.0.1
local_port = 30000
use_gzip = true
use_encryption = true
custom_domains = xx.xxxx.xx

之后打开xx.xxxx.xx:801,frps就会自动退出,并且报错,错误内容如下

➜  frp ./frps -c frps.ini
panic: cipher.NewCBCEncrypter: IV length must equal block size

goroutine 25 [running]:
panic(0x794c80, 0xc82016b6a0)
        /usr/lib/golang/src/runtime/panic.go:481 +0x3e6
crypto/cipher.NewCBCEncrypter(0x7f9ca8a6dcf0, 0xc8201757d0, 0xc820308760, 0x20, 0x20, 0x0, 0x0)
        /usr/lib/golang/src/crypto/cipher/cbc.go:37 +0xa8
github.com/fatedier/frp/src/utils/pcrypto.(*Pcrypto).Encrypt(0xc820341eb8, 0xc820114380, 0x170, 0x1c0, 0x0, 0x0, 0x0, 0x0, 0x0)
        /home/wcl/go_projects/src/github.com/fatedier/frp/src/utils/pcrypto/pcrypto.go:44 +0xd5
github.com/fatedier/frp/src/models/msg.pipeEncrypt(0xc82032e080, 0xc82032e240, 0xc82016b370, 0xe, 0x0, 0x0, 0xc82016b35c, 0x4, 0x10101, 0xc820174162, ...)
        /home/wcl/go_projects/src/github.com/fatedier/frp/src/models/msg/process.go:242 +0x92c
github.com/fatedier/frp/src/models/msg.JoinMore.func1(0xc82032e080, 0xc82032e240)
        /home/wcl/go_projects/src/github.com/fatedier/frp/src/models/msg/process.go:64 +0x104
created by github.com/fatedier/frp/src/models/msg.JoinMore
        /home/wcl/go_projects/src/github.com/fatedier/frp/src/models/msg/process.go:80 +0x1ca

The ssh cannot be connected sometimes

I can connect to client directly, but when I connect through server using ssh [email protected] -p 6022, it often stucked and after some time show:
ssh_exchange_identification: Connection closed by remote host
I can only login 1 times in 10.
When I forward 80 port, it works well.
I want to know whether it will missing some information?
The configs are as bellow:
server:

[ssh]
auth_token = password
bind_addr = 0.0.0.0
listen_port = 6022

client:

[ssh]
local_ip = 127.0.0.1
local_port = 22
use_encryption = false

ssh log:

OpenSSH_6.9p1, LibreSSL 2.1.8
debug1: Reading configuration data /Users/maple/.ssh/config
debug1: /Users/maple/.ssh/config line 1: Applying options for *
debug1: Reading configuration data /etc/ssh/ssh_config
debug1: /etc/ssh/ssh_config line 20: Applying options for *
debug1: /etc/ssh/ssh_config line 102: Applying options for *
debug1: Connecting to my.server.ip [my.server.ip] port 6022.
debug1: Connection established.
debug1: identity file /Users/maple/.ssh/id_rsa type 1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_rsa-cert type -1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_dsa type -1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_dsa-cert type -1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_ecdsa type -1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_ecdsa-cert type -1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_ed25519 type -1
debug1: key_load_public: No such file or directory
debug1: identity file /Users/maple/.ssh/id_ed25519-cert type -1
debug1: Enabling compatibility mode for protocol 2.0
debug1: Local version string SSH-2.0-OpenSSH_6.9
debug1: ssh_exchange_identification:
debug1: ssh_exchange_identification: \024\247       u\234\333\373\030\357]-]\247\210\366\241
debug1: ssh_exchange_identification: dsa-sha2-nistp256,ssh-ed25519
debug1: ssh_exchange_identification: ator.liu.se
debug1: ssh_exchange_identification: [email protected],[email protected],[email protected],[email protected],[email protected],[email protected],[email protected],[email protected],[email protected],hmac-md5,h
debug1: ssh_exchange_identification: mac-sha1,[email protected],[email protected],hmac-sha2-256,hmac-sha2-512,hmac-ripemd160,[email protected],hmac-sha1-96,hmac-md5-96
debug1: ssh_exchange_identification: [email protected],[email protected],[email protected],[email protected],[email protected],hmac-md5,hmac-sha1,[email protected],[email protected],hmac-sha2-256,hmac-sha2-512,hmac-ripemd160,hmac-ripemd
ssh_exchange_identification: Connection closed by remote host

Can't compile code

I can't compile that, The log are as:
go get github.com/fatedier/frp
package github.com/fatedier/frp
imports github.com/fatedier/frp
imports github.com/fatedier/frp: no buildable Go source files in /home/maple/ProgramFiles/golang/src/github.com/fatedier/frp

ipv6支持

请问frps可以监听ipv6的端口吗

utils\pcrypto\Encrypt 加密问题

目前是gzip/ aes / base64 顺序执行:压缩+aes加密+ASCII转码。
两个问题:

  1. 能否去掉 aes(新增配置项),以便 c端带宽有限的 节省上传带宽。
  2. 能否去除 base64, 这个是仅仅是为了分包吗?有没有更好的办法,分包而不需要重新编码,毕竟base64是http协议才更有意义,若传输的数据都是二进制数据,多次转义增加了带宽、CPU开销。
    顺带,个人认为go的优势是接近c、C++的性能,那实现某些软件细节时 也应该考虑极致的性能、高并发。望考虑

二进制程序链接更醒目些

doc里面的二进制连接也不够醒目,为了不懂go语言的盆友,只是一般网管级别技术人员,而非开发人员,只是找工具使用。
建议:
1.直接将二进制下载 链接 放到 README*.md 文件里面,并单独章节醒目些(而不是在“下载源码”章节里面顺便提一下)。
2.windows发布,建议增加服务(service 保证windows启动时 s端自动已启动)指导说明:
NSSM - the Non-Sucking Service Manager
http://www.nssm.cc/
安装服务: nssm install frps x:\xxx\frp\frps.exe
启动服务:nssm start frps
停止服务:nssm stop frps

一个细节的问题,关于auth_token的

Readme.md文档中说明了frpc 所在机器和 frps 所在机器的时间相差不能超过 15 分钟,因为时间戳会被用于加密验证中。
这里的时间差是指UTC时间,还是双方的本地时间差?

server adder error

I set server adderess in frpc.ini, and run frpc -c frpc.ini
the log show:
2016/04/07 09:00:48 [main.go:101] [I] Start frpc success
2016/04/07 09:00:48 [control.go:130] [E] ProxyName [ssh], connect to server [0.0.0.0:7000] error, dial tcp 0.0.0.0:7000: connection refused
2016/04/07 09:00:48 [control.go:39] [E] ProxyName [ssh], connect to server failed!
2016/04/07 09:00:48 [main.go:104] [W] All proxy exit!

It seems it also use 0.0.0.0 as server. Please help me to tackle this problem. Thanks

阿里云测试

在阿里云服务器 配置服务端,本地能 telnet 7000 端口,然后在家再telnet 7000 端口 不通。

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.