xref: /OK3568_Linux_fs/yocto/poky/bitbake/lib/hashserv/tests.py (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun#! /usr/bin/env python3
2*4882a593Smuzhiyun#
3*4882a593Smuzhiyun# Copyright (C) 2018-2019 Garmin Ltd.
4*4882a593Smuzhiyun#
5*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0-only
6*4882a593Smuzhiyun#
7*4882a593Smuzhiyun
8*4882a593Smuzhiyunfrom . import create_server, create_client
9*4882a593Smuzhiyunimport hashlib
10*4882a593Smuzhiyunimport logging
11*4882a593Smuzhiyunimport multiprocessing
12*4882a593Smuzhiyunimport os
13*4882a593Smuzhiyunimport sys
14*4882a593Smuzhiyunimport tempfile
15*4882a593Smuzhiyunimport threading
16*4882a593Smuzhiyunimport unittest
17*4882a593Smuzhiyunimport socket
18*4882a593Smuzhiyunimport time
19*4882a593Smuzhiyunimport signal
20*4882a593Smuzhiyun
21*4882a593Smuzhiyundef server_prefunc(server, idx):
22*4882a593Smuzhiyun    logging.basicConfig(level=logging.DEBUG, filename='bbhashserv-%d.log' % idx, filemode='w',
23*4882a593Smuzhiyun                        format='%(levelname)s %(filename)s:%(lineno)d %(message)s')
24*4882a593Smuzhiyun    server.logger.debug("Running server %d" % idx)
25*4882a593Smuzhiyun    sys.stdout = open('bbhashserv-stdout-%d.log' % idx, 'w')
26*4882a593Smuzhiyun    sys.stderr = sys.stdout
27*4882a593Smuzhiyun
28*4882a593Smuzhiyunclass HashEquivalenceTestSetup(object):
29*4882a593Smuzhiyun    METHOD = 'TestMethod'
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun    server_index = 0
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun    def start_server(self, dbpath=None, upstream=None, read_only=False, prefunc=server_prefunc):
34*4882a593Smuzhiyun        self.server_index += 1
35*4882a593Smuzhiyun        if dbpath is None:
36*4882a593Smuzhiyun            dbpath = os.path.join(self.temp_dir.name, "db%d.sqlite" % self.server_index)
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun        def cleanup_server(server):
39*4882a593Smuzhiyun            if server.process.exitcode is not None:
40*4882a593Smuzhiyun                return
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun            server.process.terminate()
43*4882a593Smuzhiyun            server.process.join()
44*4882a593Smuzhiyun
45*4882a593Smuzhiyun        server = create_server(self.get_server_addr(self.server_index),
46*4882a593Smuzhiyun                               dbpath,
47*4882a593Smuzhiyun                               upstream=upstream,
48*4882a593Smuzhiyun                               read_only=read_only)
49*4882a593Smuzhiyun        server.dbpath = dbpath
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun        server.serve_as_process(prefunc=prefunc, args=(self.server_index,))
52*4882a593Smuzhiyun        self.addCleanup(cleanup_server, server)
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun        def cleanup_client(client):
55*4882a593Smuzhiyun            client.close()
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun        client = create_client(server.address)
58*4882a593Smuzhiyun        self.addCleanup(cleanup_client, client)
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun        return (client, server)
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun    def setUp(self):
63*4882a593Smuzhiyun        if sys.version_info < (3, 5, 0):
64*4882a593Smuzhiyun            self.skipTest('Python 3.5 or later required')
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun        self.temp_dir = tempfile.TemporaryDirectory(prefix='bb-hashserv')
67*4882a593Smuzhiyun        self.addCleanup(self.temp_dir.cleanup)
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun        (self.client, self.server) = self.start_server()
70*4882a593Smuzhiyun
71*4882a593Smuzhiyun    def assertClientGetHash(self, client, taskhash, unihash):
72*4882a593Smuzhiyun        result = client.get_unihash(self.METHOD, taskhash)
73*4882a593Smuzhiyun        self.assertEqual(result, unihash)
74*4882a593Smuzhiyun
75*4882a593Smuzhiyun
76*4882a593Smuzhiyunclass HashEquivalenceCommonTests(object):
77*4882a593Smuzhiyun    def test_create_hash(self):
78*4882a593Smuzhiyun        # Simple test that hashes can be created
79*4882a593Smuzhiyun        taskhash = '35788efcb8dfb0a02659d81cf2bfd695fb30faf9'
80*4882a593Smuzhiyun        outhash = '2765d4a5884be49b28601445c2760c5f21e7e5c0ee2b7e3fce98fd7e5970796f'
81*4882a593Smuzhiyun        unihash = 'f46d3fbb439bd9b921095da657a4de906510d2cd'
82*4882a593Smuzhiyun
83*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash, None)
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash, self.METHOD, outhash, unihash)
86*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash, 'Server returned bad unihash')
87*4882a593Smuzhiyun
88*4882a593Smuzhiyun    def test_create_equivalent(self):
89*4882a593Smuzhiyun        # Tests that a second reported task with the same outhash will be
90*4882a593Smuzhiyun        # assigned the same unihash
91*4882a593Smuzhiyun        taskhash = '53b8dce672cb6d0c73170be43f540460bfc347b4'
92*4882a593Smuzhiyun        outhash = '5a9cb1649625f0bf41fc7791b635cd9c2d7118c7f021ba87dcd03f72b67ce7a8'
93*4882a593Smuzhiyun        unihash = 'f37918cc02eb5a520b1aff86faacbc0a38124646'
94*4882a593Smuzhiyun
95*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash, self.METHOD, outhash, unihash)
96*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash, 'Server returned bad unihash')
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun        # Report a different task with the same outhash. The returned unihash
99*4882a593Smuzhiyun        # should match the first task
100*4882a593Smuzhiyun        taskhash2 = '3bf6f1e89d26205aec90da04854fbdbf73afe6b4'
101*4882a593Smuzhiyun        unihash2 = 'af36b199320e611fbb16f1f277d3ee1d619ca58b'
102*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash2, self.METHOD, outhash, unihash2)
103*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash, 'Server returned bad unihash')
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun    def test_duplicate_taskhash(self):
106*4882a593Smuzhiyun        # Tests that duplicate reports of the same taskhash with different
107*4882a593Smuzhiyun        # outhash & unihash always return the unihash from the first reported
108*4882a593Smuzhiyun        # taskhash
109*4882a593Smuzhiyun        taskhash = '8aa96fcffb5831b3c2c0cb75f0431e3f8b20554a'
110*4882a593Smuzhiyun        outhash = 'afe240a439959ce86f5e322f8c208e1fedefea9e813f2140c81af866cc9edf7e'
111*4882a593Smuzhiyun        unihash = '218e57509998197d570e2c98512d0105985dffc9'
112*4882a593Smuzhiyun        self.client.report_unihash(taskhash, self.METHOD, outhash, unihash)
113*4882a593Smuzhiyun
114*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash, unihash)
115*4882a593Smuzhiyun
116*4882a593Smuzhiyun        outhash2 = '0904a7fe3dc712d9fd8a74a616ddca2a825a8ee97adf0bd3fc86082c7639914d'
117*4882a593Smuzhiyun        unihash2 = 'ae9a7d252735f0dafcdb10e2e02561ca3a47314c'
118*4882a593Smuzhiyun        self.client.report_unihash(taskhash, self.METHOD, outhash2, unihash2)
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash, unihash)
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun        outhash3 = '77623a549b5b1a31e3732dfa8fe61d7ce5d44b3370f253c5360e136b852967b4'
123*4882a593Smuzhiyun        unihash3 = '9217a7d6398518e5dc002ed58f2cbbbc78696603'
124*4882a593Smuzhiyun        self.client.report_unihash(taskhash, self.METHOD, outhash3, unihash3)
125*4882a593Smuzhiyun
126*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash, unihash)
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun    def test_huge_message(self):
129*4882a593Smuzhiyun        # Simple test that hashes can be created
130*4882a593Smuzhiyun        taskhash = 'c665584ee6817aa99edfc77a44dd853828279370'
131*4882a593Smuzhiyun        outhash = '3c979c3db45c569f51ab7626a4651074be3a9d11a84b1db076f5b14f7d39db44'
132*4882a593Smuzhiyun        unihash = '90e9bc1d1f094c51824adca7f8ea79a048d68824'
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash, None)
135*4882a593Smuzhiyun
136*4882a593Smuzhiyun        siginfo = "0" * (self.client.max_chunk * 4)
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash, self.METHOD, outhash, unihash, {
139*4882a593Smuzhiyun            'outhash_siginfo': siginfo
140*4882a593Smuzhiyun        })
141*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash, 'Server returned bad unihash')
142*4882a593Smuzhiyun
143*4882a593Smuzhiyun        result_unihash = self.client.get_taskhash(self.METHOD, taskhash, True)
144*4882a593Smuzhiyun        self.assertEqual(result_unihash['taskhash'], taskhash)
145*4882a593Smuzhiyun        self.assertEqual(result_unihash['unihash'], unihash)
146*4882a593Smuzhiyun        self.assertEqual(result_unihash['method'], self.METHOD)
147*4882a593Smuzhiyun
148*4882a593Smuzhiyun        result_outhash = self.client.get_outhash(self.METHOD, outhash, taskhash)
149*4882a593Smuzhiyun        self.assertEqual(result_outhash['taskhash'], taskhash)
150*4882a593Smuzhiyun        self.assertEqual(result_outhash['method'], self.METHOD)
151*4882a593Smuzhiyun        self.assertEqual(result_outhash['unihash'], unihash)
152*4882a593Smuzhiyun        self.assertEqual(result_outhash['outhash'], outhash)
153*4882a593Smuzhiyun        self.assertEqual(result_outhash['outhash_siginfo'], siginfo)
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun    def test_stress(self):
156*4882a593Smuzhiyun        def query_server(failures):
157*4882a593Smuzhiyun            client = Client(self.server.address)
158*4882a593Smuzhiyun            try:
159*4882a593Smuzhiyun                for i in range(1000):
160*4882a593Smuzhiyun                    taskhash = hashlib.sha256()
161*4882a593Smuzhiyun                    taskhash.update(str(i).encode('utf-8'))
162*4882a593Smuzhiyun                    taskhash = taskhash.hexdigest()
163*4882a593Smuzhiyun                    result = client.get_unihash(self.METHOD, taskhash)
164*4882a593Smuzhiyun                    if result != taskhash:
165*4882a593Smuzhiyun                        failures.append("taskhash mismatch: %s != %s" % (result, taskhash))
166*4882a593Smuzhiyun            finally:
167*4882a593Smuzhiyun                client.close()
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun        # Report hashes
170*4882a593Smuzhiyun        for i in range(1000):
171*4882a593Smuzhiyun            taskhash = hashlib.sha256()
172*4882a593Smuzhiyun            taskhash.update(str(i).encode('utf-8'))
173*4882a593Smuzhiyun            taskhash = taskhash.hexdigest()
174*4882a593Smuzhiyun            self.client.report_unihash(taskhash, self.METHOD, taskhash, taskhash)
175*4882a593Smuzhiyun
176*4882a593Smuzhiyun        failures = []
177*4882a593Smuzhiyun        threads = [threading.Thread(target=query_server, args=(failures,)) for t in range(100)]
178*4882a593Smuzhiyun
179*4882a593Smuzhiyun        for t in threads:
180*4882a593Smuzhiyun            t.start()
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun        for t in threads:
183*4882a593Smuzhiyun            t.join()
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun        self.assertFalse(failures)
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun    def test_upstream_server(self):
188*4882a593Smuzhiyun        # Tests upstream server support. This is done by creating two servers
189*4882a593Smuzhiyun        # that share a database file. The downstream server has it upstream
190*4882a593Smuzhiyun        # set to the test server, whereas the side server doesn't. This allows
191*4882a593Smuzhiyun        # verification that the hash requests are being proxied to the upstream
192*4882a593Smuzhiyun        # server by verifying that they appear on the downstream client, but not
193*4882a593Smuzhiyun        # the side client. It also verifies that the results are pulled into
194*4882a593Smuzhiyun        # the downstream database by checking that the downstream and side servers
195*4882a593Smuzhiyun        # match after the downstream is done waiting for all backfill tasks
196*4882a593Smuzhiyun        (down_client, down_server) = self.start_server(upstream=self.server.address)
197*4882a593Smuzhiyun        (side_client, side_server) = self.start_server(dbpath=down_server.dbpath)
198*4882a593Smuzhiyun
199*4882a593Smuzhiyun        def check_hash(taskhash, unihash, old_sidehash):
200*4882a593Smuzhiyun            nonlocal down_client
201*4882a593Smuzhiyun            nonlocal side_client
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun            # check upstream server
204*4882a593Smuzhiyun            self.assertClientGetHash(self.client, taskhash, unihash)
205*4882a593Smuzhiyun
206*4882a593Smuzhiyun            # Hash should *not* be present on the side server
207*4882a593Smuzhiyun            self.assertClientGetHash(side_client, taskhash, old_sidehash)
208*4882a593Smuzhiyun
209*4882a593Smuzhiyun            # Hash should be present on the downstream server, since it
210*4882a593Smuzhiyun            # will defer to the upstream server. This will trigger
211*4882a593Smuzhiyun            # the backfill in the downstream server
212*4882a593Smuzhiyun            self.assertClientGetHash(down_client, taskhash, unihash)
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun            # After waiting for the downstream client to finish backfilling the
215*4882a593Smuzhiyun            # task from the upstream server, it should appear in the side server
216*4882a593Smuzhiyun            # since the database is populated
217*4882a593Smuzhiyun            down_client.backfill_wait()
218*4882a593Smuzhiyun            self.assertClientGetHash(side_client, taskhash, unihash)
219*4882a593Smuzhiyun
220*4882a593Smuzhiyun        # Basic report
221*4882a593Smuzhiyun        taskhash = '8aa96fcffb5831b3c2c0cb75f0431e3f8b20554a'
222*4882a593Smuzhiyun        outhash = 'afe240a439959ce86f5e322f8c208e1fedefea9e813f2140c81af866cc9edf7e'
223*4882a593Smuzhiyun        unihash = '218e57509998197d570e2c98512d0105985dffc9'
224*4882a593Smuzhiyun        self.client.report_unihash(taskhash, self.METHOD, outhash, unihash)
225*4882a593Smuzhiyun
226*4882a593Smuzhiyun        check_hash(taskhash, unihash, None)
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun        # Duplicated taskhash with multiple output hashes and unihashes.
229*4882a593Smuzhiyun        # All servers should agree with the originally reported hash
230*4882a593Smuzhiyun        outhash2 = '0904a7fe3dc712d9fd8a74a616ddca2a825a8ee97adf0bd3fc86082c7639914d'
231*4882a593Smuzhiyun        unihash2 = 'ae9a7d252735f0dafcdb10e2e02561ca3a47314c'
232*4882a593Smuzhiyun        self.client.report_unihash(taskhash, self.METHOD, outhash2, unihash2)
233*4882a593Smuzhiyun
234*4882a593Smuzhiyun        check_hash(taskhash, unihash, unihash)
235*4882a593Smuzhiyun
236*4882a593Smuzhiyun        # Report an equivalent task. The sideload will originally report
237*4882a593Smuzhiyun        # no unihash until backfilled
238*4882a593Smuzhiyun        taskhash3 = "044c2ec8aaf480685a00ff6ff49e6162e6ad34e1"
239*4882a593Smuzhiyun        unihash3 = "def64766090d28f627e816454ed46894bb3aab36"
240*4882a593Smuzhiyun        self.client.report_unihash(taskhash3, self.METHOD, outhash, unihash3)
241*4882a593Smuzhiyun
242*4882a593Smuzhiyun        check_hash(taskhash3, unihash, None)
243*4882a593Smuzhiyun
244*4882a593Smuzhiyun        # Test that reporting a unihash in the downstream client isn't
245*4882a593Smuzhiyun        # propagating to the upstream server
246*4882a593Smuzhiyun        taskhash4 = "e3da00593d6a7fb435c7e2114976c59c5fd6d561"
247*4882a593Smuzhiyun        outhash4 = "1cf8713e645f491eb9c959d20b5cae1c47133a292626dda9b10709857cbe688a"
248*4882a593Smuzhiyun        unihash4 = "3b5d3d83f07f259e9086fcb422c855286e18a57d"
249*4882a593Smuzhiyun        down_client.report_unihash(taskhash4, self.METHOD, outhash4, unihash4)
250*4882a593Smuzhiyun        down_client.backfill_wait()
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun        self.assertClientGetHash(down_client, taskhash4, unihash4)
253*4882a593Smuzhiyun        self.assertClientGetHash(side_client, taskhash4, unihash4)
254*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash4, None)
255*4882a593Smuzhiyun
256*4882a593Smuzhiyun        # Test that reporting a unihash in the downstream is able to find a
257*4882a593Smuzhiyun        # match which was previously reported to the upstream server
258*4882a593Smuzhiyun        taskhash5 = '35788efcb8dfb0a02659d81cf2bfd695fb30faf9'
259*4882a593Smuzhiyun        outhash5 = '2765d4a5884be49b28601445c2760c5f21e7e5c0ee2b7e3fce98fd7e5970796f'
260*4882a593Smuzhiyun        unihash5 = 'f46d3fbb439bd9b921095da657a4de906510d2cd'
261*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash5, self.METHOD, outhash5, unihash5)
262*4882a593Smuzhiyun
263*4882a593Smuzhiyun        taskhash6 = '35788efcb8dfb0a02659d81cf2bfd695fb30fafa'
264*4882a593Smuzhiyun        unihash6 = 'f46d3fbb439bd9b921095da657a4de906510d2ce'
265*4882a593Smuzhiyun        result = down_client.report_unihash(taskhash6, self.METHOD, outhash5, unihash6)
266*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash5, 'Server failed to copy unihash from upstream')
267*4882a593Smuzhiyun
268*4882a593Smuzhiyun        # Tests read through from server with
269*4882a593Smuzhiyun        taskhash7 = '9d81d76242cc7cfaf7bf74b94b9cd2e29324ed74'
270*4882a593Smuzhiyun        outhash7 = '8470d56547eea6236d7c81a644ce74670ca0bbda998e13c629ef6bb3f0d60b69'
271*4882a593Smuzhiyun        unihash7 = '05d2a63c81e32f0a36542ca677e8ad852365c538'
272*4882a593Smuzhiyun        self.client.report_unihash(taskhash7, self.METHOD, outhash7, unihash7)
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun        result = down_client.get_taskhash(self.METHOD, taskhash7, True)
275*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash7, 'Server failed to copy unihash from upstream')
276*4882a593Smuzhiyun        self.assertEqual(result['outhash'], outhash7, 'Server failed to copy unihash from upstream')
277*4882a593Smuzhiyun        self.assertEqual(result['taskhash'], taskhash7, 'Server failed to copy unihash from upstream')
278*4882a593Smuzhiyun        self.assertEqual(result['method'], self.METHOD)
279*4882a593Smuzhiyun
280*4882a593Smuzhiyun        taskhash8 = '86978a4c8c71b9b487330b0152aade10c1ee58aa'
281*4882a593Smuzhiyun        outhash8 = 'ca8c128e9d9e4a28ef24d0508aa20b5cf880604eacd8f65c0e366f7e0cc5fbcf'
282*4882a593Smuzhiyun        unihash8 = 'd8bcf25369d40590ad7d08c84d538982f2023e01'
283*4882a593Smuzhiyun        self.client.report_unihash(taskhash8, self.METHOD, outhash8, unihash8)
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun        result = down_client.get_outhash(self.METHOD, outhash8, taskhash8)
286*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash8, 'Server failed to copy unihash from upstream')
287*4882a593Smuzhiyun        self.assertEqual(result['outhash'], outhash8, 'Server failed to copy unihash from upstream')
288*4882a593Smuzhiyun        self.assertEqual(result['taskhash'], taskhash8, 'Server failed to copy unihash from upstream')
289*4882a593Smuzhiyun        self.assertEqual(result['method'], self.METHOD)
290*4882a593Smuzhiyun
291*4882a593Smuzhiyun        taskhash9 = 'ae6339531895ddf5b67e663e6a374ad8ec71d81c'
292*4882a593Smuzhiyun        outhash9 = 'afc78172c81880ae10a1fec994b5b4ee33d196a001a1b66212a15ebe573e00b5'
293*4882a593Smuzhiyun        unihash9 = '6662e699d6e3d894b24408ff9a4031ef9b038ee8'
294*4882a593Smuzhiyun        self.client.report_unihash(taskhash9, self.METHOD, outhash9, unihash9)
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun        result = down_client.get_taskhash(self.METHOD, taskhash9, False)
297*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash9, 'Server failed to copy unihash from upstream')
298*4882a593Smuzhiyun        self.assertEqual(result['taskhash'], taskhash9, 'Server failed to copy unihash from upstream')
299*4882a593Smuzhiyun        self.assertEqual(result['method'], self.METHOD)
300*4882a593Smuzhiyun
301*4882a593Smuzhiyun    def test_ro_server(self):
302*4882a593Smuzhiyun        (ro_client, ro_server) = self.start_server(dbpath=self.server.dbpath, read_only=True)
303*4882a593Smuzhiyun
304*4882a593Smuzhiyun        # Report a hash via the read-write server
305*4882a593Smuzhiyun        taskhash = '35788efcb8dfb0a02659d81cf2bfd695fb30faf9'
306*4882a593Smuzhiyun        outhash = '2765d4a5884be49b28601445c2760c5f21e7e5c0ee2b7e3fce98fd7e5970796f'
307*4882a593Smuzhiyun        unihash = 'f46d3fbb439bd9b921095da657a4de906510d2cd'
308*4882a593Smuzhiyun
309*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash, self.METHOD, outhash, unihash)
310*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash, 'Server returned bad unihash')
311*4882a593Smuzhiyun
312*4882a593Smuzhiyun        # Check the hash via the read-only server
313*4882a593Smuzhiyun        self.assertClientGetHash(ro_client, taskhash, unihash)
314*4882a593Smuzhiyun
315*4882a593Smuzhiyun        # Ensure that reporting via the read-only server fails
316*4882a593Smuzhiyun        taskhash2 = 'c665584ee6817aa99edfc77a44dd853828279370'
317*4882a593Smuzhiyun        outhash2 = '3c979c3db45c569f51ab7626a4651074be3a9d11a84b1db076f5b14f7d39db44'
318*4882a593Smuzhiyun        unihash2 = '90e9bc1d1f094c51824adca7f8ea79a048d68824'
319*4882a593Smuzhiyun
320*4882a593Smuzhiyun        with self.assertRaises(ConnectionError):
321*4882a593Smuzhiyun            ro_client.report_unihash(taskhash2, self.METHOD, outhash2, unihash2)
322*4882a593Smuzhiyun
323*4882a593Smuzhiyun        # Ensure that the database was not modified
324*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash2, None)
325*4882a593Smuzhiyun
326*4882a593Smuzhiyun
327*4882a593Smuzhiyun    def test_slow_server_start(self):
328*4882a593Smuzhiyun        # Ensures that the server will exit correctly even if it gets a SIGTERM
329*4882a593Smuzhiyun        # before entering the main loop
330*4882a593Smuzhiyun
331*4882a593Smuzhiyun        event = multiprocessing.Event()
332*4882a593Smuzhiyun
333*4882a593Smuzhiyun        def prefunc(server, idx):
334*4882a593Smuzhiyun            nonlocal event
335*4882a593Smuzhiyun            server_prefunc(server, idx)
336*4882a593Smuzhiyun            event.wait()
337*4882a593Smuzhiyun
338*4882a593Smuzhiyun        def do_nothing(signum, frame):
339*4882a593Smuzhiyun            pass
340*4882a593Smuzhiyun
341*4882a593Smuzhiyun        old_signal = signal.signal(signal.SIGTERM, do_nothing)
342*4882a593Smuzhiyun        self.addCleanup(signal.signal, signal.SIGTERM, old_signal)
343*4882a593Smuzhiyun
344*4882a593Smuzhiyun        _, server = self.start_server(prefunc=prefunc)
345*4882a593Smuzhiyun        server.process.terminate()
346*4882a593Smuzhiyun        time.sleep(30)
347*4882a593Smuzhiyun        event.set()
348*4882a593Smuzhiyun        server.process.join(300)
349*4882a593Smuzhiyun        self.assertIsNotNone(server.process.exitcode, "Server did not exit in a timely manner!")
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun    def test_diverging_report_race(self):
352*4882a593Smuzhiyun        # Tests that a reported task will correctly pick up an updated unihash
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun        # This is a baseline report added to the database to ensure that there
355*4882a593Smuzhiyun        # is something to match against as equivalent
356*4882a593Smuzhiyun        outhash1 = 'afd11c366050bcd75ad763e898e4430e2a60659b26f83fbb22201a60672019fa'
357*4882a593Smuzhiyun        taskhash1 = '3bde230c743fc45ab61a065d7a1815fbfa01c4740e4c895af2eb8dc0f684a4ab'
358*4882a593Smuzhiyun        unihash1 = '3bde230c743fc45ab61a065d7a1815fbfa01c4740e4c895af2eb8dc0f684a4ab'
359*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash1, self.METHOD, outhash1, unihash1)
360*4882a593Smuzhiyun
361*4882a593Smuzhiyun        # Add a report that is equivalent to Task 1. It should ignore the
362*4882a593Smuzhiyun        # provided unihash and report the unihash from task 1
363*4882a593Smuzhiyun        taskhash2 = '6259ae8263bd94d454c086f501c37e64c4e83cae806902ca95b4ab513546b273'
364*4882a593Smuzhiyun        unihash2 = taskhash2
365*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash2, self.METHOD, outhash1, unihash2)
366*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash1)
367*4882a593Smuzhiyun
368*4882a593Smuzhiyun        # Add another report for Task 2, but with a different outhash (e.g. the
369*4882a593Smuzhiyun        # task is non-deterministic). It should still be marked with the Task 1
370*4882a593Smuzhiyun        # unihash because it has the Task 2 taskhash, which is equivalent to
371*4882a593Smuzhiyun        # Task 1
372*4882a593Smuzhiyun        outhash3 = 'd2187ee3a8966db10b34fe0e863482288d9a6185cb8ef58a6c1c6ace87a2f24c'
373*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash2, self.METHOD, outhash3, unihash2)
374*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash1)
375*4882a593Smuzhiyun
376*4882a593Smuzhiyun
377*4882a593Smuzhiyun    def test_diverging_report_reverse_race(self):
378*4882a593Smuzhiyun        # Same idea as the previous test, but Tasks 2 and 3 are reported in
379*4882a593Smuzhiyun        # reverse order the opposite order
380*4882a593Smuzhiyun
381*4882a593Smuzhiyun        outhash1 = 'afd11c366050bcd75ad763e898e4430e2a60659b26f83fbb22201a60672019fa'
382*4882a593Smuzhiyun        taskhash1 = '3bde230c743fc45ab61a065d7a1815fbfa01c4740e4c895af2eb8dc0f684a4ab'
383*4882a593Smuzhiyun        unihash1 = '3bde230c743fc45ab61a065d7a1815fbfa01c4740e4c895af2eb8dc0f684a4ab'
384*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash1, self.METHOD, outhash1, unihash1)
385*4882a593Smuzhiyun
386*4882a593Smuzhiyun        taskhash2 = '6259ae8263bd94d454c086f501c37e64c4e83cae806902ca95b4ab513546b273'
387*4882a593Smuzhiyun        unihash2 = taskhash2
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun        # Report Task 3 first. Since there is nothing else in the database it
390*4882a593Smuzhiyun        # will use the client provided unihash
391*4882a593Smuzhiyun        outhash3 = 'd2187ee3a8966db10b34fe0e863482288d9a6185cb8ef58a6c1c6ace87a2f24c'
392*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash2, self.METHOD, outhash3, unihash2)
393*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash2)
394*4882a593Smuzhiyun
395*4882a593Smuzhiyun        # Report Task 2. This is equivalent to Task 1 but there is already a mapping for
396*4882a593Smuzhiyun        # taskhash2 so it will report unihash2
397*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash2, self.METHOD, outhash1, unihash2)
398*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash2)
399*4882a593Smuzhiyun
400*4882a593Smuzhiyun        # The originally reported unihash for Task 3 should be unchanged even if it
401*4882a593Smuzhiyun        # shares a taskhash with Task 2
402*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash2, unihash2)
403*4882a593Smuzhiyun
404*4882a593Smuzhiyunclass TestHashEquivalenceUnixServer(HashEquivalenceTestSetup, HashEquivalenceCommonTests, unittest.TestCase):
405*4882a593Smuzhiyun    def get_server_addr(self, server_idx):
406*4882a593Smuzhiyun        return "unix://" + os.path.join(self.temp_dir.name, 'sock%d' % server_idx)
407*4882a593Smuzhiyun
408*4882a593Smuzhiyun
409*4882a593Smuzhiyunclass TestHashEquivalenceUnixServerLongPath(HashEquivalenceTestSetup, unittest.TestCase):
410*4882a593Smuzhiyun    DEEP_DIRECTORY = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa/bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb/ccccccccccccccccccccccccccccccccccccccccccc"
411*4882a593Smuzhiyun    def get_server_addr(self, server_idx):
412*4882a593Smuzhiyun        os.makedirs(os.path.join(self.temp_dir.name, self.DEEP_DIRECTORY), exist_ok=True)
413*4882a593Smuzhiyun        return "unix://" + os.path.join(self.temp_dir.name, self.DEEP_DIRECTORY, 'sock%d' % server_idx)
414*4882a593Smuzhiyun
415*4882a593Smuzhiyun
416*4882a593Smuzhiyun    def test_long_sock_path(self):
417*4882a593Smuzhiyun        # Simple test that hashes can be created
418*4882a593Smuzhiyun        taskhash = '35788efcb8dfb0a02659d81cf2bfd695fb30faf9'
419*4882a593Smuzhiyun        outhash = '2765d4a5884be49b28601445c2760c5f21e7e5c0ee2b7e3fce98fd7e5970796f'
420*4882a593Smuzhiyun        unihash = 'f46d3fbb439bd9b921095da657a4de906510d2cd'
421*4882a593Smuzhiyun
422*4882a593Smuzhiyun        self.assertClientGetHash(self.client, taskhash, None)
423*4882a593Smuzhiyun
424*4882a593Smuzhiyun        result = self.client.report_unihash(taskhash, self.METHOD, outhash, unihash)
425*4882a593Smuzhiyun        self.assertEqual(result['unihash'], unihash, 'Server returned bad unihash')
426*4882a593Smuzhiyun
427*4882a593Smuzhiyun
428*4882a593Smuzhiyunclass TestHashEquivalenceTCPServer(HashEquivalenceTestSetup, HashEquivalenceCommonTests, unittest.TestCase):
429*4882a593Smuzhiyun    def get_server_addr(self, server_idx):
430*4882a593Smuzhiyun        # Some hosts cause asyncio module to misbehave, when IPv6 is not enabled.
431*4882a593Smuzhiyun        # If IPv6 is enabled, it should be safe to use localhost directly, in general
432*4882a593Smuzhiyun        # case it is more reliable to resolve the IP address explicitly.
433*4882a593Smuzhiyun        return socket.gethostbyname("localhost") + ":0"
434