Skip to content

Commit cfe1965

Browse files
committed
Add HTTP proxy support to iotcored
Allows MQTT connections on networks where all outbound communication is gated by a proxy server. Tested by running a deployment to a remote tinyproxy server. Configure the network proxy by writing the following configuration object to services/aws.greengrass.NucleusLite/configuration/networkProxy: networkProxy: { noproxy: "192.168.1.1,*.example.com:80", proxy: { url: "http://www.myproxy.io:8888" } } These settings are equivalent to setting no_proxy and https_proxy environment variables for OpenSSl
1 parent 600b705 commit cfe1965

File tree

11 files changed

+433
-36
lines changed

11 files changed

+433
-36
lines changed

bins/ggdeploymentd/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ ggl_init_module(
1010
ggl-constants
1111
ggl-file
1212
ggl-http
13+
ggl-uri
1314
ggl-json
1415
ggl-recipe
1516
ggl-semver

bins/iotcored/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -10,4 +10,5 @@ ggl_init_module(
1010
core_mqtt
1111
ggl-backoff
1212
ggl-file
13+
ggl-uri
1314
PkgConfig::openssl)

bins/iotcored/bin/iotcored.c

+16
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,9 @@
55
#include "iotcored.h"
66
#include <argp.h>
77
#include <ggl/error.h>
8+
#include <ggl/log.h>
89
#include <ggl/version.h>
10+
#include <stdlib.h>
911

1012
__attribute__((visibility("default"))) const char *argp_program_version
1113
= GGL_VERSION;
@@ -57,6 +59,20 @@ static struct argp argp = { opts, arg_parser, 0, doc, 0, 0, 0 };
5759
int main(int argc, char **argv) {
5860
static IotcoredArgs args = { 0 };
5961

62+
// NOLINTBEGIN(concurrency-mt-unsafe)
63+
char *proxy_uri = proxy_uri = getenv("https_proxy");
64+
if (proxy_uri == NULL) {
65+
proxy_uri = getenv("HTTPS_PROXY");
66+
}
67+
args.proxy_uri = proxy_uri;
68+
69+
char *no_proxy = getenv("no_proxy");
70+
if (no_proxy == NULL) {
71+
no_proxy = getenv("NO_PROXY");
72+
}
73+
args.no_proxy = no_proxy;
74+
// NOLINTEND(concurrency-mt-unsafe)
75+
6076
// NOLINTNEXTLINE(concurrency-mt-unsafe)
6177
argp_parse(&argp, argc, argv, 0, 0, &args);
6278

bins/iotcored/include/iotcored.h

+2
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@ typedef struct {
1414
char *rootca;
1515
char *cert;
1616
char *key;
17+
char *no_proxy;
18+
char *proxy_uri;
1719
} IotcoredArgs;
1820

1921
GglError run_iotcored(IotcoredArgs *args);

bins/iotcored/src/entry.c

+75
Original file line numberDiff line numberDiff line change
@@ -8,14 +8,34 @@
88
#include <ggl/buffer.h>
99
#include <ggl/core_bus/gg_config.h>
1010
#include <ggl/error.h>
11+
#include <ggl/log.h>
1112
#include <ggl/object.h>
1213
#include <limits.h>
1314
#include <string.h>
15+
#include <stdbool.h>
1416
#include <stdint.h>
17+
#include <stdlib.h>
1518

1619
#define MAX_ENDPOINT_LEN 128
1720
#define MAX_THINGNAME_LEN 128
1821

22+
static bool getenv_copy(char *name, GglBuffer *destination) {
23+
// NOLINTNEXTLINE(concurrency-mt-unsafe)
24+
char *value = getenv(name);
25+
if (value == NULL) {
26+
return false;
27+
}
28+
GglBuffer source = ggl_buffer_from_null_term(value);
29+
if (source.len >= destination->len) {
30+
GGL_LOGW("%s too long.", name);
31+
return false;
32+
}
33+
memcpy(destination->data, source.data, source.len);
34+
destination->len = source.len;
35+
destination->data[destination->len] = '\0';
36+
return true;
37+
}
38+
1939
GglError run_iotcored(IotcoredArgs *args) {
2040
if (args->cert == NULL) {
2141
static uint8_t cert_mem[PATH_MAX] = { 0 };
@@ -94,6 +114,61 @@ GglError run_iotcored(IotcoredArgs *args) {
94114
args->rootca = (char *) rootca_mem;
95115
}
96116

117+
static uint8_t proxy_uri_mem[PATH_MAX] = { 0 };
118+
if (args->proxy_uri == NULL) {
119+
GglBuffer proxy_uri = GGL_BUF(proxy_uri_mem);
120+
if (getenv_copy("https_proxy", &proxy_uri)) {
121+
args->proxy_uri = (char *) proxy_uri_mem;
122+
} else if (getenv_copy("HTTPS_PROXY", &proxy_uri)) {
123+
args->proxy_uri = (char *) proxy_uri_mem;
124+
}
125+
}
126+
if (args->proxy_uri == NULL) {
127+
GglBuffer proxy_uri = GGL_BUF(proxy_uri_mem);
128+
proxy_uri.len -= 1;
129+
GglError ret = ggl_gg_config_read_str(
130+
GGL_BUF_LIST(
131+
GGL_STR("services"),
132+
GGL_STR("aws.greengrass.NucleusLite"),
133+
GGL_STR("configuration"),
134+
GGL_STR("networkProxy"),
135+
GGL_STR("proxy"),
136+
GGL_STR("url")
137+
),
138+
&proxy_uri
139+
);
140+
if (ret == GGL_ERR_OK) {
141+
args->proxy_uri = (char *) proxy_uri_mem;
142+
}
143+
}
144+
145+
static uint8_t no_proxy_mem[PATH_MAX] = { 0 };
146+
if (args->no_proxy == NULL) {
147+
GglBuffer no_proxy = GGL_BUF(no_proxy_mem);
148+
if (getenv_copy("no_proxy", &no_proxy)) {
149+
args->no_proxy = (char *) no_proxy_mem;
150+
} else if (getenv_copy("NO_PROXY", &no_proxy)) {
151+
args->no_proxy = (char *) no_proxy_mem;
152+
}
153+
}
154+
if (args->no_proxy == NULL) {
155+
GglBuffer no_proxy = GGL_BUF(no_proxy_mem);
156+
no_proxy.len -= 1;
157+
GglError ret = ggl_gg_config_read_str(
158+
GGL_BUF_LIST(
159+
GGL_STR("services"),
160+
GGL_STR("aws.greengrass.NucleusLite"),
161+
GGL_STR("configuration"),
162+
GGL_STR("networkProxy"),
163+
GGL_STR("noproxy"),
164+
),
165+
&no_proxy
166+
);
167+
if (ret == GGL_ERR_OK) {
168+
args->no_proxy = (char *) no_proxy_mem;
169+
}
170+
}
171+
97172
GglError ret = iotcored_mqtt_connect(args);
98173
if (ret != GGL_ERR_OK) {
99174
return ret;

0 commit comments

Comments
 (0)