Skip to content

Commit ba52cae

Browse files
authored
Merge pull request #118 from launchdarkly/eb/ch52414/proxy-config
add config option for proxy URL
2 parents 3dddca2 + 5911fd9 commit ba52cae

9 files changed

+142
-94
lines changed

ldclient/config.py

+12-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,8 @@ def __init__(self,
4343
offline=False,
4444
user_keys_capacity=1000,
4545
user_keys_flush_interval=300,
46-
inline_users_in_events=False):
46+
inline_users_in_events=False,
47+
http_proxy=None):
4748
"""
4849
:param string sdk_key: The SDK key for your LaunchDarkly account.
4950
:param string base_uri: The base URL for the LaunchDarkly server. Most users should use the default
@@ -95,6 +96,11 @@ def __init__(self,
9596
:type event_processor_class: (ldclient.config.Config) -> EventProcessor
9697
:param update_processor_class: A factory for an UpdateProcessor implementation taking the sdk key,
9798
config, and FeatureStore implementation
99+
:param http_proxy: Use a proxy when connecting to LaunchDarkly. This is the full URI of the
100+
proxy; for example: http://my-proxy.com:1234. Note that unlike the standard `http_proxy` environment
101+
variable, this is used regardless of whether the target URI is HTTP or HTTPS (the actual LaunchDarkly
102+
service uses HTTPS, but a Relay Proxy instance could use HTTP). Setting this Config parameter will
103+
override any proxy specified by an environment variable, but only for LaunchDarkly SDK connections.
98104
"""
99105
self.__sdk_key = sdk_key
100106

@@ -126,6 +132,7 @@ def __init__(self,
126132
self.__user_keys_capacity = user_keys_capacity
127133
self.__user_keys_flush_interval = user_keys_flush_interval
128134
self.__inline_users_in_events = inline_users_in_events
135+
self.__http_proxy = http_proxy
129136

130137
@classmethod
131138
def default(cls):
@@ -278,6 +285,10 @@ def user_keys_flush_interval(self):
278285
def inline_users_in_events(self):
279286
return self.__inline_users_in_events
280287

288+
@property
289+
def http_proxy(self):
290+
return self.__http_proxy
291+
281292
def _validate(self):
282293
if self.offline is False and self.sdk_key is None or self.sdk_key is '':
283294
log.warning("Missing or blank sdk_key.")

ldclient/event_processor.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -212,7 +212,7 @@ def __init__(self, inbox, config, http_client):
212212
self._inbox = inbox
213213
self._config = config
214214
self._http = create_http_pool_manager(num_pools=1, verify_ssl=config.verify_ssl,
215-
target_base_uri=config.events_uri) if http_client is None else http_client
215+
target_base_uri=config.events_uri, force_proxy=config.http_proxy) if http_client is None else http_client
216216
self._close_http = (http_client is None) # so we know whether to close it later
217217
self._disabled = False
218218
self._outbox = EventBuffer(config.events_max_pending)

ldclient/feature_requester.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,8 @@
2525
class FeatureRequesterImpl(FeatureRequester):
2626
def __init__(self, config):
2727
self._cache = dict()
28-
self._http = create_http_pool_manager(num_pools=1, verify_ssl=config.verify_ssl, target_base_uri=config.base_uri)
28+
self._http = create_http_pool_manager(num_pools=1, verify_ssl=config.verify_ssl,
29+
target_base_uri=config.base_uri, force_proxy=config.http_proxy)
2930
self._config = config
3031

3132
def get_all_data(self):

ldclient/sse_client.py

+3-2
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
class SSEClient(object):
2525
def __init__(self, url, last_id=None, retry=3000, connect_timeout=10, read_timeout=300, chunk_size=10000,
26-
verify_ssl=False, http=None, **kwargs):
26+
verify_ssl=False, http=None, http_proxy=None, **kwargs):
2727
self.url = url
2828
self.last_id = last_id
2929
self.retry = retry
@@ -32,7 +32,8 @@ def __init__(self, url, last_id=None, retry=3000, connect_timeout=10, read_timeo
3232
self._chunk_size = chunk_size
3333

3434
# Optional support for passing in an HTTP client
35-
self.http = create_http_pool_manager(num_pools=1, verify_ssl=verify_ssl, target_base_uri=url)
35+
self.http = create_http_pool_manager(num_pools=1, verify_ssl=verify_ssl, target_base_uri=url,
36+
force_proxy=http_proxy)
3637

3738
# Any extra kwargs will be fed into the request call later.
3839
self.requests_kwargs = kwargs

ldclient/streaming.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -89,7 +89,8 @@ def _connect(self):
8989
headers=_stream_headers(self._config.sdk_key),
9090
connect_timeout=self._config.connect_timeout,
9191
read_timeout=stream_read_timeout,
92-
verify_ssl=self._config.verify_ssl)
92+
verify_ssl=self._config.verify_ssl,
93+
http_proxy=self._config.http_proxy)
9394

