diff options
author | Aldo Cortesi <aldo@corte.si> | 2016-10-18 22:05:05 +1300 |
---|---|---|
committer | GitHub <noreply@github.com> | 2016-10-18 22:05:05 +1300 |
commit | a1859da390f3f84d1e2d94412da06f2b51a67b10 (patch) | |
tree | c4b8c020917ed6c7f6564c15894d748f5f48c261 | |
parent | faa26a5d6b952351749cbed4601f0376d6ab526d (diff) | |
parent | 466f5e56a156ac7506ab5523288d5a8de64a064e (diff) | |
download | mitmproxy-a1859da390f3f84d1e2d94412da06f2b51a67b10.tar.gz mitmproxy-a1859da390f3f84d1e2d94412da06f2b51a67b10.tar.bz2 mitmproxy-a1859da390f3f84d1e2d94412da06f2b51a67b10.zip |
Merge pull request #1626 from cortesi/state
Convert flow.state to an addon
30 files changed, 121 insertions, 173 deletions
diff --git a/mitmproxy/addons.py b/mitmproxy/addons.py index 5b078129..9e5677da 100644 --- a/mitmproxy/addons.py +++ b/mitmproxy/addons.py @@ -25,7 +25,7 @@ class Addons: def _options_update(self, options, updated): for i in self.chain: with self.master.handlecontext(): - i.configure(options, updated) + self.invoke_with_context(i, "configure", options, updated) def startup(self, s): """ diff --git a/mitmproxy/console/flowlist.py b/mitmproxy/console/flowlist.py index 653dfa02..ba555647 100644 --- a/mitmproxy/console/flowlist.py +++ b/mitmproxy/console/flowlist.py @@ -158,7 +158,7 @@ class ConnectionItem(urwid.WidgetWrap): (maxcol,) = xxx_todo_changeme key = common.shortcuts(key) if key == "a": - self.flow.accept_intercept(self.master) + self.flow.resume(self.master) signals.flowlist_change.send(self) elif key == "d": if self.flow.killable: diff --git a/mitmproxy/console/flowview.py b/mitmproxy/console/flowview.py index 0422e72b..b8f91bdb 100644 --- a/mitmproxy/console/flowview.py +++ b/mitmproxy/console/flowview.py @@ -519,7 +519,7 @@ class FlowView(tabs.Tabs): # Pass scroll events to the wrapped widget self._w.keypress(size, key) elif key == "a": - self.flow.accept_intercept(self.master) + self.flow.resume(self.master) signals.flow_change.send(self, flow = self.flow) elif key == "A": self.master.accept_all() diff --git a/mitmproxy/console/master.py b/mitmproxy/console/master.py index 1a413990..64761e40 100644 --- a/mitmproxy/console/master.py +++ b/mitmproxy/console/master.py @@ -222,7 +222,9 @@ class ConsoleMaster(flow.FlowMaster): palette = [] def __init__(self, server, options): - flow.FlowMaster.__init__(self, options, server, ConsoleState()) + flow.FlowMaster.__init__(self, options, server) + self.state = ConsoleState() + self.addons.add(self.state) self.stream_path = None # This line is just for type hinting self.options = self.options # type: Options diff --git a/mitmproxy/controller.py b/mitmproxy/controller.py index 148fda77..a225634a 100644 --- a/mitmproxy/controller.py +++ b/mitmproxy/controller.py @@ -27,6 +27,9 @@ Events = frozenset([ "responseheaders", "error", + "intercept", + "resume", + "websocket_handshake", "next_layer", diff --git a/mitmproxy/dump.py b/mitmproxy/dump.py index d3f894fa..d3a66876 100644 --- a/mitmproxy/dump.py +++ b/mitmproxy/dump.py @@ -33,7 +33,7 @@ class Options(options.Options): class DumpMaster(flow.FlowMaster): def __init__(self, server, options): - flow.FlowMaster.__init__(self, options, server, flow.DummyState()) + flow.FlowMaster.__init__(self, options, server) self.has_errored = False self.addons.add(termlog.TermLog()) self.addons.add(*builtins.default_addons()) diff --git a/mitmproxy/flow/__init__.py b/mitmproxy/flow/__init__.py index 9d243f49..07e1de6c 100644 --- a/mitmproxy/flow/__init__.py +++ b/mitmproxy/flow/__init__.py @@ -4,10 +4,10 @@ from mitmproxy.flow.master import FlowMaster from mitmproxy.flow.modules import ( AppRegistry ) -from mitmproxy.flow.state import State, DummyState, FlowView +from mitmproxy.flow.state import State, FlowView __all__ = [ "export", "modules", "FlowWriter", "FilteredFlowWriter", "FlowReader", "read_flows_from_paths", - "FlowMaster", "AppRegistry", "DummyState", "State", "FlowView", + "FlowMaster", "AppRegistry", "State", "FlowView", ] diff --git a/mitmproxy/flow/master.py b/mitmproxy/flow/master.py index 3d57c7bd..33502dc5 100644 --- a/mitmproxy/flow/master.py +++ b/mitmproxy/flow/master.py @@ -47,11 +47,10 @@ class FlowMaster(controller.Master): if len(self.servers) > 0: return self.servers[0] - def __init__(self, options, server, state): + def __init__(self, options, server): super().__init__(options) if server: self.add_server(server) - self.state = state self.apps = modules.AppRegistry() def start_app(self, host, port): @@ -196,7 +195,7 @@ class FlowMaster(controller.Master): @controller.handler def error(self, f): - self.state.update_flow(f) + pass @controller.handler def requestheaders(self, f): @@ -216,8 +215,6 @@ class FlowMaster(controller.Master): self.add_log("Error in wsgi app. %s" % err, "error") f.reply.kill() return - if f not in self.state.flows: # don't add again on replay - self.state.add_flow(f) @controller.handler def responseheaders(self, f): @@ -225,22 +222,14 @@ class FlowMaster(controller.Master): @controller.handler def response(self, f): - self.state.update_flow(f) + pass @controller.handler def websocket_handshake(self, f): pass - def handle_intercept(self, f): - self.state.update_flow(f) - - def handle_accept_intercept(self, f): - self.state.update_flow(f) - @controller.handler def tcp_start(self, flow): - # TODO: This would break mitmproxy currently. - # self.state.add_flow(flow) pass @controller.handler diff --git a/mitmproxy/flow/state.py b/mitmproxy/flow/state.py index 8c8e75c7..bb7460b6 100644 --- a/mitmproxy/flow/state.py +++ b/mitmproxy/flow/state.py @@ -167,7 +167,7 @@ class FlowStore(FlowList): # TODO: Should accept_all operate on views or on all flows? def accept_all(self, master): for f in self._list: - f.accept_intercept(master) + f.resume(master) def kill_all(self, master): for f in self._list: @@ -270,12 +270,19 @@ class State: self.add_flow(f2) return f2 + # Event handlers + def intercept(self, f): + self.update_flow(f) -class DummyState: - flows = () + def resume(self, f): + self.update_flow(f) - def add_flow(self, *args, **kwargs): - pass + def error(self, f): + self.update_flow(f) - def update_flow(self, *args, **kwargs): - pass + def request(self, f): + if f not in self.flows: # don't add again on replay + self.add_flow(f) + + def response(self, f): + self.update_flow(f) diff --git a/mitmproxy/models/flow.py b/mitmproxy/models/flow.py index 6d3fa0d5..2596165b 100644 --- a/mitmproxy/models/flow.py +++ b/mitmproxy/models/flow.py @@ -170,16 +170,16 @@ class Flow(stateobject.StateObject): def intercept(self, master): """ - Intercept this Flow. Processing will stop until accept_intercept is + Intercept this Flow. Processing will stop until resume is called. """ if self.intercepted: return self.intercepted = True self.reply.take() - master.handle_intercept(self) + master.addons("intercept", self) - def accept_intercept(self, master): + def resume(self, master): """ Continue with the flow - called after an intercept(). """ @@ -188,4 +188,4 @@ class Flow(stateobject.StateObject): self.intercepted = False self.reply.ack() self.reply.commit() - master.handle_accept_intercept(self) + master.addons("intercept", self) diff --git a/mitmproxy/web/app.py b/mitmproxy/web/app.py index 46bdd9e3..a81d04be 100644 --- a/mitmproxy/web/app.py +++ b/mitmproxy/web/app.py @@ -224,7 +224,7 @@ class AcceptFlows(RequestHandler): class AcceptFlow(RequestHandler): def post(self, flow_id): - self.flow.accept_intercept(self.master) + self.flow.resume(self.master) class FlowHandler(RequestHandler): diff --git a/mitmproxy/web/master.py b/mitmproxy/web/master.py index 23c95e57..3f61fd94 100644 --- a/mitmproxy/web/master.py +++ b/mitmproxy/web/master.py @@ -133,7 +133,9 @@ class Options(options.Options): class WebMaster(flow.FlowMaster): def __init__(self, server, options): - super().__init__(options, server, WebState()) + super().__init__(options, server) + self.state = WebState() + self.addons.add(self.state) self.addons.add(*builtins.default_addons()) self.app = app.Application( self, self.options.wdebug, self.options.wauthenticator diff --git a/test/mitmproxy/builtins/test_anticache.py b/test/mitmproxy/builtins/test_anticache.py index 81041e73..512c90f5 100644 --- a/test/mitmproxy/builtins/test_anticache.py +++ b/test/mitmproxy/builtins/test_anticache.py @@ -1,15 +1,13 @@ from .. import tutils, mastertest from mitmproxy.builtins import anticache from mitmproxy.flow import master -from mitmproxy.flow import state from mitmproxy import options class TestAntiCache(mastertest.MasterTest): def test_simple(self): - s = state.State() o = options.Options(anticache = True) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sa = anticache.AntiCache() m.addons.add(sa) diff --git a/test/mitmproxy/builtins/test_anticomp.py b/test/mitmproxy/builtins/test_anticomp.py index f866f33f..1014b9ba 100644 --- a/test/mitmproxy/builtins/test_anticomp.py +++ b/test/mitmproxy/builtins/test_anticomp.py @@ -1,15 +1,13 @@ from .. import tutils, mastertest from mitmproxy.builtins import anticomp from mitmproxy.flow import master -from mitmproxy.flow import state from mitmproxy import options class TestAntiComp(mastertest.MasterTest): def test_simple(self): - s = state.State() o = options.Options(anticomp = True) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sa = anticomp.AntiComp() m.addons.add(sa) diff --git a/test/mitmproxy/builtins/test_dumper.py b/test/mitmproxy/builtins/test_dumper.py index 20403eea..3ab75bee 100644 --- a/test/mitmproxy/builtins/test_dumper.py +++ b/test/mitmproxy/builtins/test_dumper.py @@ -3,7 +3,6 @@ import io from .. import tutils, mastertest from mitmproxy.builtins import dumper -from mitmproxy.flow import state from mitmproxy import exceptions from mitmproxy import dump from mitmproxy import models @@ -71,14 +70,13 @@ class TestContentView(mastertest.MasterTest): def test_contentview(self, view_auto): view_auto.side_effect = exceptions.ContentViewException("") - s = state.State() sio = io.StringIO() o = dump.Options( flow_detail=4, verbosity=3, tfile=sio, ) - m = mastertest.RecordingMaster(o, None, s) + m = mastertest.RecordingMaster(o, None) d = dumper.Dumper() m.addons.add(d) m.response(tutils.tflow()) diff --git a/test/mitmproxy/builtins/test_filestreamer.py b/test/mitmproxy/builtins/test_filestreamer.py index a43ea0b7..6fbeb40b 100644 --- a/test/mitmproxy/builtins/test_filestreamer.py +++ b/test/mitmproxy/builtins/test_filestreamer.py @@ -4,7 +4,6 @@ import os.path from mitmproxy.builtins import filestreamer from mitmproxy.flow import master, FlowReader -from mitmproxy.flow import state from mitmproxy import options @@ -17,11 +16,10 @@ class TestStream(mastertest.MasterTest): r = FlowReader(open(p, "rb")) return list(r.stream()) - s = state.State() o = options.Options( outfile = (p, "wb") ) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sa = filestreamer.FileStreamer() m.addons.add(sa) diff --git a/test/mitmproxy/builtins/test_replace.py b/test/mitmproxy/builtins/test_replace.py index e9869ab7..03943867 100644 --- a/test/mitmproxy/builtins/test_replace.py +++ b/test/mitmproxy/builtins/test_replace.py @@ -1,7 +1,6 @@ from .. import tutils, mastertest from mitmproxy.builtins import replace from mitmproxy.flow import master -from mitmproxy.flow import state from mitmproxy import options @@ -30,14 +29,13 @@ class TestReplace(mastertest.MasterTest): ) def test_simple(self): - s = state.State() o = options.Options( replacements = [ ("~q", "foo", "bar"), ("~s", "foo", "bar"), ] ) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sa = replace.Replace() m.addons.add(sa) diff --git a/test/mitmproxy/builtins/test_script.py b/test/mitmproxy/builtins/test_script.py index 683f3064..ecf2f3f7 100644 --- a/test/mitmproxy/builtins/test_script.py +++ b/test/mitmproxy/builtins/test_script.py @@ -8,7 +8,6 @@ from mitmproxy import exceptions from mitmproxy import options from mitmproxy.builtins import script from mitmproxy.flow import master -from mitmproxy.flow import state from .. import tutils, mastertest @@ -57,9 +56,8 @@ def test_load_script(): class TestScript(mastertest.MasterTest): def test_simple(self): - s = state.State() o = options.Options() - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sc = script.Script( tutils.test_data.path( "data/addonscripts/recorder.py" @@ -79,9 +77,8 @@ class TestScript(mastertest.MasterTest): assert recf[1] == "request" def test_reload(self): - s = state.State() o = options.Options() - m = mastertest.RecordingMaster(o, None, s) + m = mastertest.RecordingMaster(o, None) with tutils.tmpdir(): with open("foo.py", "w"): pass @@ -98,9 +95,8 @@ class TestScript(mastertest.MasterTest): raise AssertionError("Change event not detected.") def test_exception(self): - s = state.State() o = options.Options() - m = mastertest.RecordingMaster(o, None, s) + m = mastertest.RecordingMaster(o, None) sc = script.Script( tutils.test_data.path("data/addonscripts/error.py") ) @@ -113,25 +109,9 @@ class TestScript(mastertest.MasterTest): assert re.search('addonscripts/error.py", line \d+, in mkerr', m.event_log[0][1]) assert m.event_log[0][1].endswith("ValueError: Error!\n") - def test_duplicate_flow(self): - s = state.State() - o = options.Options() - fm = master.FlowMaster(o, None, s) - fm.addons.add( - script.Script( - tutils.test_data.path("data/addonscripts/duplicate_flow.py") - ) - ) - f = tutils.tflow() - fm.request(f) - assert fm.state.flow_count() == 2 - assert not fm.state.view[0].request.is_replay - assert fm.state.view[1].request.is_replay - def test_addon(self): - s = state.State() o = options.Options() - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sc = script.Script( tutils.test_data.path( "data/addonscripts/addon.py" @@ -163,9 +143,8 @@ class TestCutTraceback: class TestScriptLoader(mastertest.MasterTest): def test_run_once(self): - s = state.State() o = options.Options(scripts=[]) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sl = script.ScriptLoader() m.addons.add(sl) @@ -188,9 +167,8 @@ class TestScriptLoader(mastertest.MasterTest): ) def test_simple(self): - s = state.State() o = options.Options(scripts=[]) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sc = script.ScriptLoader() m.addons.add(sc) assert len(m.addons) == 1 @@ -204,16 +182,14 @@ class TestScriptLoader(mastertest.MasterTest): assert len(m.addons) == 1 def test_dupes(self): - s = state.State() o = options.Options(scripts=["one", "one"]) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sc = script.ScriptLoader() tutils.raises(exceptions.OptionsError, m.addons.add, o, sc) def test_order(self): rec = tutils.test_data.path("data/addonscripts/recorder.py") - s = state.State() o = options.Options( scripts = [ "%s %s" % (rec, "a"), @@ -221,7 +197,7 @@ class TestScriptLoader(mastertest.MasterTest): "%s %s" % (rec, "c"), ] ) - m = mastertest.RecordingMaster(o, None, s) + m = mastertest.RecordingMaster(o, None) sc = script.ScriptLoader() m.addons.add(sc) diff --git a/test/mitmproxy/builtins/test_serverplayback.py b/test/mitmproxy/builtins/test_serverplayback.py index 55ce9350..b97c01dd 100644 --- a/test/mitmproxy/builtins/test_serverplayback.py +++ b/test/mitmproxy/builtins/test_serverplayback.py @@ -4,7 +4,6 @@ import netlib.tutils from mitmproxy.builtins import serverplayback from mitmproxy import options from mitmproxy import exceptions -from mitmproxy import flow class TestServerPlayback: @@ -238,10 +237,9 @@ class TestServerPlayback: assert not s._hash(r) == s._hash(r2) def test_server_playback_full(self): - state = flow.State() s = serverplayback.ServerPlayback() o = options.Options(refresh_server_playback = True, keepserving=False) - m = mastertest.RecordingMaster(o, None, state) + m = mastertest.RecordingMaster(o, None) m.addons.add(s) f = tutils.tflow() @@ -268,10 +266,9 @@ class TestServerPlayback: assert s.stop def test_server_playback_kill(self): - state = flow.State() s = serverplayback.ServerPlayback() o = options.Options(refresh_server_playback = True, replay_kill_extra=True) - m = mastertest.RecordingMaster(o, None, state) + m = mastertest.RecordingMaster(o, None) m.addons.add(s) f = tutils.tflow() diff --git a/test/mitmproxy/builtins/test_setheaders.py b/test/mitmproxy/builtins/test_setheaders.py index d148234d..234341f4 100644 --- a/test/mitmproxy/builtins/test_setheaders.py +++ b/test/mitmproxy/builtins/test_setheaders.py @@ -1,15 +1,13 @@ from .. import tutils, mastertest from mitmproxy.builtins import setheaders -from mitmproxy.flow import state from mitmproxy import options class TestSetHeaders(mastertest.MasterTest): def mkmaster(self, **opts): - s = state.State() o = options.Options(**opts) - m = mastertest.RecordingMaster(o, None, s) + m = mastertest.RecordingMaster(o, None) sh = setheaders.SetHeaders() m.addons.add(sh) return m, sh diff --git a/test/mitmproxy/builtins/test_stickyauth.py b/test/mitmproxy/builtins/test_stickyauth.py index 8b939c24..25141554 100644 --- a/test/mitmproxy/builtins/test_stickyauth.py +++ b/test/mitmproxy/builtins/test_stickyauth.py @@ -1,15 +1,13 @@ from .. import tutils, mastertest from mitmproxy.builtins import stickyauth from mitmproxy.flow import master -from mitmproxy.flow import state from mitmproxy import options class TestStickyAuth(mastertest.MasterTest): def test_simple(self): - s = state.State() o = options.Options(stickyauth = ".*") - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sa = stickyauth.StickyAuth() m.addons.add(sa) diff --git a/test/mitmproxy/builtins/test_stickycookie.py b/test/mitmproxy/builtins/test_stickycookie.py index 4d3b7cb3..28b05a11 100644 --- a/test/mitmproxy/builtins/test_stickycookie.py +++ b/test/mitmproxy/builtins/test_stickycookie.py @@ -1,7 +1,6 @@ from .. import tutils, mastertest from mitmproxy.builtins import stickycookie from mitmproxy.flow import master -from mitmproxy.flow import state from mitmproxy import options from netlib import tutils as ntutils @@ -13,12 +12,11 @@ def test_domain_match(): class TestStickyCookie(mastertest.MasterTest): def mk(self): - s = state.State() o = options.Options(stickycookie = ".*") - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sc = stickycookie.StickyCookie() m.addons.add(sc) - return s, m, sc + return m, sc def test_config(self): sc = stickycookie.StickyCookie() @@ -29,7 +27,7 @@ class TestStickyCookie(mastertest.MasterTest): ) def test_simple(self): - s, m, sc = self.mk() + m, sc = self.mk() m.addons.add(sc) f = tutils.tflow(resp=True) @@ -47,44 +45,44 @@ class TestStickyCookie(mastertest.MasterTest): m.request(f) assert f.request.headers["cookie"] == "foo=bar" - def _response(self, s, m, sc, cookie, host): + def _response(self, m, sc, cookie, host): f = tutils.tflow(req=ntutils.treq(host=host, port=80), resp=True) f.response.headers["Set-Cookie"] = cookie m.response(f) return f def test_response(self): - s, m, sc = self.mk() + m, sc = self.mk() c = "SSID=mooo; domain=.google.com, FOO=bar; Domain=.google.com; Path=/; " \ "Expires=Wed, 13-Jan-2021 22:23:01 GMT; Secure; " - self._response(s, m, sc, c, "host") + self._response(m, sc, c, "host") assert not sc.jar.keys() - self._response(s, m, sc, c, "www.google.com") + self._response(m, sc, c, "www.google.com") assert sc.jar.keys() sc.jar.clear() self._response( - s, m, sc, "SSID=mooo", "www.google.com" + m, sc, "SSID=mooo", "www.google.com" ) assert list(sc.jar.keys())[0] == ('www.google.com', 80, '/') def test_response_multiple(self): - s, m, sc = self.mk() + m, sc = self.mk() # Test setting of multiple cookies c1 = "somecookie=test; Path=/" c2 = "othercookie=helloworld; Path=/" - f = self._response(s, m, sc, c1, "www.google.com") + f = self._response(m, sc, c1, "www.google.com") f.response.headers["Set-Cookie"] = c2 m.response(f) googlekey = list(sc.jar.keys())[0] assert len(sc.jar[googlekey].keys()) == 2 def test_response_weird(self): - s, m, sc = self.mk() + m, sc = self.mk() # Test setting of weird cookie keys f = tutils.tflow(req=ntutils.treq(host="www.google.com", port=80), resp=True) @@ -100,12 +98,12 @@ class TestStickyCookie(mastertest.MasterTest): assert len(sc.jar[googlekey].keys()) == len(cs) def test_response_overwrite(self): - s, m, sc = self.mk() + m, sc = self.mk() # Test overwriting of a cookie value c1 = "somecookie=helloworld; Path=/" c2 = "somecookie=newvalue; Path=/" - f = self._response(s, m, sc, c1, "www.google.com") + f = self._response(m, sc, c1, "www.google.com") f.response.headers["Set-Cookie"] = c2 m.response(f) googlekey = list(sc.jar.keys())[0] @@ -113,19 +111,19 @@ class TestStickyCookie(mastertest.MasterTest): assert list(sc.jar[googlekey]["somecookie"].items())[0][1] == "newvalue" def test_response_delete(self): - s, m, sc = self.mk() + m, sc = self.mk() # Test that a cookie is be deleted # by setting the expire time in the past - f = self._response(s, m, sc, "duffer=zafar; Path=/", "www.google.com") + f = self._response(m, sc, "duffer=zafar; Path=/", "www.google.com") f.response.headers["Set-Cookie"] = "duffer=; Expires=Thu, 01-Jan-1970 00:00:00 GMT" m.response(f) assert not sc.jar.keys() def test_request(self): - s, m, sc = self.mk() + m, sc = self.mk() - f = self._response(s, m, sc, "SSID=mooo", "www.google.com") + f = self._response(m, sc, "SSID=mooo", "www.google.com") assert "cookie" not in f.request.headers m.request(f) assert "cookie" in f.request.headers diff --git a/test/mitmproxy/builtins/test_streambodies.py b/test/mitmproxy/builtins/test_streambodies.py index 0e8a82f2..0a306681 100644 --- a/test/mitmproxy/builtins/test_streambodies.py +++ b/test/mitmproxy/builtins/test_streambodies.py @@ -1,5 +1,4 @@ from .. import tutils, mastertest -from mitmproxy.flow import state from mitmproxy.flow import master from mitmproxy import options @@ -8,9 +7,8 @@ from mitmproxy.builtins import streambodies class TestStreamBodies(mastertest.MasterTest): def test_simple(self): - s = state.DummyState() o = options.Options(stream_large_bodies = 10) - m = master.FlowMaster(o, None, s) + m = master.FlowMaster(o, None) sa = streambodies.StreamBodies() m.addons.add(sa) diff --git a/test/mitmproxy/mastertest.py b/test/mitmproxy/mastertest.py index 5605b1a6..ae11e577 100644 --- a/test/mitmproxy/mastertest.py +++ b/test/mitmproxy/mastertest.py @@ -50,8 +50,7 @@ class RecordingMaster(master.FlowMaster): @contextlib.contextmanager def mockctx(): - state = flow.State() o = options.Options(refresh_server_playback = True, keepserving=False) - m = RecordingMaster(o, proxy.DummyServer(o), state) + m = RecordingMaster(o, proxy.DummyServer(o)) with m.handlecontext(): yield diff --git a/test/mitmproxy/script/test_concurrent.py b/test/mitmproxy/script/test_concurrent.py index 5f943aeb..eb0e83f7 100644 --- a/test/mitmproxy/script/test_concurrent.py +++ b/test/mitmproxy/script/test_concurrent.py @@ -3,7 +3,6 @@ from mitmproxy import controller from mitmproxy.builtins import script from mitmproxy import options from mitmproxy.flow import master -from mitmproxy.flow import state import time @@ -16,8 +15,7 @@ class Thing: class TestConcurrent(mastertest.MasterTest): @tutils.skip_appveyor def test_concurrent(self): - s = state.State() - m = master.FlowMaster(options.Options(), None, s) + m = master.FlowMaster(options.Options(), None) sc = script.Script( tutils.test_data.path( "data/addonscripts/concurrent_decorator.py" @@ -34,8 +32,7 @@ class TestConcurrent(mastertest.MasterTest): raise ValueError("Script never acked") def test_concurrent_err(self): - s = state.State() - m = mastertest.RecordingMaster(options.Options(), None, s) + m = mastertest.RecordingMaster(options.Options(), None) sc = script.Script( tutils.test_data.path( "data/addonscripts/concurrent_decorator_err.py" diff --git a/test/mitmproxy/test_examples.py b/test/mitmproxy/test_examples.py index 05b7544f..f824584b 100644 --- a/test/mitmproxy/test_examples.py +++ b/test/mitmproxy/test_examples.py @@ -6,7 +6,6 @@ from mitmproxy import options from mitmproxy import contentviews from mitmproxy.builtins import script from mitmproxy.flow import master -from mitmproxy.flow import state import netlib.utils @@ -32,7 +31,7 @@ class RaiseMaster(master.FlowMaster): def tscript(cmd, args=""): o = options.Options() cmd = example_dir.path(cmd) + " " + args - m = RaiseMaster(o, None, state.State()) + m = RaiseMaster(o, None) sc = script.Script(cmd) m.addons.add(sc) return m, sc diff --git a/test/mitmproxy/test_flow.py b/test/mitmproxy/test_flow.py index 90b9b61d..78cfca99 100644 --- a/test/mitmproxy/test_flow.py +++ b/test/mitmproxy/test_flow.py @@ -121,7 +121,6 @@ class TestHTTPFlow: f = tutils.tflow() f.reply.handle() f.intercept(fm) - assert fm.handle_intercept.called assert f.killable f.kill(fm) assert not f.killable @@ -129,8 +128,10 @@ class TestHTTPFlow: assert f.reply.value == Kill def test_killall(self): + srv = DummyServer(None) s = flow.State() - fm = flow.FlowMaster(None, None, s) + fm = flow.FlowMaster(None, srv) + fm.addons.add(s) f = tutils.tflow() f.reply.handle() @@ -140,12 +141,12 @@ class TestHTTPFlow: for i in s.view: assert "killed" in str(i.error) - def test_accept_intercept(self): + def test_resume(self): f = tutils.tflow() f.reply.handle() f.intercept(mock.Mock()) assert f.reply.state == "taken" - f.accept_intercept(mock.Mock()) + f.resume(mock.Mock()) assert f.reply.state == "committed" def test_replace_unicode(self): @@ -382,7 +383,8 @@ class TestSerialize: def test_load_flows(self): r = self._treader() s = flow.State() - fm = flow.FlowMaster(None, None, s) + fm = flow.FlowMaster(None, None) + fm.addons.add(s) fm.load_flows(r) assert len(s.flows) == 6 @@ -394,7 +396,8 @@ class TestSerialize: upstream_server="https://use-this-domain" ) conf = ProxyConfig(opts) - fm = flow.FlowMaster(opts, DummyServer(conf), s) + fm = flow.FlowMaster(opts, DummyServer(conf)) + fm.addons.add(s) fm.load_flows(r) assert s.flows[0].request.host == "use-this-domain" @@ -440,8 +443,7 @@ class TestSerialize: class TestFlowMaster: def test_replay(self): - s = flow.State() - fm = flow.FlowMaster(None, None, s) + fm = flow.FlowMaster(None, None) f = tutils.tflow(resp=True) f.request.content = None tutils.raises("missing", fm.replay_request, f) @@ -453,13 +455,13 @@ class TestFlowMaster: tutils.raises("live", fm.replay_request, f) def test_create_flow(self): - s = flow.State() - fm = flow.FlowMaster(None, None, s) + fm = flow.FlowMaster(None, None) assert fm.create_request("GET", "http", "example.com", 80, "/") def test_all(self): s = flow.State() - fm = flow.FlowMaster(None, None, s) + fm = flow.FlowMaster(None, None) + fm.addons.add(s) f = tutils.tflow(req=None) fm.clientconnect(f.client_conn) f.request = HTTPRequest.wrap(netlib.tutils.treq()) diff --git a/test/mitmproxy/test_flow_state.py b/test/mitmproxy/test_flow_state.py index cc05537d..a658e13e 100644 --- a/test/mitmproxy/test_flow_state.py +++ b/test/mitmproxy/test_flow_state.py @@ -5,11 +5,13 @@ from . import tutils class TestState: def test_duplicate_flow(self): s = flow.State() - fm = flow.FlowMaster(None, None, s) + fm = flow.FlowMaster(None, None) + fm.addons.add(s) f = tutils.tflow(resp=True) fm.load_flow(f) assert s.flow_count() == 1 - f2 = fm.state.duplicate_flow(f) + + f2 = s.duplicate_flow(f) assert f2.response assert s.flow_count() == 2 assert s.index(f2) == 1 diff --git a/test/mitmproxy/test_server.py b/test/mitmproxy/test_server.py index e9db210e..0938571b 100644 --- a/test/mitmproxy/test_server.py +++ b/test/mitmproxy/test_server.py @@ -1,13 +1,12 @@ import os import socket import time -import types import netlib.tutils from mitmproxy import controller from mitmproxy import options from mitmproxy.builtins import script -from mitmproxy.models import Error, HTTPResponse, HTTPFlow +from mitmproxy.models import HTTPResponse, HTTPFlow from mitmproxy.proxy.config import HostMatcher, parse_server_spec from netlib import tcp, http, socks from netlib.certutils import SSLCert @@ -986,6 +985,17 @@ class TestUpstreamProxySSL( assert self.chain[1].tmaster.state.flow_count() == 1 +class RequestKiller: + def __init__(self, exclude): + self.exclude = exclude + self.k = 0 + + def request(self, f): + self.k += 1 + if self.k not in self.exclude: + f.reply.kill() + + class TestProxyChainingSSLReconnect(tservers.HTTPUpstreamProxyTest): ssl = True @@ -995,39 +1005,18 @@ class TestProxyChainingSSLReconnect(tservers.HTTPUpstreamProxyTest): If we have a disconnect on a secure connection that's transparently proxified to an upstream http proxy, we need to send the CONNECT request again. """ - - def kill_requests(master, attr, exclude): - k = [0] # variable scope workaround: put into array - _func = getattr(master, attr) - - @controller.handler - def handler(*args): - f = args[-1] - k[0] += 1 - if not (k[0] in exclude): - f.client_conn.finish() - f.error = Error("terminated") - f.reply.kill() - return _func(f) - - setattr(master, attr, types.MethodType(handler, master)) - - kill_requests( - self.chain[1].tmaster, - "request", - exclude = [ - # fail first request - 2, # allow second request - ] + self.chain[1].tmaster.addons.add( + RequestKiller([2]) + ) + self.chain[0].tmaster.addons.add( + RequestKiller( + [ + 1, # CONNECT + 3, # reCONNECT + 4 # request + ] + ) ) - - kill_requests(self.chain[0].tmaster, "request", - exclude=[ - 1, # CONNECT - # fail first request - 3, # reCONNECT - 4, # request - ]) p = self.pathoc() with p.connect(): diff --git a/test/mitmproxy/tservers.py b/test/mitmproxy/tservers.py index 431e0f90..c4230d6f 100644 --- a/test/mitmproxy/tservers.py +++ b/test/mitmproxy/tservers.py @@ -7,6 +7,7 @@ import sys from mitmproxy.proxy.config import ProxyConfig from mitmproxy.proxy.server import ProxyServer +from mitmproxy.flow import state import pathod.test import pathod.pathoc from mitmproxy import flow, controller, options @@ -34,8 +35,9 @@ class TestMaster(flow.FlowMaster): def __init__(self, opts, config): s = ProxyServer(config) - state = flow.State() - flow.FlowMaster.__init__(self, opts, s, state) + flow.FlowMaster.__init__(self, opts, s) + self.state = state.State() + self.addons.add(self.state) self.addons.add(*builtins.default_addons()) self.apps.add(testapp, "testapp", 80) self.apps.add(errapp, "errapp", 80) @@ -116,8 +118,8 @@ class ProxyTestBase: raise def setup(self): - self.master.clear_log() self.master.state.clear() + self.master.clear_log() self.server.clear_log() self.server2.clear_log() |