| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  | # -*- coding: utf-8 -*- | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # 2010-2011 Steven Armstrong (steven-cdist at armstrong.cc) | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This file is part of cdist. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # cdist is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | # it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  | # the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | # (at your option) any later version. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # cdist is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  | # GNU General Public License for more details. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | # along with cdist. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2011-10-06 01:04:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 20:27:07 +02:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  | import collections | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  | import cdist | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  | import cdist.core.property | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 20:27:07 +02:00
										 |  |  | log = logging.getLogger(__name__) | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | DOT_CDIST = '.cdist' | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  | class Object(object): | 
					
						
							|  |  |  |     """Represents a cdist object.
 | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  |     All interaction with objects in cdist should be done through this class. | 
					
						
							|  |  |  |     Directly accessing an object through the file system from python code is  | 
					
						
							|  |  |  |     a bug. | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def base_dir(): | 
					
						
							|  |  |  |         """Return the absolute path to the top level directory where objects
 | 
					
						
							|  |  |  |         are defined. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Requires the environment variable '__cdist_out_dir' to be set. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2011-10-06 12:57:38 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             base_dir = os.path.join( | 
					
						
							|  |  |  |                 os.environ['__cdist_out_dir'], | 
					
						
							|  |  |  |                 'object' | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         except KeyError as e: | 
					
						
							|  |  |  |             raise cdist.MissingEnvironmentVariableError(e.args[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |         # FIXME: should directory be created elsewhere? | 
					
						
							|  |  |  |         if not os.path.isdir(base_dir): | 
					
						
							|  |  |  |             os.mkdir(base_dir) | 
					
						
							|  |  |  |         return base_dir | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def list_objects(cls): | 
					
						
							|  |  |  |         """Return a list of object instances""" | 
					
						
							|  |  |  |         for object_name in cls.list_object_names(): | 
					
						
							|  |  |  |             type_name = object_name.split(os.sep)[0] | 
					
						
							|  |  |  |             object_id = os.sep.join(object_name.split(os.sep)[1:]) | 
					
						
							|  |  |  |             yield cls(Type(type_name), object_id=object_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def list_type_names(cls): | 
					
						
							|  |  |  |         """Return a list of type names""" | 
					
						
							|  |  |  |         return os.listdir(cls.base_dir()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def list_object_names(cls): | 
					
						
							|  |  |  |         """Return a list of object names""" | 
					
						
							|  |  |  |         for path, dirs, files in os.walk(cls.base_dir()): | 
					
						
							|  |  |  |             # FIXME: use constant instead of string | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  |             if DOT_CDIST in dirs: | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |                 yield os.path.relpath(path, cls.base_dir()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  |     def __init__(self, type, object_id=None, parameters=None, requirements=None): | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |         self.type = type # instance of Type | 
					
						
							|  |  |  |         self.object_id = object_id | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  |         self.name = os.path.join(self.type.name, self.object_id) | 
					
						
							|  |  |  |         self.parameters = parameters or {} | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |         self.requirements = requirements or [] | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.__parameters = None | 
					
						
							|  |  |  |         self.__requirements = None | 
					
						
							| 
									
										
										
										
											2011-10-06 19:40:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Whether this object was prepared/ran | 
					
						
							|  |  |  |         self.prepared = False | 
					
						
							|  |  |  |         self.ran = False | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |          | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  |         return '<Object %s>' % self.name | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def path(self): | 
					
						
							|  |  |  |         return os.path.join( | 
					
						
							|  |  |  |             self.base_dir(), | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  |             self.name, | 
					
						
							|  |  |  |             DOT_CDIST | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ### requirements | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def requirements(self): | 
					
						
							|  |  |  |         if not self.__requirements: | 
					
						
							|  |  |  |             self.__requirements = cdist.core.property.FileList(os.path.join(self.path, "require")) | 
					
						
							|  |  |  |         return self.__requirements | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @requirements.setter | 
					
						
							|  |  |  |     def requirements(self, value): | 
					
						
							|  |  |  |         if isinstance(value, cdist.core.property.FileList): | 
					
						
							|  |  |  |             self.__requirements = value | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.__requirements = cdist.core.property.FileList(os.path.join(self.path, "require"), value) | 
					
						
							|  |  |  |     ### /requirements | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ### parameters | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parameters(self): | 
					
						
							|  |  |  |         if not self.__parameters: | 
					
						
							|  |  |  |             self.__parameters = cdist.core.property.DirectoryDict(os.path.join(self.path, "parameter")) | 
					
						
							|  |  |  |         return self.__parameters | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @parameters.setter | 
					
						
							|  |  |  |     def parameters(self, value): | 
					
						
							|  |  |  |         if isinstance(value, cdist.core.property.DirectoryDict): | 
					
						
							|  |  |  |             self.__parameters = value | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.__parameters = cdist.core.property.DirectoryDict(os.path.join(self.path, "parameter"), value) | 
					
						
							|  |  |  |     ### /parameters | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ### changed | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def changed(self): | 
					
						
							|  |  |  |         """Check whether the object has been changed.""" | 
					
						
							|  |  |  |         return os.path.isfile(os.path.join(self.path, "changed")) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @changed.setter | 
					
						
							|  |  |  |     def changed(self, value): | 
					
						
							|  |  |  |         """Change the objects changed status.""" | 
					
						
							|  |  |  |         path = os.path.join(self.path, "changed") | 
					
						
							|  |  |  |         if value: | 
					
						
							|  |  |  |             open(path, "w").close() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 os.remove(path) | 
					
						
							|  |  |  |             except EnvironmentError: | 
					
						
							|  |  |  |                 # ignore | 
					
						
							|  |  |  |                 pass | 
					
						
							| 
									
										
										
										
											2011-10-06 17:15:45 +02:00
										 |  |  |     ### /changed | 
					
						
							| 
									
										
										
										
											2011-10-06 00:58:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # FIXME: implement other properties/methods |