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

Side by Side Diff: recipe_engine/stream.py

Issue 2792333003: 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 75 matching lines...) Expand 10 before | Expand all | Expand 10 after
86 raise NotImplementedError() 86 raise NotImplementedError()
87 87
88 def set_build_property(self, key, value): 88 def set_build_property(self, key, value):
89 raise NotImplementedError() 89 raise NotImplementedError()
90 90
91 def trigger(self, trigger_spec): 91 def trigger(self, trigger_spec):
92 raise NotImplementedError() 92 raise NotImplementedError()
93 93
94 def make_step_stream(self, name, **kwargs): 94 def make_step_stream(self, name, **kwargs):
95 """Shorthand for creating a step stream from a step configuration dict.""" 95 """Shorthand for creating a step stream from a step configuration dict."""
96 assert '\n' not in name, 'Stream name must not contain a newline.'
96 kwargs['name'] = name 97 kwargs['name'] = name
97 return self.new_step_stream(recipe_api.StepConfig.create(**kwargs)) 98 return self.new_step_stream(recipe_api.StepConfig.create(**kwargs))
98 99
99 def new_step_stream(self, step_config): 100 def new_step_stream(self, step_config):
100 """Creates a new StepStream in this engine. 101 """Creates a new StepStream in this engine.
101 102
102 The step will be considered started at the moment this method is called. 103 The step will be considered started at the moment this method is called.
103 104
104 TODO(luqui): allow_subannotations is a bit of a hack, whether to allow 105 TODO(luqui): allow_subannotations is a bit of a hack, whether to allow
105 annotations that this step emits through to the annotator (True), or 106 annotations that this step emits through to the annotator (True), or
(...skipping 201 matching lines...) Expand 10 before | Expand all | Expand 10 after
307 class StepStream(StreamEngine.StepStream): 308 class StepStream(StreamEngine.StepStream):
308 def __init__(self, engine, step_name): 309 def __init__(self, engine, step_name):
309 super(StreamEngineInvariants.StepStream, self).__init__() 310 super(StreamEngineInvariants.StepStream, self).__init__()
310 self._engine = engine 311 self._engine = engine
311 self._step_name = step_name 312 self._step_name = step_name
312 self._open = True 313 self._open = True
313 self._logs = {} 314 self._logs = {}
314 self._status = 'SUCCESS' 315 self._status = 'SUCCESS'
315 316
316 def write_line(self, line): 317 def write_line(self, line):
317 assert '\n' not in line 318 assert '\n' not in line, 'Individual line must not contain a newline.'
318 assert self._open 319 assert self._open
319 320
320 def close(self): 321 def close(self):
321 assert self._open 322 assert self._open
322 for log_name, log in self._logs.iteritems(): 323 for log_name, log in self._logs.iteritems():
323 assert not log._open, 'Log %s still open when closing step %s' % ( 324 assert not log._open, 'Log %s still open when closing step %s' % (
324 log_name, self._step_name) 325 log_name, self._step_name)
325 self._open = False 326 self._open = False
326 327
327 def new_log_stream(self, log_name): 328 def new_log_stream(self, log_name):
328 assert self._open 329 assert self._open
329 assert log_name not in self._logs, 'Log %s already exists in step %s' % ( 330 assert log_name not in self._logs, 'Log %s already exists in step %s' % (
330 log_name, self._step_name) 331 log_name, self._step_name)
332 assert '\n' not in log_name, 'Log name must not contain a newline.'
331 ret = self._engine.LogStream(self, log_name) 333 ret = self._engine.LogStream(self, log_name)
332 self._logs[log_name] = ret 334 self._logs[log_name] = ret
333 return ret 335 return ret
334 336
335 def add_step_text(self, text): 337 def add_step_text(self, text):
336 pass 338 assert '\n' not in text, 'Step text must not contain a newline.'
337 339
338 def add_step_summary_text(self, text): 340 def add_step_summary_text(self, text):
339 pass 341 assert '\n' not in text, 'Step summary text must not contain a newline.'
340 342
341 def add_step_link(self, name, url): 343 def add_step_link(self, name, url):
342 assert isinstance(name, basestring), 'Link name %s is not a string' % name 344 assert isinstance(name, basestring), 'Link name %s is not a string' % name
343 assert isinstance(url, basestring), 'Link url %s is not a string' % url 345 assert '\n' not in name, 'Link name must not contain a newline.'
346 assert isinstance(url, basestring), 'Link URL %s is not a string' % url
347 assert '\n' not in url, 'Link URL must not contain a newline.'
iannucci 2017/04/05 02:40:23 Since these values are user-controlled (i.e. they
dnj 2017/04/05 15:08:07 Done.
344 348
345 def set_step_status(self, status): 349 def set_step_status(self, status):
346 assert status in ('SUCCESS', 'WARNING', 'FAILURE', 'EXCEPTION') 350 assert status in ('SUCCESS', 'WARNING', 'FAILURE', 'EXCEPTION')
347 if status == 'SUCCESS': 351 if status == 'SUCCESS':
348 # A constraint imposed by the annotations implementation 352 # A constraint imposed by the annotations implementation
349 assert self._status == 'SUCCESS', ( 353 assert self._status == 'SUCCESS', (
350 'Cannot set successful status after status is %s' % self._status) 354 'Cannot set successful status after status is %s' % self._status)
351 self._status = status 355 self._status = status
352 356
353 def set_build_property(self, key, value): 357 def set_build_property(self, key, value):
354 pass 358 assert '\n' not in key, 'Property key must not contain a newline.'
359 assert '\n' not in value, 'Property value must not contain a newline.'
360 json.loads(value) # value must be a valud JSON object.
iannucci 2017/04/05 02:40:23 nit: valud
dnj 2017/04/05 15:08:07 Done.
355 361
356 def trigger(self, spec): 362 def trigger(self, spec):
357 assert '\n' not in spec # Spec must fit on one line. 363 assert '\n' not in spec, 'Spec must not contain a newline.'
358 json.loads(spec) # Spec must be a valid json object. 364 json.loads(spec) # Spec must be a valid JSON object.
359 365
360 class LogStream(StreamEngine.Stream): 366 class LogStream(StreamEngine.Stream):
361 def __init__(self, step_stream, log_name): 367 def __init__(self, step_stream, log_name):
362 self._step_stream = step_stream 368 self._step_stream = step_stream
363 self._log_name = log_name 369 self._log_name = log_name
364 self._open = True 370 self._open = True
365 371
366 def write_line(self, line): 372 def write_line(self, line):
367 assert '\n' not in line 373 assert '\n' not in line, 'Inividual line must not contain a newline.'
368 assert self._step_stream._open 374 assert self._step_stream._open
369 assert self._open 375 assert self._open
370 376
371 def close(self): 377 def close(self):
372 assert self._step_stream._open 378 assert self._step_stream._open
373 assert self._open 379 assert self._open
374 self._open = False 380 self._open = False
375 381
376 def new_step_stream(self, step_config): 382 def new_step_stream(self, step_config):
377 assert step_config.name not in self._streams, ( 383 assert step_config.name not in self._streams, (
(...skipping 152 matching lines...) Expand 10 before | Expand all | Expand 10 after
530 string (which means it might not be valid ascii), we decode the string with 536 string (which means it might not be valid ascii), we decode the string with
531 the 'replace' error mode, which replaces invalid characters with a suitable 537 the 'replace' error mode, which replaces invalid characters with a suitable
532 replacement character. 538 replacement character.
533 """ 539 """
534 try: 540 try:
535 return str(s) 541 return str(s)
536 except UnicodeEncodeError: 542 except UnicodeEncodeError:
537 return s.encode('utf-8', 'replace') 543 return s.encode('utf-8', 'replace')
538 except UnicodeDecodeError: 544 except UnicodeDecodeError:
539 return s.decode('utf-8', 'replace') 545 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