9495
def stop(self):
9596
log.info("Stopping StreamingUpdateProcessor")

ldclient/util.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -85,8 +85,8 @@ def status(self):
8585
return self._status
8686

8787

88-
def create_http_pool_manager(num_pools=1, verify_ssl=False, target_base_uri=None):
89-
proxy_url = _get_proxy_url(target_base_uri)
88+
def create_http_pool_manager(num_pools=1, verify_ssl=False, target_base_uri=None, force_proxy=None):
89+
proxy_url = force_proxy or _get_proxy_url(target_base_uri)
9090

9191
if not verify_ssl:
9292
if proxy_url is None:

testing/test_event_processor.py

+39-29
Original file line numberDiff line numberDiff line change
@@ -469,41 +469,51 @@ def start_consuming_events():
469469
assert had_no_more
470470

471471
def test_can_use_http_proxy_via_environment_var(monkeypatch):
472-
fake_events_uri = 'http://not-real'
473-
474472
with start_server() as server:
475473
monkeypatch.setenv('http_proxy', server.uri)
476-
config = Config(sdk_key = 'sdk-key', events_uri = fake_events_uri)
477-
server.setup_response(fake_events_uri + '/bulk', 200, None)
478-
479-
with DefaultEventProcessor(config) as ep:
480-
ep.send_event({ 'kind': 'identify', 'user': user })
481-
ep.flush()
482-
ep._wait_until_inactive()
483-
484-
# For an insecure proxy request, our stub server behaves enough like the real thing to satisfy the
485-
# HTTP client, so we should be able to see the request go through. Note that the URI path will
486-
# actually be an absolute URI for a proxy request.
487-
req = server.require_request()
488-
assert req.method == 'POST'
474+
config = Config(sdk_key = 'sdk-key', events_uri = 'http://not-real')
475+
_verify_http_proxy_is_used(server, config)
489476

490477
def test_can_use_https_proxy_via_environment_var(monkeypatch):
491-
fake_events_uri = 'https://not-real'
492-
493478
with start_server() as server:
494479
monkeypatch.setenv('https_proxy', server.uri)
495-
config = Config(sdk_key = 'sdk-key', events_uri = fake_events_uri)
496-
server.setup_response(fake_events_uri + '/bulk', 200, None)
497-
498-
with DefaultEventProcessor(config) as ep:
499-
ep.send_event({ 'kind': 'identify', 'user': user })
500-
ep.flush()
501-
ep._wait_until_inactive()
502-
503-
# Our simple stub server implementation can't really do HTTPS proxying, so the request will fail, but
504-
# it can still record that it *got* the request, which proves that the request went to the proxy.
505-
req = server.require_request()
506-
assert req.method == 'CONNECT'
480+
config = Config(sdk_key = 'sdk-key', events_uri = 'https://not-real')
481+
_verify_https_proxy_is_used(server, config)
482+
483+
def test_can_use_http_proxy_via_config():
484+
with start_server() as server:
485+
config = Config(sdk_key = 'sdk-key', events_uri = 'http://not-real', http_proxy=server.uri)
486+
_verify_http_proxy_is_used(server, config)
487+
488+
def test_can_use_https_proxy_via_config():
489+
with start_server() as server:
490+
config = Config(sdk_key = 'sdk-key', events_uri = 'https://not-real', http_proxy=server.uri)
491+
_verify_https_proxy_is_used(server, config)
492+
493+
def _verify_http_proxy_is_used(server, config):
494+
server.setup_response(config.events_uri + '/bulk', 200, None)
495+
with DefaultEventProcessor(config) as ep:
496+
ep.send_event({ 'kind': 'identify', 'user': user })
497+
ep.flush()
498+
ep._wait_until_inactive()
499+
500+
# For an insecure proxy request, our stub server behaves enough like the real thing to satisfy the
501+
# HTTP client, so we should be able to see the request go through. Note that the URI path will
502+
# actually be an absolute URI for a proxy request.
503+
req = server.require_request()
504+
assert req.method == 'POST'
505+
506+
def _verify_https_proxy_is_used(server, config):
507+
server.setup_response(config.events_uri + '/bulk', 200, None)
508+
with DefaultEventProcessor(config) as ep:
509+
ep.send_event({ 'kind': 'identify', 'user': user })
510+
ep.flush()
511+
ep._wait_until_inactive()
512+
513+
# Our simple stub server implementation can't really do HTTPS proxying, so the request will fail, but
514+
# it can still record that it *got* the request, which proves that the request went to the proxy.
515+
req = server.require_request()
516+
assert req.method == 'CONNECT'
507517

