yedit_test.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. #!/usr/bin/env python2
  2. '''
  3. Unit tests for yedit
  4. '''
  5. import unittest
  6. import os
  7. import yaml
  8. class YeditException(Exception):
  9. ''' Exception class for Yedit '''
  10. pass
  11. class Yedit(object):
  12. ''' Class to modify yaml files '''
  13. def __init__(self, filename):
  14. self.filename = filename
  15. self.__yaml_dict = None
  16. self.get()
  17. @property
  18. def yaml_dict(self):
  19. ''' get property for yaml_dict '''
  20. return self.__yaml_dict
  21. @yaml_dict.setter
  22. def yaml_dict(self, value):
  23. ''' setter method for yaml_dict '''
  24. self.__yaml_dict = value
  25. @staticmethod
  26. def remove_entry(data, keys):
  27. ''' remove an item from a dictionary with key notation a.b.c
  28. d = {'a': {'b': 'c'}}}
  29. keys = a.b
  30. item = c
  31. '''
  32. if "." in keys:
  33. key, rest = keys.split(".", 1)
  34. if key in data.keys():
  35. Yedit.remove_entry(data[key], rest)
  36. else:
  37. del data[keys]
  38. @staticmethod
  39. def add_entry(data, keys, item):
  40. ''' Add an item to a dictionary with key notation a.b.c
  41. d = {'a': {'b': 'c'}}}
  42. keys = a.b
  43. item = c
  44. '''
  45. if "." in keys:
  46. key, rest = keys.split(".", 1)
  47. if key not in data:
  48. data[key] = {}
  49. if not isinstance(data, dict):
  50. raise YeditException('Invalid add_entry called on data [%s].' % data)
  51. else:
  52. Yedit.add_entry(data[key], rest, item)
  53. else:
  54. data[keys] = item
  55. @staticmethod
  56. def get_entry(data, keys):
  57. ''' Get an item from a dictionary with key notation a.b.c
  58. d = {'a': {'b': 'c'}}}
  59. keys = a.b
  60. return c
  61. '''
  62. if keys and "." in keys:
  63. key, rest = keys.split(".", 1)
  64. if not isinstance(data[key], dict):
  65. raise YeditException('Invalid get_entry called on a [%s] of type [%s].' % (data, type(data)))
  66. else:
  67. return Yedit.get_entry(data[key], rest)
  68. else:
  69. return data.get(keys, None)
  70. def write(self):
  71. ''' write to file '''
  72. with open(self.filename, 'w') as yfd:
  73. yfd.write(yaml.dump(self.yaml_dict, default_flow_style=False))
  74. def read(self):
  75. ''' write to file '''
  76. # check if it exists
  77. if not self.exists():
  78. return None
  79. contents = None
  80. with open(self.filename) as yfd:
  81. contents = yfd.read()
  82. return contents
  83. def exists(self):
  84. ''' return whether file exists '''
  85. if os.path.exists(self.filename):
  86. return True
  87. return False
  88. def get(self):
  89. ''' return yaml file '''
  90. contents = self.read()
  91. if not contents:
  92. return None
  93. # check if it is yaml
  94. try:
  95. self.yaml_dict = yaml.load(contents)
  96. except yaml.YAMLError as _:
  97. # Error loading yaml
  98. return None
  99. return self.yaml_dict
  100. def delete(self, key):
  101. ''' put key, value into a yaml file '''
  102. try:
  103. entry = Yedit.get_entry(self.yaml_dict, key)
  104. except KeyError as _:
  105. entry = None
  106. if not entry:
  107. return (False, self.yaml_dict)
  108. Yedit.remove_entry(self.yaml_dict, key)
  109. self.write()
  110. return (True, self.get())
  111. def put(self, key, value):
  112. ''' put key, value into a yaml file '''
  113. try:
  114. entry = Yedit.get_entry(self.yaml_dict, key)
  115. except KeyError as _:
  116. entry = None
  117. if entry == value:
  118. return (False, self.yaml_dict)
  119. Yedit.add_entry(self.yaml_dict, key, value)
  120. self.write()
  121. return (True, self.get())
  122. def create(self, key, value):
  123. ''' create the file '''
  124. if not self.exists():
  125. self.yaml_dict = {key: value}
  126. self.write()
  127. return (True, self.get())
  128. return (False, self.get())
  129. # Removing invalid variable names for tests so that I can
  130. # keep them brief
  131. # pylint: disable=invalid-name
  132. class YeditTest(unittest.TestCase):
  133. '''
  134. Test class for yedit
  135. '''
  136. data = {'a': 'a',
  137. 'b': {'c': {'d': ['e', 'f', 'g']}},
  138. }
  139. filename = 'yedit_test.yml'
  140. def setUp(self):
  141. ''' setup method will create a file and set to known configuration '''
  142. yed = Yedit(YeditTest.filename)
  143. yed.yaml_dict = YeditTest.data
  144. yed.write()
  145. def test_get(self):
  146. ''' Testing a get '''
  147. yed = Yedit('yedit_test.yml')
  148. self.assertEqual(yed.yaml_dict, self.data)
  149. def test_write(self):
  150. ''' Testing a simple write '''
  151. yed = Yedit('yedit_test.yml')
  152. yed.put('key1', 1)
  153. yed.write()
  154. yed.get()
  155. self.assertTrue(yed.yaml_dict.has_key('key1'))
  156. self.assertEqual(yed.yaml_dict['key1'], 1)
  157. def test_write_x_y_z(self):
  158. '''Testing a write of multilayer key'''
  159. yed = Yedit('yedit_test.yml')
  160. yed.put('x.y.z', 'modified')
  161. yed.write()
  162. self.assertEqual(Yedit.get_entry(yed.get(), 'x.y.z'), 'modified')
  163. def test_delete_a(self):
  164. '''Testing a simple delete '''
  165. yed = Yedit('yedit_test.yml')
  166. yed.delete('a')
  167. yed.write()
  168. yed.get()
  169. self.assertTrue(not yed.yaml_dict.has_key('a'))
  170. def test_delete_b_c(self):
  171. '''Testing delete of layered key '''
  172. yed = Yedit('yedit_test.yml')
  173. yed.delete('b.c')
  174. yed.write()
  175. yed.get()
  176. self.assertTrue(yed.yaml_dict.has_key('b'))
  177. self.assertFalse(yed.yaml_dict['b'].has_key('c'))
  178. def test_create(self):
  179. '''Testing a create '''
  180. os.unlink(YeditTest.filename)
  181. yed = Yedit('yedit_test.yml')
  182. yed.create('foo', 'bar')
  183. yed.write()
  184. yed.get()
  185. self.assertTrue(yed.yaml_dict.has_key('foo'))
  186. self.assertTrue(yed.yaml_dict['foo'], 'bar')
  187. def tearDown(self):
  188. '''TearDown method'''
  189. os.unlink(YeditTest.filename)
  190. if __name__ == "__main__":
  191. unittest.main()