diff options
author | Joshua Watt <jpewhacker@gmail.com> | 2019-01-04 10:20:14 -0600 |
---|---|---|
committer | Richard Purdie <richard.purdie@linuxfoundation.org> | 2019-01-08 11:16:03 +0000 |
commit | df85dd54857b06ca514d01091929b577753ce94e (patch) | |
tree | beff08af54f80783dcacef45fc34220dadca6a33 /bitbake/lib/hashserv/__init__.py | |
parent | cea00c128311539a870d0cd233366480ddaff605 (diff) | |
download | poky-df85dd54857b06ca514d01091929b577753ce94e.tar.gz |
bitbake: bitbake: hashserv: Add hash equivalence reference server
Implements a reference implementation of the hash equivalence server.
This server has minimal dependencies (and no dependencies outside of the
standard Python library), and implements the minimum required to be a
conforming hash equivalence server.
[YOCTO #13030]
(Bitbake rev: 1bb2ad0b44b94ee04870bf3f7dac4e663bed6e4d)
Signed-off-by: Joshua Watt <JPEWhacker@gmail.com>
Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
Diffstat (limited to 'bitbake/lib/hashserv/__init__.py')
-rw-r--r-- | bitbake/lib/hashserv/__init__.py | 152 |
1 files changed, 152 insertions, 0 deletions
diff --git a/bitbake/lib/hashserv/__init__.py b/bitbake/lib/hashserv/__init__.py new file mode 100644 index 0000000000..46bca7cab3 --- /dev/null +++ b/bitbake/lib/hashserv/__init__.py | |||
@@ -0,0 +1,152 @@ | |||
1 | # Copyright (C) 2018 Garmin Ltd. | ||
2 | # | ||
3 | # This program is free software; you can redistribute it and/or modify | ||
4 | # it under the terms of the GNU General Public License version 2 as | ||
5 | # published by the Free Software Foundation. | ||
6 | # | ||
7 | # This program is distributed in the hope that it will be useful, | ||
8 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
9 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
10 | # GNU General Public License for more details. | ||
11 | # | ||
12 | # You should have received a copy of the GNU General Public License along | ||
13 | # with this program; if not, write to the Free Software Foundation, Inc., | ||
14 | # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | ||
15 | |||
16 | from http.server import BaseHTTPRequestHandler, HTTPServer | ||
17 | import contextlib | ||
18 | import urllib.parse | ||
19 | import sqlite3 | ||
20 | import json | ||
21 | import traceback | ||
22 | import logging | ||
23 | from datetime import datetime | ||
24 | |||
25 | logger = logging.getLogger('hashserv') | ||
26 | |||
27 | class HashEquivalenceServer(BaseHTTPRequestHandler): | ||
28 | def log_message(self, f, *args): | ||
29 | logger.debug(f, *args) | ||
30 | |||
31 | def do_GET(self): | ||
32 | try: | ||
33 | p = urllib.parse.urlparse(self.path) | ||
34 | |||
35 | if p.path != self.prefix + '/v1/equivalent': | ||
36 | self.send_error(404) | ||
37 | return | ||
38 | |||
39 | query = urllib.parse.parse_qs(p.query, strict_parsing=True) | ||
40 | method = query['method'][0] | ||
41 | taskhash = query['taskhash'][0] | ||
42 | |||
43 | d = None | ||
44 | with contextlib.closing(self.db.cursor()) as cursor: | ||
45 | cursor.execute('SELECT taskhash, method, unihash FROM tasks_v1 WHERE method=:method AND taskhash=:taskhash ORDER BY created ASC LIMIT 1', | ||
46 | {'method': method, 'taskhash': taskhash}) | ||
47 | |||
48 | row = cursor.fetchone() | ||
49 | |||
50 | if row is not None: | ||
51 | logger.debug('Found equivalent task %s', row['taskhash']) | ||
52 | d = {k: row[k] for k in ('taskhash', 'method', 'unihash')} | ||
53 | |||
54 | self.send_response(200) | ||
55 | self.send_header('Content-Type', 'application/json; charset=utf-8') | ||
56 | self.end_headers() | ||
57 | self.wfile.write(json.dumps(d).encode('utf-8')) | ||
58 | except: | ||
59 | logger.exception('Error in GET') | ||
60 | self.send_error(400, explain=traceback.format_exc()) | ||
61 | return | ||
62 | |||
63 | def do_POST(self): | ||
64 | try: | ||
65 | p = urllib.parse.urlparse(self.path) | ||
66 | |||
67 | if p.path != self.prefix + '/v1/equivalent': | ||
68 | self.send_error(404) | ||
69 | return | ||
70 | |||
71 | length = int(self.headers['content-length']) | ||
72 | data = json.loads(self.rfile.read(length).decode('utf-8')) | ||
73 | |||
74 | with contextlib.closing(self.db.cursor()) as cursor: | ||
75 | cursor.execute(''' | ||
76 | SELECT taskhash, method, unihash FROM tasks_v1 WHERE method=:method AND outhash=:outhash | ||
77 | ORDER BY CASE WHEN taskhash=:taskhash THEN 1 ELSE 2 END, | ||
78 | created ASC | ||
79 | LIMIT 1 | ||
80 | ''', {k: data[k] for k in ('method', 'outhash', 'taskhash')}) | ||
81 | |||
82 | row = cursor.fetchone() | ||
83 | |||
84 | if row is None or row['taskhash'] != data['taskhash']: | ||
85 | unihash = data['unihash'] | ||
86 | if row is not None: | ||
87 | unihash = row['unihash'] | ||
88 | |||
89 | insert_data = { | ||
90 | 'method': data['method'], | ||
91 | 'outhash': data['outhash'], | ||
92 | 'taskhash': data['taskhash'], | ||
93 | 'unihash': unihash, | ||
94 | 'created': datetime.now() | ||
95 | } | ||
96 | |||
97 | for k in ('owner', 'PN', 'PV', 'PR', 'task', 'outhash_siginfo'): | ||
98 | if k in data: | ||
99 | insert_data[k] = data[k] | ||
100 | |||
101 | cursor.execute('''INSERT INTO tasks_v1 (%s) VALUES (%s)''' % ( | ||
102 | ', '.join(sorted(insert_data.keys())), | ||
103 | ', '.join(':' + k for k in sorted(insert_data.keys()))), | ||
104 | insert_data) | ||
105 | |||
106 | logger.info('Adding taskhash %s with unihash %s', data['taskhash'], unihash) | ||
107 | cursor.execute('SELECT taskhash, method, unihash FROM tasks_v1 WHERE id=:id', {'id': cursor.lastrowid}) | ||
108 | row = cursor.fetchone() | ||
109 | |||
110 | self.db.commit() | ||
111 | |||
112 | d = {k: row[k] for k in ('taskhash', 'method', 'unihash')} | ||
113 | |||
114 | self.send_response(200) | ||
115 | self.send_header('Content-Type', 'application/json; charset=utf-8') | ||
116 | self.end_headers() | ||
117 | self.wfile.write(json.dumps(d).encode('utf-8')) | ||
118 | except: | ||
119 | logger.exception('Error in POST') | ||
120 | self.send_error(400, explain=traceback.format_exc()) | ||
121 | return | ||
122 | |||
123 | def create_server(addr, db, prefix=''): | ||
124 | class Handler(HashEquivalenceServer): | ||
125 | pass | ||
126 | |||
127 | Handler.prefix = prefix | ||
128 | Handler.db = db | ||
129 | db.row_factory = sqlite3.Row | ||
130 | |||
131 | with contextlib.closing(db.cursor()) as cursor: | ||
132 | cursor.execute(''' | ||
133 | CREATE TABLE IF NOT EXISTS tasks_v1 ( | ||
134 | id INTEGER PRIMARY KEY AUTOINCREMENT, | ||
135 | method TEXT NOT NULL, | ||
136 | outhash TEXT NOT NULL, | ||
137 | taskhash TEXT NOT NULL, | ||
138 | unihash TEXT NOT NULL, | ||
139 | created DATETIME, | ||
140 | |||
141 | -- Optional fields | ||
142 | owner TEXT, | ||
143 | PN TEXT, | ||
144 | PV TEXT, | ||
145 | PR TEXT, | ||
146 | task TEXT, | ||
147 | outhash_siginfo TEXT | ||
148 | ) | ||
149 | ''') | ||
150 | |||
151 | logger.info('Starting server on %s', addr) | ||
152 | return HTTPServer(addr, Handler) | ||