oo_filters.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. # vim: expandtab:tabstop=4:shiftwidth=4
  4. '''
  5. Custom filters for use in openshift-ansible
  6. '''
  7. from ansible import errors
  8. from operator import itemgetter
  9. import pdb
  10. class FilterModule(object):
  11. ''' Custom ansible filters '''
  12. @staticmethod
  13. def oo_pdb(arg):
  14. ''' This pops you into a pdb instance where arg is the data passed in
  15. from the filter.
  16. Ex: "{{ hostvars | oo_pdb }}"
  17. '''
  18. pdb.set_trace()
  19. return arg
  20. @staticmethod
  21. def get_attr(data, attribute=None):
  22. ''' This looks up dictionary attributes of the form a.b.c and returns
  23. the value.
  24. Ex: data = {'a': {'b': {'c': 5}}}
  25. attribute = "a.b.c"
  26. returns 5
  27. '''
  28. if not attribute:
  29. raise errors.AnsibleFilterError("|failed expects attribute to be set")
  30. ptr = data
  31. for attr in attribute.split('.'):
  32. ptr = ptr[attr]
  33. return ptr
  34. @staticmethod
  35. def oo_flatten(data):
  36. ''' This filter plugin will flatten a list of lists
  37. '''
  38. if not issubclass(type(data), list):
  39. raise errors.AnsibleFilterError("|failed expects to flatten a List")
  40. return [item for sublist in data for item in sublist]
  41. @staticmethod
  42. def oo_collect(data, attribute=None, filters=None):
  43. ''' This takes a list of dict and collects all attributes specified into a
  44. list. If filter is specified then we will include all items that
  45. match _ALL_ of filters. If a dict entry is missing the key in a
  46. filter it will be excluded from the match.
  47. Ex: data = [ {'a':1, 'b':5, 'z': 'z'}, # True, return
  48. {'a':2, 'z': 'z'}, # True, return
  49. {'a':3, 'z': 'z'}, # True, return
  50. {'a':4, 'z': 'b'}, # FAILED, obj['z'] != obj['z']
  51. ]
  52. attribute = 'a'
  53. filters = {'z': 'z'}
  54. returns [1, 2, 3]
  55. '''
  56. if not issubclass(type(data), list):
  57. raise errors.AnsibleFilterError("|failed expects to filter on a List")
  58. if not attribute:
  59. raise errors.AnsibleFilterError("|failed expects attribute to be set")
  60. if filters is not None:
  61. if not issubclass(type(filters), dict):
  62. raise errors.AnsibleFilterError("|fialed expects filter to be a"
  63. " dict")
  64. retval = [FilterModule.get_attr(d, attribute) for d in data if (
  65. all([d.get(key, None) == filters[key] for key in filters]))]
  66. else:
  67. retval = [FilterModule.get_attr(d, attribute) for d in data]
  68. return retval
  69. @staticmethod
  70. def oo_select_keys(data, keys):
  71. ''' This returns a list, which contains the value portions for the keys
  72. Ex: data = { 'a':1, 'b':2, 'c':3 }
  73. keys = ['a', 'c']
  74. returns [1, 3]
  75. '''
  76. if not issubclass(type(data), dict):
  77. raise errors.AnsibleFilterError("|failed expects to filter on a dict")
  78. if not issubclass(type(keys), list):
  79. raise errors.AnsibleFilterError("|failed expects first param is a list")
  80. # Gather up the values for the list of keys passed in
  81. retval = [data[key] for key in keys]
  82. return retval
  83. @staticmethod
  84. def oo_prepend_strings_in_list(data, prepend):
  85. ''' This takes a list of strings and prepends a string to each item in the
  86. list
  87. Ex: data = ['cart', 'tree']
  88. prepend = 'apple-'
  89. returns ['apple-cart', 'apple-tree']
  90. '''
  91. if not issubclass(type(data), list):
  92. raise errors.AnsibleFilterError("|failed expects first param is a list")
  93. if not all(isinstance(x, basestring) for x in data):
  94. raise errors.AnsibleFilterError("|failed expects first param is a list"
  95. " of strings")
  96. retval = [prepend + s for s in data]
  97. return retval
  98. @staticmethod
  99. def oo_combine_key_value(data, joiner='='):
  100. '''Take a list of dict in the form of { 'key': 'value'} and
  101. arrange them as a list of strings ['key=value']
  102. '''
  103. if not issubclass(type(data), list):
  104. raise errors.AnsibleFilterError("|failed expects first param is a list")
  105. rval = []
  106. for item in data:
  107. rval.append("%s%s%s" % (item['key'], joiner, item['value']))
  108. return rval
  109. @staticmethod
  110. def oo_ami_selector(data, image_name):
  111. ''' This takes a list of amis and an image name and attempts to return
  112. the latest ami.
  113. '''
  114. if not issubclass(type(data), list):
  115. raise errors.AnsibleFilterError("|failed expects first param is a list")
  116. if not data:
  117. return None
  118. else:
  119. if image_name is None or not image_name.endswith('_*'):
  120. ami = sorted(data, key=itemgetter('name'), reverse=True)[0]
  121. return ami['ami_id']
  122. else:
  123. ami_info = [(ami, ami['name'].split('_')[-1]) for ami in data]
  124. ami = sorted(ami_info, key=itemgetter(1), reverse=True)[0][0]
  125. return ami['ami_id']
  126. @staticmethod
  127. def oo_ec2_volume_definition(data, host_type, docker_ephemeral=False):
  128. ''' This takes a dictionary of volume definitions and returns a valid ec2
  129. volume definition based on the host_type and the values in the
  130. dictionary.
  131. The dictionary should look similar to this:
  132. { 'master':
  133. { 'root':
  134. { 'volume_size': 10, 'device_type': 'gp2',
  135. 'iops': 500
  136. }
  137. },
  138. 'node':
  139. { 'root':
  140. { 'volume_size': 10, 'device_type': 'io1',
  141. 'iops': 1000
  142. },
  143. 'docker':
  144. { 'volume_size': 40, 'device_type': 'gp2',
  145. 'iops': 500, 'ephemeral': 'true'
  146. }
  147. }
  148. }
  149. '''
  150. if not issubclass(type(data), dict):
  151. raise errors.AnsibleFilterError("|failed expects first param is a dict")
  152. if host_type not in ['master', 'node']:
  153. raise errors.AnsibleFilterError("|failed expects either master or node"
  154. " host type")
  155. root_vol = data[host_type]['root']
  156. root_vol['device_name'] = '/dev/sda1'
  157. root_vol['delete_on_termination'] = True
  158. if root_vol['device_type'] != 'io1':
  159. root_vol.pop('iops', None)
  160. if host_type == 'node':
  161. docker_vol = data[host_type]['docker']
  162. docker_vol['device_name'] = '/dev/xvdb'
  163. docker_vol['delete_on_termination'] = True
  164. if docker_vol['device_type'] != 'io1':
  165. docker_vol.pop('iops', None)
  166. if docker_ephemeral:
  167. docker_vol.pop('device_type', None)
  168. docker_vol.pop('delete_on_termination', None)
  169. docker_vol['ephemeral'] = 'ephemeral0'
  170. return [root_vol, docker_vol]
  171. return [root_vol]
  172. @staticmethod
  173. def oo_split(string, separator=','):
  174. ''' This splits the input string into a list
  175. '''
  176. return string.split(separator)
  177. @staticmethod
  178. def oo_filter_list(data, filter_attr=None):
  179. ''' This returns a list, which contains all items where filter_attr
  180. evaluates to true
  181. Ex: data = [ { a: 1, b: True },
  182. { a: 3, b: False },
  183. { a: 5, b: True } ]
  184. filter_attr = 'b'
  185. returns [ { a: 1, b: True },
  186. { a: 5, b: True } ]
  187. '''
  188. if not issubclass(type(data), list):
  189. raise errors.AnsibleFilterError("|failed expects to filter on a list")
  190. if not issubclass(type(filter_attr), str):
  191. raise errors.AnsibleFilterError("|failed expects filter_attr is a str")
  192. # Gather up the values for the list of keys passed in
  193. return [x for x in data if x[filter_attr]]
  194. def filters(self):
  195. ''' returns a mapping of filters to methods '''
  196. return {
  197. "oo_select_keys": self.oo_select_keys,
  198. "oo_collect": self.oo_collect,
  199. "oo_flatten": self.oo_flatten,
  200. "oo_pdb": self.oo_pdb,
  201. "oo_prepend_strings_in_list": self.oo_prepend_strings_in_list,
  202. "oo_ami_selector": self.oo_ami_selector,
  203. "oo_ec2_volume_definition": self.oo_ec2_volume_definition,
  204. "oo_combine_key_value": self.oo_combine_key_value,
  205. "oo_split": self.oo_split,
  206. "oo_filter_list": self.oo_filter_list
  207. }