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

Side by Side Diff: recipe_modules/path/api.py

Issue 2808713003: path: Add tracking of volatile paths. (Closed)
Patch Set: more generic expression 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_modules/path/config.py » ('j') | recipe_modules/path/config.py » ('J')
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 # Copyright 2013 The LUCI Authors. All rights reserved. 1 # Copyright 2013 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 import functools 5 import functools
6 import itertools 6 import itertools
7 import os 7 import os
8 import tempfile 8 import tempfile
9 9
10 from recipe_engine import recipe_api 10 from recipe_engine import recipe_api
(...skipping 143 matching lines...) Expand 10 before | Expand all | Expand 10 after
154 154
155 def __init__(self, **kwargs): 155 def __init__(self, **kwargs):
156 super(PathApi, self).__init__(**kwargs) 156 super(PathApi, self).__init__(**kwargs)
157 config_types.Path.set_tostring_fn( 157 config_types.Path.set_tostring_fn(
158 PathToString(self, self._test_data)) 158 PathToString(self, self._test_data))
159 config_types.NamedBasePath.set_path_api(self) 159 config_types.NamedBasePath.set_path_api(self)
160 160
161 # Used in mkdtemp when generating and checking expectations. 161 # Used in mkdtemp when generating and checking expectations.
162 self._test_counter = 0 162 self._test_counter = 0
163 163
164 def _read_module_property(self, property_name, default=None):
165 """Reads a path property module from "$recipe_engine/path". If absent,
166 returns the default.
167 """
168 props = self.m.properties.get('$recipe_engine/path', {})
169 return props.get(property_name, default)
170
164 def _read_path(self, property_name, default): # pragma: no cover 171 def _read_path(self, property_name, default): # pragma: no cover
165 """Reads a path from a property. If absent, returns the default. 172 """Reads a path from a property. If absent, returns the default.
166 173
167 Validates that the path is absolute. 174 Validates that the path is absolute.
168 """ 175 """
169 props = self.m.properties.get('$recipe_engine/path', {}) 176 value = self._read_module_property(property_name)
170 value = props.get(property_name)
171 if not value: 177 if not value:
172 assert os.path.isabs(default), default 178 assert os.path.isabs(default), default
173 return default 179 return default
174 if not os.path.isabs(value): 180 if not os.path.isabs(value):
175 raise Error( 181 raise Error(
176 'Path "%s" specified by module property %s is not absolute' % ( 182 'Path "%s" specified by module property %s is not absolute' % (
177 value, property_name)) 183 value, property_name))
178 return value 184 return value
179 185
180 def _ensure_dir(self, path): # pragma: no cover 186 def _ensure_dir(self, path): # pragma: no cover
(...skipping 10 matching lines...) Expand all
191 197
192 tmp_dir = self._read_path('temp_dir', tempfile.gettempdir()) 198 tmp_dir = self._read_path('temp_dir', tempfile.gettempdir())
193 self._ensure_dir(tmp_dir) 199 self._ensure_dir(tmp_dir)
194 self._temp_dir = _split_path(tmp_dir) 200 self._temp_dir = _split_path(tmp_dir)
195 201
196 cache_dir = self._read_path('cache_dir', os.path.join(os.getcwd(), 'cache' )) 202 cache_dir = self._read_path('cache_dir', os.path.join(os.getcwd(), 'cache' ))
197 self._ensure_dir(cache_dir) 203 self._ensure_dir(cache_dir)
198 self._cache_dir = _split_path(cache_dir) 204 self._cache_dir = _split_path(cache_dir)
199 else: 205 else:
200 self._path_mod = fake_path(self, self._test_data.get('exists', [])) 206 self._path_mod = fake_path(self, self._test_data.get('exists', []))
207 self._tmp_dir_volatile = self._test_data.get('tmp_dir_volatile', False)
nodir 2017/04/10 16:59:21 self._tmp_dir_volatile is not used
dnj 2017/04/10 17:05:20 oh whoops, thanks
201 self._startup_cwd = ['/', 'b', 'FakeTestingCWD'] 208 self._startup_cwd = ['/', 'b', 'FakeTestingCWD']
202 # Appended to placeholder '[TMP]' to get fake path in test. 209 # Appended to placeholder '[TMP]' to get fake path in test.
203 self._temp_dir = ['/'] 210 self._temp_dir = ['/']
204 self._cache_dir = ['/', 'b', 'c'] 211 self._cache_dir = ['/', 'b', 'c']
205 212
213 self._volatile_paths = self._read_module_property('volatile', [])
214
206 self.set_config('BASE') 215 self.set_config('BASE')
207 216
208 def mock_add_paths(self, path): 217 def mock_add_paths(self, path):
209 """For testing purposes, assert that |path| exists.""" 218 """For testing purposes, assert that |path| exists."""
210 if self._test_data.enabled: 219 if self._test_data.enabled:
211 self._path_mod.mock_add_paths(path) 220 self._path_mod.mock_add_paths(path)
212 221
213 def assert_absolute(self, path): 222 def assert_absolute(self, path):
214 assert self.abspath(path) == str(path), '%s is not absolute' % path 223 assert self.abspath(path) == str(path), '%s is not absolute' % path
215 224
225 def is_volatile(self, base):
226 """Returns (bool): True if the named path is declared as volatile.
227
228 A path is volatile if it is scoped to the recipe engine execution. Note that
229 some paths may be volatile, but not explicitly declared as such.
230 """
231 return base in self.c.volatile_paths or base in self._volatile_paths
232
216 def mkdtemp(self, prefix): 233 def mkdtemp(self, prefix):
217 """Makes a new temp directory, returns path to it.""" 234 """Makes a new temp directory, returns path to it."""
218 if not self._test_data.enabled: # pragma: no cover 235 if not self._test_data.enabled: # pragma: no cover
219 # New path as str. 236 # New path as str.
220 new_path = tempfile.mkdtemp(prefix=prefix, dir=str(self['tmp_base'])) 237 new_path = tempfile.mkdtemp(prefix=prefix, dir=str(self['tmp_base']))
221 # Ensure it's under self._temp_dir, convert to Path. 238 # Ensure it's under self._temp_dir, convert to Path.
222 new_path = _split_path(new_path) 239 new_path = _split_path(new_path)
223 assert new_path[:len(self._temp_dir)] == self._temp_dir 240 assert new_path[:len(self._temp_dir)] == self._temp_dir
224 temp_dir = self['tmp_base'].join(*new_path[len(self._temp_dir):]) 241 temp_dir = self['tmp_base'].join(*new_path[len(self._temp_dir):])
225 else: 242 else:
(...skipping 73 matching lines...) Expand 10 before | Expand all | Expand 10 after
299 if name in self.OK_ATTRS: 316 if name in self.OK_ATTRS:
300 return getattr(self._path_mod, name) 317 return getattr(self._path_mod, name)
301 if name in self.FILTER_METHODS: 318 if name in self.FILTER_METHODS:
302 return string_filter(getattr(self._path_mod, name)) 319 return string_filter(getattr(self._path_mod, name))
303 raise AttributeError("'%s' object has no attribute '%s'" % 320 raise AttributeError("'%s' object has no attribute '%s'" %
304 (self._path_mod, name)) # pragma: no cover 321 (self._path_mod, name)) # pragma: no cover
305 322
306 def __dir__(self): # pragma: no cover 323 def __dir__(self): # pragma: no cover
307 # Used for helping out show_me_the_modules.py 324 # Used for helping out show_me_the_modules.py
308 return self.__dict__.keys() + list(self.OK_ATTRS + self.FILTER_METHODS) 325 return self.__dict__.keys() + list(self.OK_ATTRS + self.FILTER_METHODS)
OLDNEW
« no previous file with comments | « no previous file | recipe_modules/path/config.py » ('j') | recipe_modules/path/config.py » ('J')

Powered by Google App Engine
This is Rietveld 408576698