508518
def verify_unrecoverable_http_error(status):
509519
with DefaultEventProcessor(Config(sdk_key = 'SDK_KEY'), mock_http) as ep:

testing/test_feature_requester.py

+43-29
Original file line numberDiff line numberDiff line change
@@ -127,39 +127,53 @@ def test_get_one_flag_does_not_use_etags():
127127
assert 'If-None-Match' not in req.headers.keys() # did not send etag from previous request
128128

129129
def test_can_use_http_proxy_via_environment_var(monkeypatch):
130-
fake_base_uri = 'http://not-real'
131130
with start_server() as server:
132131
monkeypatch.setenv('http_proxy', server.uri)
133-
config = Config(sdk_key = 'sdk-key', base_uri = fake_base_uri)
134-
fr = FeatureRequesterImpl(config)
135-
136-
resp_data = { 'flags': {}, 'segments': {} }
137-
expected_data = { FEATURES: {}, SEGMENTS: {} }
138-
server.setup_json_response(fake_base_uri + '/sdk/latest-all', resp_data)
139-
140-
# For an insecure proxy request, our stub server behaves enough like the real thing to satisfy the
141-
# HTTP client, so we should be able to see the request go through. Note that the URI path will
142-
# actually be an absolute URI for a proxy request.
143-
result = fr.get_all_data()
144-
assert result == expected_data
145-
req = server.require_request()
146-
assert req.method == 'GET'
132+
config = Config(sdk_key = 'sdk-key', base_uri = 'http://not-real')
133+
_verify_http_proxy_is_used(server, config)
147134

148135
def test_can_use_https_proxy_via_environment_var(monkeypatch):
149-
fake_base_uri = 'https://not-real'
150136
with start_server() as server:
151137
monkeypatch.setenv('https_proxy', server.uri)
152-
config = Config(sdk_key = 'sdk-key', base_uri = fake_base_uri)
153-
fr = FeatureRequesterImpl(config)
138+
config = Config(sdk_key = 'sdk-key', base_uri = 'https://not-real')
139+
_verify_https_proxy_is_used(server, config)
154140

155-
resp_data = { 'flags': {}, 'segments': {} }
156-
server.setup_json_response(fake_base_uri + '/sdk/latest-all', resp_data)
157-
158-
# Our simple stub server implementation can't really do HTTPS proxying, so the request will fail, but
159-
# it can still record that it *got* the request, which proves that the request went to the proxy.
160-
try:
161-
fr.get_all_data()
162-
except:
163-
pass
164-
req = server.require_request()
165-
assert req.method == 'CONNECT'
141+
def test_can_use_http_proxy_via_config():
142+
with start_server() as server:
143+
config = Config(sdk_key = 'sdk-key', base_uri = 'http://not-real', http_proxy = server.uri)
144+
_verify_http_proxy_is_used(server, config)
145+
146+
def test_can_use_https_proxy_via_config():
147+
with start_server() as server:
148+
config = Config(sdk_key = 'sdk-key', base_uri = 'https://not-real', http_proxy = server.uri)
149+
_verify_https_proxy_is_used(server, config)
150+
151+
def _verify_http_proxy_is_used(server, config):
152+
fr = FeatureRequesterImpl(config)
153+
154+
resp_data = { 'flags': {}, 'segments': {} }
155+
expected_data = { FEATURES: {}, SEGMENTS: {} }
156+
server.setup_json_response(config.base_uri + '/sdk/latest-all', resp_data)
157+
158+
# For an insecure proxy request, our stub server behaves enough like the real thing to satisfy the
159+
# HTTP client, so we should be able to see the request go through. Note that the URI path will
160+
# actually be an absolute URI for a proxy request.
161+
result = fr.get_all_data()
162+
assert result == expected_data
163+
req = server.require_request()
164+
assert req.method == 'GET'
165+
166+
def _verify_https_proxy_is_used(server, config):
167+
fr = FeatureRequesterImpl(config)
168+
169+
resp_data = { 'flags': {}, 'segments': {} }
170+
server.setup_json_response(config.base_uri + '/sdk/latest-all', resp_data)
171+
172+
# Our simple stub server implementation can't really do HTTPS proxying, so the request will fail, but
173+
# it can still record that it *got* the request, which proves that the request went to the proxy.
174+
try:
175+
fr.get_all_data()
176+
except:
177+
pass
178+
req = server.require_request()
179+
assert req.method == 'CONNECT'

