Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(653)

Side by Side Diff: recipe_engine/stream.py

Issue 2798223005: Revert of Assert validity of stream parameters. (Closed)
Patch Set: Created 3 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « no previous file | recipe_engine/unittests/stream_test.py » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 # Copyright 2015 The LUCI Authors. All rights reserved. 1 # Copyright 2015 The LUCI Authors. All rights reserved.
2 # Use of this source code is governed under the Apache License, Version 2.0 2 # Use of this source code is governed under the Apache License, Version 2.0
3 # that can be found in the LICENSE file. 3 # that can be found in the LICENSE file.
4 4
5 """Abstract stream interface for representing recipe runs. 5 """Abstract stream interface for representing recipe runs.
6 6
7 We need to create streams for steps (and substeps) and also LOG_LINE steps. 7 We need to create streams for steps (and substeps) and also LOG_LINE steps.
8 LogDog will implement LOG_LINE steps as real logs (i.e. uniformly), but 8 LogDog will implement LOG_LINE steps as real logs (i.e. uniformly), but
9 annotations will implement them differently from normal logs, so we need 9 annotations will implement them differently from normal logs, so we need
10 a way to distinguish. 10 a way to distinguish.
(...skipping 28 matching lines...) Expand all
39 Args: 39 Args:
40 stream (StreamEngine.Stream): The stream to output to. 40 stream (StreamEngine.Stream): The stream to output to.
41 it (iterable): An iterable that yields strings to write. 41 it (iterable): An iterable that yields strings to write.
42 """ 42 """
43 for text in it: 43 for text in it:
44 lines = (text.split('\n') if text else ('',)) 44 lines = (text.split('\n') if text else ('',))
45 for line in lines: 45 for line in lines:
46 stream.write_line(line) 46 stream.write_line(line)
47 47
48 48
49 def _check(cond, msg, **kwargs):
50 """Runtime assertion used for parameter correctness."""
51 if not cond:
52 raise ValueError('Failed assertion: %s %r' % (msg, kwargs))
53
54
55 class StreamEngine(object): 49 class StreamEngine(object):
56
57 class Stream(object): 50 class Stream(object):
58 def write_line(self, line): 51 def write_line(self, line):
59 raise NotImplementedError() 52 raise NotImplementedError()
60 53
61 def write_split(self, string): 54 def write_split(self, string):
62 """Write a string (which may contain newlines) to the stream. It will 55 """Write a string (which may contain newlines) to the stream. It will
63 be terminated by a newline.""" 56 be terminated by a newline."""
64 for actual_line in string.splitlines() or ['']: # preserve empty lines 57 for actual_line in string.splitlines() or ['']: # preserve empty lines
65 self.write_line(actual_line) 58 self.write_line(actual_line)
66 59
(...skipping 26 matching lines...) Expand all
93 raise NotImplementedError() 86 raise NotImplementedError()
94 87
95 def set_build_property(self, key, value): 88 def set_build_property(self, key, value):
96 raise NotImplementedError() 89 raise NotImplementedError()
97 90
98 def trigger(self, trigger_spec): 91 def trigger(self, trigger_spec):
99 raise NotImplementedError() 92 raise NotImplementedError()
100 93
101 def make_step_stream(self, name, **kwargs): 94 def make_step_stream(self, name, **kwargs):
102 """Shorthand for creating a step stream from a step configuration dict.""" 95 """Shorthand for creating a step stream from a step configuration dict."""
103 _check('\n' not in name, 'Stream name must not contain a newline.',
104 name=name)
105 kwargs['name'] = name 96 kwargs['name'] = name
106 return self.new_step_stream(recipe_api.StepConfig.create(**kwargs)) 97 return self.new_step_stream(recipe_api.StepConfig.create(**kwargs))
107 98
108 def new_step_stream(self, step_config): 99 def new_step_stream(self, step_config):
109 """Creates a new StepStream in this engine. 100 """Creates a new StepStream in this engine.
110 101
111 The step will be considered started at the moment this method is called. 102 The step will be considered started at the moment this method is called.
112 103
113 TODO(luqui): allow_subannotations is a bit of a hack, whether to allow 104 TODO(luqui): allow_subannotations is a bit of a hack, whether to allow
114 annotations that this step emits through to the annotator (True), or 105 annotations that this step emits through to the annotator (True), or
(...skipping 201 matching lines...) Expand 10 before | Expand all | Expand 10 after
316 class StepStream(StreamEngine.StepStream): 307 class StepStream(StreamEngine.StepStream):
317 def __init__(self, engine, step_name): 308 def __init__(self, engine, step_name):
318 super(StreamEngineInvariants.StepStream, self).__init__() 309 super(StreamEngineInvariants.StepStream, self).__init__()
319 self._engine = engine 310 self._engine = engine
320 self._step_name = step_name 311 self._step_name = step_name
321 self._open = True 312 self._open = True
322 self._logs = {} 313 self._logs = {}
323 self._status = 'SUCCESS' 314 self._status = 'SUCCESS'
324 315
325 def write_line(self, line): 316 def write_line(self, line):
326 assert '\n' not in line, 'Individual line must not contain a newline.' 317 assert '\n' not in line
327 assert self._open 318 assert self._open
328 319
329 def close(self): 320 def close(self):
330 assert self._open 321 assert self._open
331 for log_name, log in self._logs.iteritems(): 322 for log_name, log in self._logs.iteritems():
332 assert not log._open, 'Log %s still open when closing step %s' % ( 323 assert not log._open, 'Log %s still open when closing step %s' % (
333 log_name, self._step_name) 324 log_name, self._step_name)
334 self._open = False 325 self._open = False
335 326
336 def new_log_stream(self, log_name): 327 def new_log_stream(self, log_name):
337 assert self._open 328 assert self._open
338 assert log_name not in self._logs, 'Log %s already exists in step %s' % ( 329 assert log_name not in self._logs, 'Log %s already exists in step %s' % (
339 log_name, self._step_name) 330 log_name, self._step_name)
340
341 _check('\n' not in log_name, 'Log name must not contain a newline.',
342 log_name=log_name)
343 ret = self._engine.LogStream(self, log_name) 331 ret = self._engine.LogStream(self, log_name)
344 self._logs[log_name] = ret 332 self._logs[log_name] = ret
345 return ret 333 return ret
346 334
347 def add_step_text(self, text): 335 def add_step_text(self, text):
348 _check('\n' not in text, 'Step text must not contain a newline.', 336 pass
349 text=text)
350 337
351 def add_step_summary_text(self, text): 338 def add_step_summary_text(self, text):
352 _check('\n' not in text, 'Step summary text must not contain a newline.', 339 pass
353 text=text)
354 340
355 def add_step_link(self, name, url): 341 def add_step_link(self, name, url):
356 _check(isinstance(name, basestring), 'Link name is not a string', 342 assert isinstance(name, basestring), 'Link name %s is not a string' % name
357 name=name) 343 assert isinstance(url, basestring), 'Link url %s is not a string' % url
358 _check('\n' not in name, 'Link name must not contain a newline.',
359 name=name)
360 _check(isinstance(url, basestring),
361 'Link URL is not a string', url=url)
362 _check('\n' not in url, 'Link URL must not contain a newline.',
363 url=url)
364 344
365 def set_step_status(self, status): 345 def set_step_status(self, status):
366 assert status in ('SUCCESS', 'WARNING', 'FAILURE', 'EXCEPTION') 346 assert status in ('SUCCESS', 'WARNING', 'FAILURE', 'EXCEPTION')
367 if status == 'SUCCESS': 347 if status == 'SUCCESS':
368 # A constraint imposed by the annotations implementation 348 # A constraint imposed by the annotations implementation
369 assert self._status == 'SUCCESS', ( 349 assert self._status == 'SUCCESS', (
370 'Cannot set successful status after status is %s' % self._status) 350 'Cannot set successful status after status is %s' % self._status)
371 self._status = status 351 self._status = status
372 352
373 def set_build_property(self, key, value): 353 def set_build_property(self, key, value):
374 _check('\n' not in key, 'Property key must not contain a newline.', 354 pass
375 key=key)
376 _check('\n' not in value, 'Property value must not contain a newline.',
377 value=value)
378 json.loads(value) # value must be a valid JSON object.
379 355
380 def trigger(self, spec): 356 def trigger(self, spec):
381 _check('\n' not in spec, 'Spec must not contain a newline.', 357 assert '\n' not in spec # Spec must fit on one line.
382 spec=spec) 358 json.loads(spec) # Spec must be a valid json object.
383 json.loads(spec) # Spec must be a valid JSON object.
384 359
385 class LogStream(StreamEngine.Stream): 360 class LogStream(StreamEngine.Stream):
386 def __init__(self, step_stream, log_name): 361 def __init__(self, step_stream, log_name):
387 self._step_stream = step_stream 362 self._step_stream = step_stream
388 self._log_name = log_name 363 self._log_name = log_name
389 self._open = True 364 self._open = True
390 365
391 def write_line(self, line): 366 def write_line(self, line):
392 assert '\n' not in line, 'Individual line must not contain a newline.' 367 assert '\n' not in line
393 assert self._step_stream._open 368 assert self._step_stream._open
394 assert self._open 369 assert self._open
395 370
396 def close(self): 371 def close(self):
397 assert self._step_stream._open 372 assert self._step_stream._open
398 assert self._open 373 assert self._open
399 self._open = False 374 self._open = False
400 375
401 def new_step_stream(self, step_config): 376 def new_step_stream(self, step_config):
402 assert step_config.name not in self._streams, ( 377 assert step_config.name not in self._streams, (
(...skipping 152 matching lines...) Expand 10 before | Expand all | Expand 10 after
555 string (which means it might not be valid ascii), we decode the string with 530 string (which means it might not be valid ascii), we decode the string with
556 the 'replace' error mode, which replaces invalid characters with a suitable 531 the 'replace' error mode, which replaces invalid characters with a suitable
557 replacement character. 532 replacement character.
558 """ 533 """
559 try: 534 try:
560 return str(s) 535 return str(s)
561 except UnicodeEncodeError: 536 except UnicodeEncodeError:
562 return s.encode('utf-8', 'replace') 537 return s.encode('utf-8', 'replace')
563 except UnicodeDecodeError: 538 except UnicodeDecodeError:
564 return s.decode('utf-8', 'replace') 539 return s.decode('utf-8', 'replace')
OLDNEW
« no previous file with comments | « no previous file | recipe_engine/unittests/stream_test.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698