diff --git a/developer_tools/vpnserver-jsonrpc-clients/README.html b/developer_tools/vpnserver-jsonrpc-clients/README.html
index 2d41f959..d7a7ea47 100644
--- a/developer_tools/vpnserver-jsonrpc-clients/README.html
+++ b/developer_tools/vpnserver-jsonrpc-clients/README.html
@@ -30,6 +30,7 @@
- Older versions of SoftEther VPN before June 2019 don't support JSON-RPC APIs.
- If you want to completely disable the JSON-RPC on your VPN Server, set the
DisableJsonRpcWebApi
variable to true
on the vpn_server.config
.
+- You may also restrict access to JSON-RPC API to a specific subnet, e.g. your internal network, by setting the
JsonRpcWebApiAllowedSubnet
variable to, for example, 192.168.0.0/16
.
JSON-RPC specification
You must use HTTPS 1.1 POST
method to call each of JSON-RPC APIs.
diff --git a/developer_tools/vpnserver-jsonrpc-clients/README.md b/developer_tools/vpnserver-jsonrpc-clients/README.md
index 1ea33f9f..23ab02d6 100644
--- a/developer_tools/vpnserver-jsonrpc-clients/README.md
+++ b/developer_tools/vpnserver-jsonrpc-clients/README.md
@@ -25,6 +25,7 @@ https://:/api/
- Older versions of SoftEther VPN before June 2019 don't support JSON-RPC APIs.
- If you want to completely disable the JSON-RPC on your VPN Server, set the `DisableJsonRpcWebApi` variable to `true` on the `vpn_server.config`.
+ - You may also restrict access to JSON-RPC API to a specific subnet, e.g. your internal network, by setting the `JsonRpcWebApiAllowedSubnet` variable to, for example, `192.168.0.0/16`.
### JSON-RPC specification
diff --git a/developer_tools/vpnserver-jsonrpc-codegen/Templates/doc.txt b/developer_tools/vpnserver-jsonrpc-codegen/Templates/doc.txt
index aed5c743..45be851d 100644
--- a/developer_tools/vpnserver-jsonrpc-codegen/Templates/doc.txt
+++ b/developer_tools/vpnserver-jsonrpc-codegen/Templates/doc.txt
@@ -25,6 +25,7 @@ https://:/api/
- Older versions of SoftEther VPN before June 2019 don't support JSON-RPC APIs.
- If you want to completely disable the JSON-RPC on your VPN Server, set the `DisableJsonRpcWebApi` variable to `true` on the `vpn_server.config`.
+ - You may also restrict access to JSON-RPC API to a specific subnet, e.g. your internal network, by setting the `JsonRpcWebApiAllowedSubnet` variable to, for example, `192.168.0.0/16`.
### JSON-RPC specification
diff --git a/src/Cedar/Protocol.c b/src/Cedar/Protocol.c
index a0da6c70..f20e8c87 100644
--- a/src/Cedar/Protocol.c
+++ b/src/Cedar/Protocol.c
@@ -5740,6 +5740,7 @@ bool ServerDownloadSignature(CONNECTION *c, char **error_detail_str)
UINT num = 0, max = 19;
SERVER *server;
char *vpn_http_target = HTTP_VPN_TARGET2;
+ bool disableJsonRpcWebApi;
// Validate arguments
if (c == NULL)
{
@@ -5750,6 +5751,15 @@ bool ServerDownloadSignature(CONNECTION *c, char **error_detail_str)
s = c->FirstSock;
+ disableJsonRpcWebApi = server->DisableJsonRpcWebApi;
+ if (!disableJsonRpcWebApi && !IsZeroIP(&server->JsonRpcWebApiAllowedSubnetAddr)
+ && !IsZeroIP(&server->JsonRpcWebApiAllowedSubnetMask)) {
+ // restrict JSON-RPC Web API to specified subnet only
+ if (!IsInSameNetwork(&s->RemoteIP, &server->JsonRpcWebApiAllowedSubnetAddr, &server->JsonRpcWebApiAllowedSubnetMask)) {
+ disableJsonRpcWebApi = true;
+ }
+ }
+
while (true)
{
bool not_found_error = false;
@@ -5782,7 +5792,7 @@ bool ServerDownloadSignature(CONNECTION *c, char **error_detail_str)
// Receive the data since it's POST
data_size = GetContentLength(h);
- if (server->DisableJsonRpcWebApi == false)
+ if (disableJsonRpcWebApi == false)
{
if (StrCmpi(h->Target, "/api") == 0 || StrCmpi(h->Target, "/api/") == 0)
{
@@ -5868,7 +5878,7 @@ bool ServerDownloadSignature(CONNECTION *c, char **error_detail_str)
}
else if (StrCmpi(h->Method, "OPTIONS") == 0)
{
- if (server->DisableJsonRpcWebApi == false)
+ if (disableJsonRpcWebApi == false)
{
if (StrCmpi(h->Target, "/api") == 0 || StrCmpi(h->Target, "/api/") == 0 || StartWith(h->Target, "/admin"))
{
@@ -5939,7 +5949,7 @@ bool ServerDownloadSignature(CONNECTION *c, char **error_detail_str)
BUF *b = NULL;
*error_detail_str = "HTTP_ROOT";
- if (server->DisableJsonRpcWebApi == false)
+ if (disableJsonRpcWebApi == false)
{
b = ReadDump("|wwwroot/index.html");
}
@@ -6019,7 +6029,7 @@ bool ServerDownloadSignature(CONNECTION *c, char **error_detail_str)
if (b == false)
{
- if (server->DisableJsonRpcWebApi == false)
+ if (disableJsonRpcWebApi == false)
{
if (StartWith(h->Target, "/api?") || StartWith(h->Target, "/api/") || StrCmpi(h->Target, "/api") == 0)
{
diff --git a/src/Cedar/Server.c b/src/Cedar/Server.c
index 1aad0d94..6eb53d44 100644
--- a/src/Cedar/Server.c
+++ b/src/Cedar/Server.c
@@ -30,6 +30,7 @@
#include "Mayaqua/Internat.h"
#include "Mayaqua/Memory.h"
#include "Mayaqua/Microsoft.h"
+#include "Mayaqua/Network.h"
#include "Mayaqua/Object.h"
#include "Mayaqua/OS.h"
#include "Mayaqua/Pack.h"
@@ -6032,6 +6033,15 @@ void SiLoadServerCfg(SERVER *s, FOLDER *f)
// Disable JSON-RPC Web API
s->DisableJsonRpcWebApi = CfgGetBool(f, "DisableJsonRpcWebApi");
+ char tmpaddr[MAX_PATH];
+ if (CfgGetStr(f, "JsonRpcWebApiAllowedSubnet", tmpaddr, sizeof(tmpaddr))) {
+ IP _subnet, _mask;
+ if (ParseIpAndMask46(tmpaddr, &_subnet, &_mask)) {
+ s->JsonRpcWebApiAllowedSubnetAddr = _subnet;
+ s->JsonRpcWebApiAllowedSubnetMask = _mask;
+ }
+ }
+
// Bits of Diffie-Hellman parameters
c->DhParamBits = CfgGetInt(f, "DhParamBits");
if (c->DhParamBits == 0)
@@ -6365,6 +6375,11 @@ void SiWriteServerCfg(FOLDER *f, SERVER *s)
// Disable JSON-RPC Web API
CfgAddBool(f, "DisableJsonRpcWebApi", s->DisableJsonRpcWebApi);
+
+ char tmpaddr[MAX_PATH];
+ IPAndMaskToStr(tmpaddr, sizeof(tmpaddr),
+ &s->JsonRpcWebApiAllowedSubnetAddr, &s->JsonRpcWebApiAllowedSubnetMask);
+ CfgAddStr(f, "JsonRpcWebApiAllowedSubnet", tmpaddr);
}
Unlock(c->lock);
}
diff --git a/src/Cedar/Server.h b/src/Cedar/Server.h
index 01fbddf7..82e8a22c 100644
--- a/src/Cedar/Server.h
+++ b/src/Cedar/Server.h
@@ -276,6 +276,9 @@ struct SERVER
IP ListenIP; // Listen IP
bool StrictSyslogDatetimeFormat; // Make syslog datetime format strict RFC3164
bool DisableJsonRpcWebApi; // Disable JSON-RPC Web API
+
+ IP JsonRpcWebApiAllowedSubnetAddr; // If set, allow access to JSON-RPC Web API from
+ IP JsonRpcWebApiAllowedSubnetMask; // this subnet only
};
diff --git a/src/Mayaqua/Network.c b/src/Mayaqua/Network.c
index 6b53ecfb..cc4567d1 100644
--- a/src/Mayaqua/Network.c
+++ b/src/Mayaqua/Network.c
@@ -6993,6 +6993,18 @@ void IPToStr6Inner(char *str, IP *ip)
}
}
+// Format IP and subnet mask as "/"
+void IPAndMaskToStr(char *str, UINT size, IP *ip, IP *subnet)
+{
+ int iplen;
+ UINT masksize;
+
+ IPToStr(str, size, ip);
+ iplen = StrLen(str);
+ masksize = SubnetMaskToInt(subnet);
+ Format(str + iplen, size - iplen, "/%d", masksize);
+}
+
// Convert the string to an IP address
bool StrToIP6(IP *ip, char *str)
{
diff --git a/src/Mayaqua/Network.h b/src/Mayaqua/Network.h
index 94a50c2b..6d7271bc 100644
--- a/src/Mayaqua/Network.h
+++ b/src/Mayaqua/Network.h
@@ -1293,6 +1293,7 @@ void IPToStr6(char *str, UINT size, IP *ip);
void IP6AddrToStr(char *str, UINT size, IPV6_ADDR *addr);
void IPToStr6Array(char *str, UINT size, UCHAR *bytes);
void IPToStr6Inner(char *str, IP *ip);
+void IPAndMaskToStr(char *str, UINT size, IP *ip, IP *subnet);
void IntToSubnetMask6(IP *ip, UINT i);
void IPAnd6(IP *dst, IP *a, IP *b);
void GetAllRouterMulticastAddress6(IP *ip);