testing/test_streaming.py

+38-28
Original file line numberDiff line numberDiff line change
@@ -44,38 +44,48 @@ def test_sends_headers():
4444
assert req.headers['User-Agent'] == 'PythonClient/' + VERSION
4545

4646
def test_can_use_http_proxy_via_environment_var(monkeypatch):
47-
store = InMemoryFeatureStore()
48-
ready = Event()
49-
fake_stream_uri = 'http://not-real'
50-
5147
with start_server() as server:
48+
config = Config(sdk_key = 'sdk-key', stream_uri = 'http://not-real')
5249
monkeypatch.setenv('http_proxy', server.uri)
53-
config = Config(sdk_key = 'sdk-key', stream_uri = fake_stream_uri)
54-
server.setup_response(fake_stream_uri + '/all', 200, fake_event, { 'Content-Type': 'text/event-stream' })
55-
56-
with StreamingUpdateProcessor(config, None, store, ready) as sp:
57-
sp.start()
58-
# For an insecure proxy request, our stub server behaves enough like the real thing to satisfy the
59-
# HTTP client, so we should be able to see the request go through. Note that the URI path will
60-
# actually be an absolute URI for a proxy request.
61-
req = server.await_request()
62-
assert req.method == 'GET'
63-
ready.wait(1)
64-
assert sp.initialized()
50+
_verify_http_proxy_is_used(server, config)
6551

6652
def test_can_use_https_proxy_via_environment_var(monkeypatch):
67-
store = InMemoryFeatureStore()
68-
ready = Event()
69-
fake_stream_uri = 'https://not-real'
70-
7153
with start_server() as server:
54+
config = Config(sdk_key = 'sdk-key', stream_uri = 'https://not-real')
7255
monkeypatch.setenv('https_proxy', server.uri)
73-
config = Config(sdk_key = 'sdk-key', stream_uri = fake_stream_uri)
74-
server.setup_response(fake_stream_uri + '/all', 200, fake_event, { 'Content-Type': 'text/event-stream' })
56+
_verify_https_proxy_is_used(server, config)
7557

76-
with StreamingUpdateProcessor(config, None, store, ready) as sp:
77-
sp.start()
78-
# Our simple stub server implementation can't really do HTTPS proxying, so the request will fail, but
79-
# it can still record that it *got* the request, which proves that the request went to the proxy.
80-
req = server.await_request()
81-
assert req.method == 'CONNECT'
58+
def test_can_use_http_proxy_via_config():
59+
with start_server() as server:
60+
config = Config(sdk_key = 'sdk-key', stream_uri = 'http://not-real', http_proxy=server.uri)
61+
_verify_http_proxy_is_used(server, config)
62+
63+
def test_can_use_https_proxy_via_config():
64+
with start_server() as server:
65+
config = Config(sdk_key = 'sdk-key', stream_uri = 'https://not-real', http_proxy=server.uri)
66+
_verify_https_proxy_is_used(server, config)
67+
68+
def _verify_http_proxy_is_used(server, config):
69+
store = InMemoryFeatureStore()
70+
ready = Event()
71+
server.setup_response(config.stream_base_uri + '/all', 200, fake_event, { 'Content-Type': 'text/event-stream' })
72+
with StreamingUpdateProcessor(config, None, store, ready) as sp:
73+
sp.start()
74+
# For an insecure proxy request, our stub server behaves enough like the real thing to satisfy the
75+
# HTTP client, so we should be able to see the request go through. Note that the URI path will
76+
# actually be an absolute URI for a proxy request.
77+
req = server.await_request()
78+
assert req.method == 'GET'
79+
ready.wait(1)
80+
assert sp.initialized()
81+
82+
def _verify_https_proxy_is_used(server, config):
83+
store = InMemoryFeatureStore()
84+
ready = Event()
85+
server.setup_response(config.stream_base_uri + '/all', 200, fake_event, { 'Content-Type': 'text/event-stream' })
86+
with StreamingUpdateProcessor(config, None, store, ready) as sp:
87+
sp.start()
88+
# Our simple stub server implementation can't really do HTTPS proxying, so the request will fail, but
89+
# it can still record that it *got* the request, which proves that the request went to the proxy.
90+
req = server.await_request()
91+
assert req.method == 'CONNECT'

0 commit comments

Comments
 (0)