oo_filters.py 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  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 match
  45. _ALL_ of filters.
  46. Ex: data = [ {'a':1, 'b':5, 'z': 'z'}, # True, return
  47. {'a':2, 'z': 'z'}, # True, return
  48. {'a':3, 'z': 'z'}, # True, return
  49. {'a':4, 'z': 'b'}, # FAILED, obj['z'] != obj['z']
  50. ]
  51. attribute = 'a'
  52. filters = {'z': 'z'}
  53. returns [1, 2, 3]
  54. '''
  55. if not issubclass(type(data), list):
  56. raise errors.AnsibleFilterError("|failed expects to filter on a List")
  57. if not attribute:
  58. raise errors.AnsibleFilterError("|failed expects attribute to be set")
  59. if filters is not None:
  60. if not issubclass(type(filters), dict):
  61. raise errors.AnsibleFilterError("|fialed expects filter to be a"
  62. " dict")
  63. retval = [FilterModule.get_attr(d, attribute) for d in data if (
  64. all([d[key] == filters[key] for key in filters]))]
  65. else:
  66. retval = [FilterModule.get_attr(d, attribute) for d in data]
  67. return retval
  68. @staticmethod
  69. def oo_select_keys(data, keys):
  70. ''' This returns a list, which contains the value portions for the keys
  71. Ex: data = { 'a':1, 'b':2, 'c':3 }
  72. keys = ['a', 'c']
  73. returns [1, 3]
  74. '''
  75. if not issubclass(type(data), dict):
  76. raise errors.AnsibleFilterError("|failed expects to filter on a dict")
  77. if not issubclass(type(keys), list):
  78. raise errors.AnsibleFilterError("|failed expects first param is a list")
  79. # Gather up the values for the list of keys passed in
  80. retval = [data[key] for key in keys]
  81. return retval
  82. @staticmethod
  83. def oo_prepend_strings_in_list(data, prepend):
  84. ''' This takes a list of strings and prepends a string to each item in the
  85. list
  86. Ex: data = ['cart', 'tree']
  87. prepend = 'apple-'
  88. returns ['apple-cart', 'apple-tree']
  89. '''
  90. if not issubclass(type(data), list):
  91. raise errors.AnsibleFilterError("|failed expects first param is a list")
  92. if not all(isinstance(x, basestring) for x in data):
  93. raise errors.AnsibleFilterError("|failed expects first param is a list"
  94. " of strings")
  95. retval = [prepend + s for s in data]
  96. return retval
  97. @staticmethod
  98. def oo_combine_key_value(data, joiner='='):
  99. '''Take a list of dict in the form of { 'key': 'value'} and
  100. arrange them as a list of strings ['key=value']
  101. '''
  102. if not issubclass(type(data), list):
  103. raise errors.AnsibleFilterError("|failed expects first param is a list")
  104. rval = []
  105. for item in data:
  106. rval.append("%s%s%s" % (item['key'], joiner, item['value']))
  107. return rval
  108. @staticmethod
  109. def oo_ami_selector(data, image_name):
  110. ''' This takes a list of amis and an image name and attempts to return
  111. the latest ami.
  112. '''
  113. if not issubclass(type(data), list):
  114. raise errors.AnsibleFilterError("|failed expects first param is a list")
  115. if not data:
  116. return None
  117. else:
  118. if image_name is None or not image_name.endswith('_*'):
  119. ami = sorted(data, key=itemgetter('name'), reverse=True)[0]
  120. return ami['ami_id']
  121. else:
  122. ami_info = [(ami, ami['name'].split('_')[-1]) for ami in data]
  123. ami = sorted(ami_info, key=itemgetter(1), reverse=True)[0][0]
  124. return ami['ami_id']
  125. @staticmethod
  126. def oo_ec2_volume_definition(data, host_type, docker_ephemeral=False):
  127. ''' This takes a dictionary of volume definitions and returns a valid ec2
  128. volume definition based on the host_type and the values in the
  129. dictionary.
  130. The dictionary should look similar to this:
  131. { 'master':
  132. { 'root':
  133. { 'volume_size': 10, 'device_type': 'gp2',
  134. 'iops': 500
  135. }
  136. },
  137. 'node':
  138. { 'root':
  139. { 'volume_size': 10, 'device_type': 'io1',
  140. 'iops': 1000
  141. },
  142. 'docker':
  143. { 'volume_size': 40, 'device_type': 'gp2',
  144. 'iops': 500, 'ephemeral': 'true'
  145. }
  146. }
  147. }
  148. '''
  149. if not issubclass(type(data), dict):
  150. raise errors.AnsibleFilterError("|failed expects first param is a dict")
  151. if host_type not in ['master', 'node']:
  152. raise errors.AnsibleFilterError("|failed expects either master or node"
  153. " host type")
  154. root_vol = data[host_type]['root']
  155. root_vol['device_name'] = '/dev/sda1'
  156. root_vol['delete_on_termination'] = True
  157. if root_vol['device_type'] != 'io1':
  158. root_vol.pop('iops', None)
  159. if host_type == 'node':
  160. docker_vol = data[host_type]['docker']
  161. docker_vol['device_name'] = '/dev/xvdb'
  162. docker_vol['delete_on_termination'] = True
  163. if docker_vol['device_type'] != 'io1':
  164. docker_vol.pop('iops', None)
  165. if docker_ephemeral:
  166. docker_vol.pop('device_type', None)
  167. docker_vol.pop('delete_on_termination', None)
  168. docker_vol['ephemeral'] = 'ephemeral0'
  169. return [root_vol, docker_vol]
  170. return [root_vol]
  171. @staticmethod
  172. def oo_split(string, separator=','):
  173. ''' This splits the input string into a list
  174. '''
  175. return string.split(separator)
  176. @staticmethod
  177. def oo_filter_list(data, filter_attr=None):
  178. ''' This returns a list, which contains all items where filter_attr
  179. evaluates to true
  180. Ex: data = [ { a: 1, b: True },
  181. { a: 3, b: False },
  182. { a: 5, b: True } ]
  183. filter_attr = 'b'
  184. returns [ { a: 1, b: True },
  185. { a: 5, b: True } ]
  186. '''
  187. if not issubclass(type(data), list):
  188. raise errors.AnsibleFilterError("|failed expects to filter on a list")
  189. if not issubclass(type(filter_attr), str):
  190. raise errors.AnsibleFilterError("|failed expects filter_attr is a str")
  191. # Gather up the values for the list of keys passed in
  192. return [x for x in data if x[filter_attr]]
  193. def filters(self):
  194. ''' returns a mapping of filters to methods '''
  195. return {
  196. "oo_select_keys": self.oo_select_keys,
  197. "oo_collect": self.oo_collect,
  198. "oo_flatten": self.oo_flatten,
  199. "oo_pdb": self.oo_pdb,
  200. "oo_prepend_strings_in_list": self.oo_prepend_strings_in_list,
  201. "oo_ami_selector": self.oo_ami_selector,
  202. "oo_ec2_volume_definition": self.oo_ec2_volume_definition,
  203. "oo_combine_key_value": self.oo_combine_key_value,
  204. "oo_split": self.oo_split,
  205. "oo_filter_list": self.oo_filter_list
  206. }