oc_obj.py 59 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744
  1. #!/usr/bin/env python
  2. # pylint: disable=missing-docstring
  3. # flake8: noqa: T001
  4. # ___ ___ _ _ ___ ___ _ _____ ___ ___
  5. # / __| __| \| | __| _ \ /_\_ _| __| \
  6. # | (_ | _|| .` | _|| / / _ \| | | _|| |) |
  7. # \___|___|_|\_|___|_|_\/_/_\_\_|_|___|___/_ _____
  8. # | \ / _ \ | \| |/ _ \_ _| | __| \_ _|_ _|
  9. # | |) | (_) | | .` | (_) || | | _|| |) | | | |
  10. # |___/ \___/ |_|\_|\___/ |_| |___|___/___| |_|
  11. #
  12. # Copyright 2016 Red Hat, Inc. and/or its affiliates
  13. # and other contributors as indicated by the @author tags.
  14. #
  15. # Licensed under the Apache License, Version 2.0 (the "License");
  16. # you may not use this file except in compliance with the License.
  17. # You may obtain a copy of the License at
  18. #
  19. # http://www.apache.org/licenses/LICENSE-2.0
  20. #
  21. # Unless required by applicable law or agreed to in writing, software
  22. # distributed under the License is distributed on an "AS IS" BASIS,
  23. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  24. # See the License for the specific language governing permissions and
  25. # limitations under the License.
  26. #
  27. # -*- -*- -*- Begin included fragment: lib/import.py -*- -*- -*-
  28. '''
  29. OpenShiftCLI class that wraps the oc commands in a subprocess
  30. '''
  31. # pylint: disable=too-many-lines
  32. from __future__ import print_function
  33. import atexit
  34. import copy
  35. import fcntl
  36. import json
  37. import time
  38. import os
  39. import re
  40. import shutil
  41. import subprocess
  42. import tempfile
  43. # pylint: disable=import-error
  44. try:
  45. import ruamel.yaml as yaml
  46. except ImportError:
  47. import yaml
  48. from ansible.module_utils.basic import AnsibleModule
  49. # -*- -*- -*- End included fragment: lib/import.py -*- -*- -*-
  50. # -*- -*- -*- Begin included fragment: doc/obj -*- -*- -*-
  51. DOCUMENTATION = '''
  52. ---
  53. module: oc_obj
  54. short_description: Generic interface to openshift objects
  55. description:
  56. - Manage openshift objects programmatically.
  57. options:
  58. state:
  59. description:
  60. - Currently present is only supported state.
  61. required: true
  62. default: present
  63. choices: ["present", "absent", "list"]
  64. aliases: []
  65. kubeconfig:
  66. description:
  67. - The path for the kubeconfig file to use for authentication
  68. required: false
  69. default: /etc/origin/master/admin.kubeconfig
  70. aliases: []
  71. debug:
  72. description:
  73. - Turn on debug output.
  74. required: false
  75. default: False
  76. aliases: []
  77. name:
  78. description:
  79. - Name of the object that is being queried.
  80. required: false
  81. default: None
  82. aliases: []
  83. namespace:
  84. description:
  85. - The namespace where the object lives.
  86. required: false
  87. default: str
  88. aliases: []
  89. all_namespaces:
  90. description:
  91. - Search in all namespaces for the object.
  92. required: false
  93. default: false
  94. aliases: []
  95. kind:
  96. description:
  97. - The kind attribute of the object. e.g. dc, bc, svc, route. May be a comma-separated list, e.g. "dc,po,svc".
  98. required: True
  99. default: None
  100. aliases: []
  101. files:
  102. description:
  103. - A list of files provided for object
  104. required: false
  105. default: None
  106. aliases: []
  107. delete_after:
  108. description:
  109. - Whether or not to delete the files after processing them.
  110. required: false
  111. default: false
  112. aliases: []
  113. content:
  114. description:
  115. - Content of the object being managed.
  116. required: false
  117. default: None
  118. aliases: []
  119. force:
  120. description:
  121. - Whether or not to force the operation
  122. required: false
  123. default: None
  124. aliases: []
  125. selector:
  126. description:
  127. - Selector that gets added to the query.
  128. required: false
  129. default: None
  130. aliases: []
  131. field_selector:
  132. description:
  133. - Field selector that gets added to the query.
  134. required: false
  135. default: None
  136. aliases: []
  137. author:
  138. - "Kenny Woodson <kwoodson@redhat.com>"
  139. extends_documentation_fragment: []
  140. '''
  141. EXAMPLES = '''
  142. oc_obj:
  143. kind: dc
  144. name: router
  145. namespace: default
  146. register: router_output
  147. '''
  148. # -*- -*- -*- End included fragment: doc/obj -*- -*- -*-
  149. # -*- -*- -*- Begin included fragment: ../../lib_utils/src/class/yedit.py -*- -*- -*-
  150. class YeditException(Exception): # pragma: no cover
  151. ''' Exception class for Yedit '''
  152. pass
  153. # pylint: disable=too-many-public-methods
  154. class Yedit(object): # pragma: no cover
  155. ''' Class to modify yaml files '''
  156. re_valid_key = r"(((\[-?\d+\])|([0-9a-zA-Z%s/_-]+)).?)+$"
  157. re_key = r"(?:\[(-?\d+)\])|([0-9a-zA-Z{}/_-]+)"
  158. com_sep = set(['.', '#', '|', ':'])
  159. # pylint: disable=too-many-arguments
  160. def __init__(self,
  161. filename=None,
  162. content=None,
  163. content_type='yaml',
  164. separator='.',
  165. backup=False):
  166. self.content = content
  167. self._separator = separator
  168. self.filename = filename
  169. self.__yaml_dict = content
  170. self.content_type = content_type
  171. self.backup = backup
  172. self.load(content_type=self.content_type)
  173. if self.__yaml_dict is None:
  174. self.__yaml_dict = {}
  175. @property
  176. def separator(self):
  177. ''' getter method for separator '''
  178. return self._separator
  179. @separator.setter
  180. def separator(self, inc_sep):
  181. ''' setter method for separator '''
  182. self._separator = inc_sep
  183. @property
  184. def yaml_dict(self):
  185. ''' getter method for yaml_dict '''
  186. return self.__yaml_dict
  187. @yaml_dict.setter
  188. def yaml_dict(self, value):
  189. ''' setter method for yaml_dict '''
  190. self.__yaml_dict = value
  191. @staticmethod
  192. def parse_key(key, sep='.'):
  193. '''parse the key allowing the appropriate separator'''
  194. common_separators = list(Yedit.com_sep - set([sep]))
  195. return re.findall(Yedit.re_key.format(''.join(common_separators)), key)
  196. @staticmethod
  197. def valid_key(key, sep='.'):
  198. '''validate the incoming key'''
  199. common_separators = list(Yedit.com_sep - set([sep]))
  200. if not re.match(Yedit.re_valid_key.format(''.join(common_separators)), key):
  201. return False
  202. return True
  203. # pylint: disable=too-many-return-statements,too-many-branches
  204. @staticmethod
  205. def remove_entry(data, key, index=None, value=None, sep='.'):
  206. ''' remove data at location key '''
  207. if key == '' and isinstance(data, dict):
  208. if value is not None:
  209. data.pop(value)
  210. elif index is not None:
  211. raise YeditException("remove_entry for a dictionary does not have an index {}".format(index))
  212. else:
  213. data.clear()
  214. return True
  215. elif key == '' and isinstance(data, list):
  216. ind = None
  217. if value is not None:
  218. try:
  219. ind = data.index(value)
  220. except ValueError:
  221. return False
  222. elif index is not None:
  223. ind = index
  224. else:
  225. del data[:]
  226. if ind is not None:
  227. data.pop(ind)
  228. return True
  229. if not (key and Yedit.valid_key(key, sep)) and \
  230. isinstance(data, (list, dict)):
  231. return None
  232. key_indexes = Yedit.parse_key(key, sep)
  233. for arr_ind, dict_key in key_indexes[:-1]:
  234. if dict_key and isinstance(data, dict):
  235. data = data.get(dict_key)
  236. elif (arr_ind and isinstance(data, list) and
  237. int(arr_ind) <= len(data) - 1):
  238. data = data[int(arr_ind)]
  239. else:
  240. return None
  241. # process last index for remove
  242. # expected list entry
  243. if key_indexes[-1][0]:
  244. if isinstance(data, list) and int(key_indexes[-1][0]) <= len(data) - 1: # noqa: E501
  245. del data[int(key_indexes[-1][0])]
  246. return True
  247. # expected dict entry
  248. elif key_indexes[-1][1]:
  249. if isinstance(data, dict):
  250. del data[key_indexes[-1][1]]
  251. return True
  252. @staticmethod
  253. def add_entry(data, key, item=None, sep='.'):
  254. ''' Get an item from a dictionary with key notation a.b.c
  255. d = {'a': {'b': 'c'}}}
  256. key = a#b
  257. return c
  258. '''
  259. if key == '':
  260. pass
  261. elif (not (key and Yedit.valid_key(key, sep)) and
  262. isinstance(data, (list, dict))):
  263. return None
  264. key_indexes = Yedit.parse_key(key, sep)
  265. for arr_ind, dict_key in key_indexes[:-1]:
  266. if dict_key:
  267. if isinstance(data, dict) and dict_key in data and data[dict_key]: # noqa: E501
  268. data = data[dict_key]
  269. continue
  270. elif data and not isinstance(data, dict):
  271. raise YeditException("Unexpected item type found while going through key " +
  272. "path: {} (at key: {})".format(key, dict_key))
  273. data[dict_key] = {}
  274. data = data[dict_key]
  275. elif (arr_ind and isinstance(data, list) and
  276. int(arr_ind) <= len(data) - 1):
  277. data = data[int(arr_ind)]
  278. else:
  279. raise YeditException("Unexpected item type found while going through key path: {}".format(key))
  280. if key == '':
  281. data = item
  282. # process last index for add
  283. # expected list entry
  284. elif key_indexes[-1][0] and isinstance(data, list) and int(key_indexes[-1][0]) <= len(data) - 1: # noqa: E501
  285. data[int(key_indexes[-1][0])] = item
  286. # expected dict entry
  287. elif key_indexes[-1][1] and isinstance(data, dict):
  288. data[key_indexes[-1][1]] = item
  289. # didn't add/update to an existing list, nor add/update key to a dict
  290. # so we must have been provided some syntax like a.b.c[<int>] = "data" for a
  291. # non-existent array
  292. else:
  293. raise YeditException("Error adding to object at path: {}".format(key))
  294. return data
  295. @staticmethod
  296. def get_entry(data, key, sep='.'):
  297. ''' Get an item from a dictionary with key notation a.b.c
  298. d = {'a': {'b': 'c'}}}
  299. key = a.b
  300. return c
  301. '''
  302. if key == '':
  303. pass
  304. elif (not (key and Yedit.valid_key(key, sep)) and
  305. isinstance(data, (list, dict))):
  306. return None
  307. key_indexes = Yedit.parse_key(key, sep)
  308. for arr_ind, dict_key in key_indexes:
  309. if dict_key and isinstance(data, dict):
  310. data = data.get(dict_key)
  311. elif (arr_ind and isinstance(data, list) and
  312. int(arr_ind) <= len(data) - 1):
  313. data = data[int(arr_ind)]
  314. else:
  315. return None
  316. return data
  317. @staticmethod
  318. def _write(filename, contents):
  319. ''' Actually write the file contents to disk. This helps with mocking. '''
  320. tmp_filename = filename + '.yedit'
  321. with open(tmp_filename, 'w') as yfd:
  322. fcntl.flock(yfd, fcntl.LOCK_EX | fcntl.LOCK_NB)
  323. yfd.write(contents)
  324. fcntl.flock(yfd, fcntl.LOCK_UN)
  325. os.rename(tmp_filename, filename)
  326. def write(self):
  327. ''' write to file '''
  328. if not self.filename:
  329. raise YeditException('Please specify a filename.')
  330. if self.backup and self.file_exists():
  331. shutil.copy(self.filename, '{}.{}'.format(self.filename, time.strftime("%Y%m%dT%H%M%S")))
  332. # Try to set format attributes if supported
  333. try:
  334. self.yaml_dict.fa.set_block_style()
  335. except AttributeError:
  336. pass
  337. # Try to use RoundTripDumper if supported.
  338. if self.content_type == 'yaml':
  339. try:
  340. Yedit._write(self.filename, yaml.dump(self.yaml_dict, Dumper=yaml.RoundTripDumper))
  341. except AttributeError:
  342. Yedit._write(self.filename, yaml.safe_dump(self.yaml_dict, default_flow_style=False))
  343. elif self.content_type == 'json':
  344. Yedit._write(self.filename, json.dumps(self.yaml_dict, indent=4, sort_keys=True))
  345. else:
  346. raise YeditException('Unsupported content_type: {}.'.format(self.content_type) +
  347. 'Please specify a content_type of yaml or json.')
  348. return (True, self.yaml_dict)
  349. def read(self):
  350. ''' read from file '''
  351. # check if it exists
  352. if self.filename is None or not self.file_exists():
  353. return None
  354. contents = None
  355. with open(self.filename) as yfd:
  356. contents = yfd.read()
  357. return contents
  358. def file_exists(self):
  359. ''' return whether file exists '''
  360. if os.path.exists(self.filename):
  361. return True
  362. return False
  363. def load(self, content_type='yaml'):
  364. ''' return yaml file '''
  365. contents = self.read()
  366. if not contents and not self.content:
  367. return None
  368. if self.content:
  369. if isinstance(self.content, dict):
  370. self.yaml_dict = self.content
  371. return self.yaml_dict
  372. elif isinstance(self.content, str):
  373. contents = self.content
  374. # check if it is yaml
  375. try:
  376. if content_type == 'yaml' and contents:
  377. # Try to set format attributes if supported
  378. try:
  379. self.yaml_dict.fa.set_block_style()
  380. except AttributeError:
  381. pass
  382. # Try to use RoundTripLoader if supported.
  383. try:
  384. self.yaml_dict = yaml.load(contents, yaml.RoundTripLoader)
  385. except AttributeError:
  386. self.yaml_dict = yaml.safe_load(contents)
  387. # Try to set format attributes if supported
  388. try:
  389. self.yaml_dict.fa.set_block_style()
  390. except AttributeError:
  391. pass
  392. elif content_type == 'json' and contents:
  393. self.yaml_dict = json.loads(contents)
  394. except yaml.YAMLError as err:
  395. # Error loading yaml or json
  396. raise YeditException('Problem with loading yaml file. {}'.format(err))
  397. return self.yaml_dict
  398. def get(self, key):
  399. ''' get a specified key'''
  400. try:
  401. entry = Yedit.get_entry(self.yaml_dict, key, self.separator)
  402. except KeyError:
  403. entry = None
  404. return entry
  405. def pop(self, path, key_or_item):
  406. ''' remove a key, value pair from a dict or an item for a list'''
  407. try:
  408. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  409. except KeyError:
  410. entry = None
  411. if entry is None:
  412. return (False, self.yaml_dict)
  413. if isinstance(entry, dict):
  414. # AUDIT:maybe-no-member makes sense due to fuzzy types
  415. # pylint: disable=maybe-no-member
  416. if key_or_item in entry:
  417. entry.pop(key_or_item)
  418. return (True, self.yaml_dict)
  419. return (False, self.yaml_dict)
  420. elif isinstance(entry, list):
  421. # AUDIT:maybe-no-member makes sense due to fuzzy types
  422. # pylint: disable=maybe-no-member
  423. ind = None
  424. try:
  425. ind = entry.index(key_or_item)
  426. except ValueError:
  427. return (False, self.yaml_dict)
  428. entry.pop(ind)
  429. return (True, self.yaml_dict)
  430. return (False, self.yaml_dict)
  431. def delete(self, path, index=None, value=None):
  432. ''' remove path from a dict'''
  433. try:
  434. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  435. except KeyError:
  436. entry = None
  437. if entry is None:
  438. return (False, self.yaml_dict)
  439. result = Yedit.remove_entry(self.yaml_dict, path, index, value, self.separator)
  440. if not result:
  441. return (False, self.yaml_dict)
  442. return (True, self.yaml_dict)
  443. def exists(self, path, value):
  444. ''' check if value exists at path'''
  445. try:
  446. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  447. except KeyError:
  448. entry = None
  449. if isinstance(entry, list):
  450. if value in entry:
  451. return True
  452. return False
  453. elif isinstance(entry, dict):
  454. if isinstance(value, dict):
  455. rval = False
  456. for key, val in value.items():
  457. if entry[key] != val:
  458. rval = False
  459. break
  460. else:
  461. rval = True
  462. return rval
  463. return value in entry
  464. return entry == value
  465. def append(self, path, value):
  466. '''append value to a list'''
  467. try:
  468. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  469. except KeyError:
  470. entry = None
  471. if entry is None:
  472. self.put(path, [])
  473. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  474. if not isinstance(entry, list):
  475. return (False, self.yaml_dict)
  476. # AUDIT:maybe-no-member makes sense due to loading data from
  477. # a serialized format.
  478. # pylint: disable=maybe-no-member
  479. entry.append(value)
  480. return (True, self.yaml_dict)
  481. # pylint: disable=too-many-arguments
  482. def update(self, path, value, index=None, curr_value=None):
  483. ''' put path, value into a dict '''
  484. try:
  485. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  486. except KeyError:
  487. entry = None
  488. if isinstance(entry, dict):
  489. # AUDIT:maybe-no-member makes sense due to fuzzy types
  490. # pylint: disable=maybe-no-member
  491. if not isinstance(value, dict):
  492. raise YeditException('Cannot replace key, value entry in dict with non-dict type. ' +
  493. 'value=[{}] type=[{}]'.format(value, type(value)))
  494. entry.update(value)
  495. return (True, self.yaml_dict)
  496. elif isinstance(entry, list):
  497. # AUDIT:maybe-no-member makes sense due to fuzzy types
  498. # pylint: disable=maybe-no-member
  499. ind = None
  500. if curr_value:
  501. try:
  502. ind = entry.index(curr_value)
  503. except ValueError:
  504. return (False, self.yaml_dict)
  505. elif index is not None:
  506. ind = index
  507. if ind is not None and entry[ind] != value:
  508. entry[ind] = value
  509. return (True, self.yaml_dict)
  510. # see if it exists in the list
  511. try:
  512. ind = entry.index(value)
  513. except ValueError:
  514. # doesn't exist, append it
  515. entry.append(value)
  516. return (True, self.yaml_dict)
  517. # already exists, return
  518. if ind is not None:
  519. return (False, self.yaml_dict)
  520. return (False, self.yaml_dict)
  521. def put(self, path, value):
  522. ''' put path, value into a dict '''
  523. try:
  524. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  525. except KeyError:
  526. entry = None
  527. if entry == value:
  528. return (False, self.yaml_dict)
  529. # deepcopy didn't work
  530. # Try to use ruamel.yaml and fallback to pyyaml
  531. try:
  532. tmp_copy = yaml.load(yaml.round_trip_dump(self.yaml_dict,
  533. default_flow_style=False),
  534. yaml.RoundTripLoader)
  535. except AttributeError:
  536. tmp_copy = copy.deepcopy(self.yaml_dict)
  537. # set the format attributes if available
  538. try:
  539. tmp_copy.fa.set_block_style()
  540. except AttributeError:
  541. pass
  542. result = Yedit.add_entry(tmp_copy, path, value, self.separator)
  543. if result is None:
  544. return (False, self.yaml_dict)
  545. # When path equals "" it is a special case.
  546. # "" refers to the root of the document
  547. # Only update the root path (entire document) when its a list or dict
  548. if path == '':
  549. if isinstance(result, list) or isinstance(result, dict):
  550. self.yaml_dict = result
  551. return (True, self.yaml_dict)
  552. return (False, self.yaml_dict)
  553. self.yaml_dict = tmp_copy
  554. return (True, self.yaml_dict)
  555. def create(self, path, value):
  556. ''' create a yaml file '''
  557. if not self.file_exists():
  558. # deepcopy didn't work
  559. # Try to use ruamel.yaml and fallback to pyyaml
  560. try:
  561. tmp_copy = yaml.load(yaml.round_trip_dump(self.yaml_dict,
  562. default_flow_style=False),
  563. yaml.RoundTripLoader)
  564. except AttributeError:
  565. tmp_copy = copy.deepcopy(self.yaml_dict)
  566. # set the format attributes if available
  567. try:
  568. tmp_copy.fa.set_block_style()
  569. except AttributeError:
  570. pass
  571. result = Yedit.add_entry(tmp_copy, path, value, self.separator)
  572. if result is not None:
  573. self.yaml_dict = tmp_copy
  574. return (True, self.yaml_dict)
  575. return (False, self.yaml_dict)
  576. @staticmethod
  577. def get_curr_value(invalue, val_type):
  578. '''return the current value'''
  579. if invalue is None:
  580. return None
  581. curr_value = invalue
  582. if val_type == 'yaml':
  583. try:
  584. # AUDIT:maybe-no-member makes sense due to different yaml libraries
  585. # pylint: disable=maybe-no-member
  586. curr_value = yaml.safe_load(invalue, Loader=yaml.RoundTripLoader)
  587. except AttributeError:
  588. curr_value = yaml.safe_load(invalue)
  589. elif val_type == 'json':
  590. curr_value = json.loads(invalue)
  591. return curr_value
  592. @staticmethod
  593. def parse_value(inc_value, vtype=''):
  594. '''determine value type passed'''
  595. true_bools = ['y', 'Y', 'yes', 'Yes', 'YES', 'true', 'True', 'TRUE',
  596. 'on', 'On', 'ON', ]
  597. false_bools = ['n', 'N', 'no', 'No', 'NO', 'false', 'False', 'FALSE',
  598. 'off', 'Off', 'OFF']
  599. # It came in as a string but you didn't specify value_type as string
  600. # we will convert to bool if it matches any of the above cases
  601. if isinstance(inc_value, str) and 'bool' in vtype:
  602. if inc_value not in true_bools and inc_value not in false_bools:
  603. raise YeditException('Not a boolean type. str=[{}] vtype=[{}]'.format(inc_value, vtype))
  604. elif isinstance(inc_value, bool) and 'str' in vtype:
  605. inc_value = str(inc_value)
  606. # There is a special case where '' will turn into None after yaml loading it so skip
  607. if isinstance(inc_value, str) and inc_value == '':
  608. pass
  609. # If vtype is not str then go ahead and attempt to yaml load it.
  610. elif isinstance(inc_value, str) and 'str' not in vtype:
  611. try:
  612. inc_value = yaml.safe_load(inc_value)
  613. except Exception:
  614. raise YeditException('Could not determine type of incoming value. ' +
  615. 'value=[{}] vtype=[{}]'.format(type(inc_value), vtype))
  616. return inc_value
  617. @staticmethod
  618. def process_edits(edits, yamlfile):
  619. '''run through a list of edits and process them one-by-one'''
  620. results = []
  621. for edit in edits:
  622. value = Yedit.parse_value(edit['value'], edit.get('value_type', ''))
  623. if edit.get('action') == 'update':
  624. # pylint: disable=line-too-long
  625. curr_value = Yedit.get_curr_value(
  626. Yedit.parse_value(edit.get('curr_value')),
  627. edit.get('curr_value_format'))
  628. rval = yamlfile.update(edit['key'],
  629. value,
  630. edit.get('index'),
  631. curr_value)
  632. elif edit.get('action') == 'append':
  633. rval = yamlfile.append(edit['key'], value)
  634. else:
  635. rval = yamlfile.put(edit['key'], value)
  636. if rval[0]:
  637. results.append({'key': edit['key'], 'edit': rval[1]})
  638. return {'changed': len(results) > 0, 'results': results}
  639. # pylint: disable=too-many-return-statements,too-many-branches
  640. @staticmethod
  641. def run_ansible(params):
  642. '''perform the idempotent crud operations'''
  643. yamlfile = Yedit(filename=params['src'],
  644. backup=params['backup'],
  645. content_type=params['content_type'],
  646. separator=params['separator'])
  647. state = params['state']
  648. if params['src']:
  649. rval = yamlfile.load()
  650. if yamlfile.yaml_dict is None and state != 'present':
  651. return {'failed': True,
  652. 'msg': 'Error opening file [{}]. Verify that the '.format(params['src']) +
  653. 'file exists, that it is has correct permissions, and is valid yaml.'}
  654. if state == 'list':
  655. if params['content']:
  656. content = Yedit.parse_value(params['content'], params['content_type'])
  657. yamlfile.yaml_dict = content
  658. if params['key']:
  659. rval = yamlfile.get(params['key'])
  660. return {'changed': False, 'result': rval, 'state': state}
  661. elif state == 'absent':
  662. if params['content']:
  663. content = Yedit.parse_value(params['content'], params['content_type'])
  664. yamlfile.yaml_dict = content
  665. if params['update']:
  666. rval = yamlfile.pop(params['key'], params['value'])
  667. else:
  668. rval = yamlfile.delete(params['key'], params['index'], params['value'])
  669. if rval[0] and params['src']:
  670. yamlfile.write()
  671. return {'changed': rval[0], 'result': rval[1], 'state': state}
  672. elif state == 'present':
  673. # check if content is different than what is in the file
  674. if params['content']:
  675. content = Yedit.parse_value(params['content'], params['content_type'])
  676. # We had no edits to make and the contents are the same
  677. if yamlfile.yaml_dict == content and \
  678. params['value'] is None:
  679. return {'changed': False, 'result': yamlfile.yaml_dict, 'state': state}
  680. yamlfile.yaml_dict = content
  681. # If we were passed a key, value then
  682. # we enapsulate it in a list and process it
  683. # Key, Value passed to the module : Converted to Edits list #
  684. edits = []
  685. _edit = {}
  686. if params['value'] is not None:
  687. _edit['value'] = params['value']
  688. _edit['value_type'] = params['value_type']
  689. _edit['key'] = params['key']
  690. if params['update']:
  691. _edit['action'] = 'update'
  692. _edit['curr_value'] = params['curr_value']
  693. _edit['curr_value_format'] = params['curr_value_format']
  694. _edit['index'] = params['index']
  695. elif params['append']:
  696. _edit['action'] = 'append'
  697. edits.append(_edit)
  698. elif params['edits'] is not None:
  699. edits = params['edits']
  700. if edits:
  701. results = Yedit.process_edits(edits, yamlfile)
  702. # if there were changes and a src provided to us we need to write
  703. if results['changed'] and params['src']:
  704. yamlfile.write()
  705. return {'changed': results['changed'], 'result': results['results'], 'state': state}
  706. # no edits to make
  707. if params['src']:
  708. # pylint: disable=redefined-variable-type
  709. rval = yamlfile.write()
  710. return {'changed': rval[0],
  711. 'result': rval[1],
  712. 'state': state}
  713. # We were passed content but no src, key or value, or edits. Return contents in memory
  714. return {'changed': False, 'result': yamlfile.yaml_dict, 'state': state}
  715. return {'failed': True, 'msg': 'Unkown state passed'}
  716. # -*- -*- -*- End included fragment: ../../lib_utils/src/class/yedit.py -*- -*- -*-
  717. # -*- -*- -*- Begin included fragment: lib/base.py -*- -*- -*-
  718. # pylint: disable=too-many-lines
  719. # noqa: E301,E302,E303,T001
  720. class OpenShiftCLIError(Exception):
  721. '''Exception class for openshiftcli'''
  722. pass
  723. ADDITIONAL_PATH_LOOKUPS = ['/usr/local/bin', os.path.expanduser('~/bin')]
  724. def locate_oc_binary():
  725. ''' Find and return oc binary file '''
  726. # https://github.com/openshift/openshift-ansible/issues/3410
  727. # oc can be in /usr/local/bin in some cases, but that may not
  728. # be in $PATH due to ansible/sudo
  729. paths = os.environ.get("PATH", os.defpath).split(os.pathsep) + ADDITIONAL_PATH_LOOKUPS
  730. oc_binary = 'oc'
  731. # Use shutil.which if it is available, otherwise fallback to a naive path search
  732. try:
  733. which_result = shutil.which(oc_binary, path=os.pathsep.join(paths))
  734. if which_result is not None:
  735. oc_binary = which_result
  736. except AttributeError:
  737. for path in paths:
  738. if os.path.exists(os.path.join(path, oc_binary)):
  739. oc_binary = os.path.join(path, oc_binary)
  740. break
  741. return oc_binary
  742. # pylint: disable=too-few-public-methods
  743. class OpenShiftCLI(object):
  744. ''' Class to wrap the command line tools '''
  745. def __init__(self,
  746. namespace,
  747. kubeconfig='/etc/origin/master/admin.kubeconfig',
  748. verbose=False,
  749. all_namespaces=False):
  750. ''' Constructor for OpenshiftCLI '''
  751. self.namespace = namespace
  752. self.verbose = verbose
  753. self.kubeconfig = Utils.create_tmpfile_copy(kubeconfig)
  754. self.all_namespaces = all_namespaces
  755. self.oc_binary = locate_oc_binary()
  756. # Pylint allows only 5 arguments to be passed.
  757. # pylint: disable=too-many-arguments
  758. def _replace_content(self, resource, rname, content, edits=None, force=False, sep='.'):
  759. ''' replace the current object with the content '''
  760. res = self._get(resource, rname)
  761. if not res['results']:
  762. return res
  763. fname = Utils.create_tmpfile(rname + '-')
  764. yed = Yedit(fname, res['results'][0], separator=sep)
  765. updated = False
  766. if content is not None:
  767. changes = []
  768. for key, value in content.items():
  769. changes.append(yed.put(key, value))
  770. if any([change[0] for change in changes]):
  771. updated = True
  772. elif edits is not None:
  773. results = Yedit.process_edits(edits, yed)
  774. if results['changed']:
  775. updated = True
  776. if updated:
  777. yed.write()
  778. atexit.register(Utils.cleanup, [fname])
  779. return self._replace(fname, force)
  780. return {'returncode': 0, 'updated': False}
  781. def _replace(self, fname, force=False):
  782. '''replace the current object with oc replace'''
  783. # We are removing the 'resourceVersion' to handle
  784. # a race condition when modifying oc objects
  785. yed = Yedit(fname)
  786. results = yed.delete('metadata.resourceVersion')
  787. if results[0]:
  788. yed.write()
  789. cmd = ['replace', '-f', fname]
  790. if force:
  791. cmd.append('--force')
  792. return self.openshift_cmd(cmd)
  793. def _create_from_content(self, rname, content):
  794. '''create a temporary file and then call oc create on it'''
  795. fname = Utils.create_tmpfile(rname + '-')
  796. yed = Yedit(fname, content=content)
  797. yed.write()
  798. atexit.register(Utils.cleanup, [fname])
  799. return self._create(fname)
  800. def _create(self, fname):
  801. '''call oc create on a filename'''
  802. return self.openshift_cmd(['create', '-f', fname])
  803. def _delete(self, resource, name=None, selector=None):
  804. '''call oc delete on a resource'''
  805. cmd = ['delete', resource]
  806. if selector is not None:
  807. cmd.append('--selector={}'.format(selector))
  808. elif name is not None:
  809. cmd.append(name)
  810. else:
  811. raise OpenShiftCLIError('Either name or selector is required when calling delete.')
  812. return self.openshift_cmd(cmd)
  813. def _process(self, template_name, create=False, params=None, template_data=None): # noqa: E501
  814. '''process a template
  815. template_name: the name of the template to process
  816. create: whether to send to oc create after processing
  817. params: the parameters for the template
  818. template_data: the incoming template's data; instead of a file
  819. '''
  820. cmd = ['process']
  821. if template_data:
  822. cmd.extend(['-f', '-'])
  823. else:
  824. cmd.append(template_name)
  825. if params:
  826. param_str = ["{}={}".format(key, str(value).replace("'", r'"')) for key, value in params.items()]
  827. cmd.append('-p')
  828. cmd.extend(param_str)
  829. results = self.openshift_cmd(cmd, output=True, input_data=template_data)
  830. if results['returncode'] != 0 or not create:
  831. return results
  832. fname = Utils.create_tmpfile(template_name + '-')
  833. yed = Yedit(fname, results['results'])
  834. yed.write()
  835. atexit.register(Utils.cleanup, [fname])
  836. return self.openshift_cmd(['create', '-f', fname])
  837. def _get(self, resource, name=None, selector=None, field_selector=None):
  838. '''return a resource by name '''
  839. cmd = ['get', resource]
  840. if selector is not None:
  841. cmd.append('--selector={}'.format(selector))
  842. if field_selector is not None:
  843. cmd.append('--field-selector={}'.format(field_selector))
  844. # Name cannot be used with selector or field_selector.
  845. if selector is None and field_selector is None and name is not None:
  846. cmd.append(name)
  847. cmd.extend(['-o', 'json'])
  848. rval = self.openshift_cmd(cmd, output=True)
  849. # Ensure results are retuned in an array
  850. if 'items' in rval:
  851. rval['results'] = rval['items']
  852. elif not isinstance(rval['results'], list):
  853. rval['results'] = [rval['results']]
  854. return rval
  855. def _schedulable(self, node=None, selector=None, schedulable=True):
  856. ''' perform oadm manage-node scheduable '''
  857. cmd = ['manage-node']
  858. if node:
  859. cmd.extend(node)
  860. else:
  861. cmd.append('--selector={}'.format(selector))
  862. cmd.append('--schedulable={}'.format(schedulable))
  863. return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw') # noqa: E501
  864. def _list_pods(self, node=None, selector=None, pod_selector=None):
  865. ''' perform oadm list pods
  866. node: the node in which to list pods
  867. selector: the label selector filter if provided
  868. pod_selector: the pod selector filter if provided
  869. '''
  870. cmd = ['manage-node']
  871. if node:
  872. cmd.extend(node)
  873. else:
  874. cmd.append('--selector={}'.format(selector))
  875. if pod_selector:
  876. cmd.append('--pod-selector={}'.format(pod_selector))
  877. cmd.extend(['--list-pods', '-o', 'json'])
  878. return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw')
  879. # pylint: disable=too-many-arguments
  880. def _evacuate(self, node=None, selector=None, pod_selector=None, dry_run=False, grace_period=None, force=False):
  881. ''' perform oadm manage-node evacuate '''
  882. cmd = ['manage-node']
  883. if node:
  884. cmd.extend(node)
  885. else:
  886. cmd.append('--selector={}'.format(selector))
  887. if dry_run:
  888. cmd.append('--dry-run')
  889. if pod_selector:
  890. cmd.append('--pod-selector={}'.format(pod_selector))
  891. if grace_period:
  892. cmd.append('--grace-period={}'.format(int(grace_period)))
  893. if force:
  894. cmd.append('--force')
  895. cmd.append('--evacuate')
  896. return self.openshift_cmd(cmd, oadm=True, output=True, output_type='raw')
  897. def _version(self):
  898. ''' return the openshift version'''
  899. return self.openshift_cmd(['version'], output=True, output_type='raw')
  900. def _import_image(self, url=None, name=None, tag=None):
  901. ''' perform image import '''
  902. cmd = ['import-image']
  903. image = '{0}'.format(name)
  904. if tag:
  905. image += ':{0}'.format(tag)
  906. cmd.append(image)
  907. if url:
  908. cmd.append('--from={0}/{1}'.format(url, image))
  909. cmd.append('-n{0}'.format(self.namespace))
  910. cmd.append('--confirm')
  911. return self.openshift_cmd(cmd)
  912. def _run(self, cmds, input_data):
  913. ''' Actually executes the command. This makes mocking easier. '''
  914. curr_env = os.environ.copy()
  915. curr_env.update({'KUBECONFIG': self.kubeconfig})
  916. proc = subprocess.Popen(cmds,
  917. stdin=subprocess.PIPE,
  918. stdout=subprocess.PIPE,
  919. stderr=subprocess.PIPE,
  920. env=curr_env)
  921. stdout, stderr = proc.communicate(input_data)
  922. return proc.returncode, stdout.decode('utf-8'), stderr.decode('utf-8')
  923. # pylint: disable=too-many-arguments,too-many-branches
  924. def openshift_cmd(self, cmd, oadm=False, output=False, output_type='json', input_data=None):
  925. '''Base command for oc '''
  926. cmds = [self.oc_binary]
  927. if oadm:
  928. cmds.append('adm')
  929. cmds.extend(cmd)
  930. if self.all_namespaces:
  931. cmds.extend(['--all-namespaces'])
  932. elif self.namespace is not None and self.namespace.lower() not in ['none', 'emtpy']: # E501
  933. cmds.extend(['-n', self.namespace])
  934. if self.verbose:
  935. print(' '.join(cmds))
  936. try:
  937. returncode, stdout, stderr = self._run(cmds, input_data)
  938. except OSError as ex:
  939. returncode, stdout, stderr = 1, '', 'Failed to execute {}: {}'.format(subprocess.list2cmdline(cmds), ex)
  940. rval = {"returncode": returncode,
  941. "cmd": ' '.join(cmds)}
  942. if output_type == 'json':
  943. rval['results'] = {}
  944. if output and stdout:
  945. try:
  946. rval['results'] = json.loads(stdout)
  947. except ValueError as verr:
  948. if "No JSON object could be decoded" in verr.args:
  949. rval['err'] = verr.args
  950. elif output_type == 'raw':
  951. rval['results'] = stdout if output else ''
  952. if self.verbose:
  953. print("STDOUT: {0}".format(stdout))
  954. print("STDERR: {0}".format(stderr))
  955. if 'err' in rval or returncode != 0:
  956. rval.update({"stderr": stderr,
  957. "stdout": stdout})
  958. return rval
  959. class Utils(object): # pragma: no cover
  960. ''' utilities for openshiftcli modules '''
  961. @staticmethod
  962. def _write(filename, contents):
  963. ''' Actually write the file contents to disk. This helps with mocking. '''
  964. with open(filename, 'w') as sfd:
  965. sfd.write(str(contents))
  966. @staticmethod
  967. def create_tmp_file_from_contents(rname, data, ftype='yaml'):
  968. ''' create a file in tmp with name and contents'''
  969. tmp = Utils.create_tmpfile(prefix=rname)
  970. if ftype == 'yaml':
  971. # AUDIT:no-member makes sense here due to ruamel.YAML/PyYAML usage
  972. # pylint: disable=no-member
  973. if hasattr(yaml, 'RoundTripDumper'):
  974. Utils._write(tmp, yaml.dump(data, Dumper=yaml.RoundTripDumper))
  975. else:
  976. Utils._write(tmp, yaml.safe_dump(data, default_flow_style=False))
  977. elif ftype == 'json':
  978. Utils._write(tmp, json.dumps(data))
  979. else:
  980. Utils._write(tmp, data)
  981. # Register cleanup when module is done
  982. atexit.register(Utils.cleanup, [tmp])
  983. return tmp
  984. @staticmethod
  985. def create_tmpfile_copy(inc_file):
  986. '''create a temporary copy of a file'''
  987. tmpfile = Utils.create_tmpfile('lib_openshift-')
  988. Utils._write(tmpfile, open(inc_file).read())
  989. # Cleanup the tmpfile
  990. atexit.register(Utils.cleanup, [tmpfile])
  991. return tmpfile
  992. @staticmethod
  993. def create_tmpfile(prefix='tmp'):
  994. ''' Generates and returns a temporary file name '''
  995. with tempfile.NamedTemporaryFile(prefix=prefix, delete=False) as tmp:
  996. return tmp.name
  997. @staticmethod
  998. def create_tmp_files_from_contents(content, content_type=None):
  999. '''Turn an array of dict: filename, content into a files array'''
  1000. if not isinstance(content, list):
  1001. content = [content]
  1002. files = []
  1003. for item in content:
  1004. path = Utils.create_tmp_file_from_contents(item['path'] + '-',
  1005. item['data'],
  1006. ftype=content_type)
  1007. files.append({'name': os.path.basename(item['path']),
  1008. 'path': path})
  1009. return files
  1010. @staticmethod
  1011. def cleanup(files):
  1012. '''Clean up on exit '''
  1013. for sfile in files:
  1014. if os.path.exists(sfile):
  1015. if os.path.isdir(sfile):
  1016. shutil.rmtree(sfile)
  1017. elif os.path.isfile(sfile):
  1018. os.remove(sfile)
  1019. @staticmethod
  1020. def exists(results, _name):
  1021. ''' Check to see if the results include the name '''
  1022. if not results:
  1023. return False
  1024. if Utils.find_result(results, _name):
  1025. return True
  1026. return False
  1027. @staticmethod
  1028. def find_result(results, _name):
  1029. ''' Find the specified result by name'''
  1030. rval = None
  1031. for result in results:
  1032. if 'metadata' in result and result['metadata']['name'] == _name:
  1033. rval = result
  1034. break
  1035. return rval
  1036. @staticmethod
  1037. def get_resource_file(sfile, sfile_type='yaml'):
  1038. ''' return the service file '''
  1039. contents = None
  1040. with open(sfile) as sfd:
  1041. contents = sfd.read()
  1042. if sfile_type == 'yaml':
  1043. # AUDIT:no-member makes sense here due to ruamel.YAML/PyYAML usage
  1044. # pylint: disable=no-member
  1045. if hasattr(yaml, 'RoundTripLoader'):
  1046. contents = yaml.load(contents, yaml.RoundTripLoader)
  1047. else:
  1048. contents = yaml.safe_load(contents)
  1049. elif sfile_type == 'json':
  1050. contents = json.loads(contents)
  1051. return contents
  1052. @staticmethod
  1053. def filter_versions(stdout):
  1054. ''' filter the oc version output '''
  1055. version_dict = {}
  1056. version_search = ['oc', 'openshift', 'kubernetes']
  1057. for line in stdout.strip().split('\n'):
  1058. for term in version_search:
  1059. if not line:
  1060. continue
  1061. if line.startswith(term):
  1062. version_dict[term] = line.split()[-1]
  1063. # horrible hack to get openshift version in Openshift 3.2
  1064. # By default "oc version in 3.2 does not return an "openshift" version
  1065. if "openshift" not in version_dict:
  1066. version_dict["openshift"] = version_dict["oc"]
  1067. return version_dict
  1068. @staticmethod
  1069. def add_custom_versions(versions):
  1070. ''' create custom versions strings '''
  1071. versions_dict = {}
  1072. for tech, version in versions.items():
  1073. # clean up "-" from version
  1074. if "-" in version:
  1075. version = version.split("-")[0]
  1076. if version.startswith('v'):
  1077. versions_dict[tech + '_numeric'] = version[1:].split('+')[0]
  1078. # "v3.3.0.33" is what we have, we want "3.3"
  1079. versions_dict[tech + '_short'] = version[1:4]
  1080. return versions_dict
  1081. @staticmethod
  1082. def openshift_installed():
  1083. ''' check if openshift is installed '''
  1084. import rpm
  1085. transaction_set = rpm.TransactionSet()
  1086. rpmquery = transaction_set.dbMatch("name", "atomic-openshift")
  1087. return rpmquery.count() > 0
  1088. # Disabling too-many-branches. This is a yaml dictionary comparison function
  1089. # pylint: disable=too-many-branches,too-many-return-statements,too-many-statements
  1090. @staticmethod
  1091. def check_def_equal(user_def, result_def, skip_keys=None, debug=False):
  1092. ''' Given a user defined definition, compare it with the results given back by our query. '''
  1093. # Currently these values are autogenerated and we do not need to check them
  1094. skip = ['metadata', 'status']
  1095. if skip_keys:
  1096. skip.extend(skip_keys)
  1097. for key, value in result_def.items():
  1098. if key in skip:
  1099. continue
  1100. # Both are lists
  1101. if isinstance(value, list):
  1102. if key not in user_def:
  1103. if debug:
  1104. print('User data does not have key [%s]' % key)
  1105. print('User data: %s' % user_def)
  1106. return False
  1107. if not isinstance(user_def[key], list):
  1108. if debug:
  1109. print('user_def[key] is not a list key=[%s] user_def[key]=%s' % (key, user_def[key]))
  1110. return False
  1111. if len(user_def[key]) != len(value):
  1112. if debug:
  1113. print("List lengths are not equal.")
  1114. print("key=[%s]: user_def[%s] != value[%s]" % (key, len(user_def[key]), len(value)))
  1115. print("user_def: %s" % user_def[key])
  1116. print("value: %s" % value)
  1117. return False
  1118. for values in zip(user_def[key], value):
  1119. if isinstance(values[0], dict) and isinstance(values[1], dict):
  1120. if debug:
  1121. print('sending list - list')
  1122. print(type(values[0]))
  1123. print(type(values[1]))
  1124. result = Utils.check_def_equal(values[0], values[1], skip_keys=skip_keys, debug=debug)
  1125. if not result:
  1126. print('list compare returned false')
  1127. return False
  1128. elif value != user_def[key]:
  1129. if debug:
  1130. print('value should be identical')
  1131. print(user_def[key])
  1132. print(value)
  1133. return False
  1134. # recurse on a dictionary
  1135. elif isinstance(value, dict):
  1136. if key not in user_def:
  1137. if debug:
  1138. print("user_def does not have key [%s]" % key)
  1139. return False
  1140. if not isinstance(user_def[key], dict):
  1141. if debug:
  1142. print("dict returned false: not instance of dict")
  1143. return False
  1144. # before passing ensure keys match
  1145. api_values = set(value.keys()) - set(skip)
  1146. user_values = set(user_def[key].keys()) - set(skip)
  1147. if api_values != user_values:
  1148. if debug:
  1149. print("keys are not equal in dict")
  1150. print(user_values)
  1151. print(api_values)
  1152. return False
  1153. result = Utils.check_def_equal(user_def[key], value, skip_keys=skip_keys, debug=debug)
  1154. if not result:
  1155. if debug:
  1156. print("dict returned false")
  1157. print(result)
  1158. return False
  1159. # Verify each key, value pair is the same
  1160. else:
  1161. if key not in user_def or value != user_def[key]:
  1162. if debug:
  1163. print("value not equal; user_def does not have key")
  1164. print(key)
  1165. print(value)
  1166. if key in user_def:
  1167. print(user_def[key])
  1168. return False
  1169. if debug:
  1170. print('returning true')
  1171. return True
  1172. class OpenShiftCLIConfig(object):
  1173. '''Generic Config'''
  1174. def __init__(self, rname, namespace, kubeconfig, options):
  1175. self.kubeconfig = kubeconfig
  1176. self.name = rname
  1177. self.namespace = namespace
  1178. self._options = options
  1179. @property
  1180. def config_options(self):
  1181. ''' return config options '''
  1182. return self._options
  1183. def to_option_list(self, ascommalist=''):
  1184. '''return all options as a string
  1185. if ascommalist is set to the name of a key, and
  1186. the value of that key is a dict, format the dict
  1187. as a list of comma delimited key=value pairs'''
  1188. return self.stringify(ascommalist)
  1189. def stringify(self, ascommalist=''):
  1190. ''' return the options hash as cli params in a string
  1191. if ascommalist is set to the name of a key, and
  1192. the value of that key is a dict, format the dict
  1193. as a list of comma delimited key=value pairs '''
  1194. rval = []
  1195. for key in sorted(self.config_options.keys()):
  1196. data = self.config_options[key]
  1197. if data['include'] \
  1198. and (data['value'] is not None or isinstance(data['value'], int)):
  1199. if key == ascommalist:
  1200. val = ','.join(['{}={}'.format(kk, vv) for kk, vv in sorted(data['value'].items())])
  1201. else:
  1202. val = data['value']
  1203. rval.append('--{}={}'.format(key.replace('_', '-'), val))
  1204. return rval
  1205. # -*- -*- -*- End included fragment: lib/base.py -*- -*- -*-
  1206. # -*- -*- -*- Begin included fragment: class/oc_obj.py -*- -*- -*-
  1207. # pylint: disable=too-many-instance-attributes
  1208. class OCObject(OpenShiftCLI):
  1209. ''' Class to wrap the oc command line tools '''
  1210. # pylint allows 5. we need 6
  1211. # pylint: disable=too-many-arguments
  1212. def __init__(self,
  1213. kind,
  1214. namespace,
  1215. name=None,
  1216. selector=None,
  1217. kubeconfig='/etc/origin/master/admin.kubeconfig',
  1218. verbose=False,
  1219. all_namespaces=False,
  1220. field_selector=None):
  1221. ''' Constructor for OpenshiftOC '''
  1222. super(OCObject, self).__init__(namespace, kubeconfig=kubeconfig, verbose=verbose,
  1223. all_namespaces=all_namespaces)
  1224. self.kind = kind
  1225. self.name = name
  1226. self.selector = selector
  1227. self.field_selector = field_selector
  1228. def get(self):
  1229. '''return a kind by name '''
  1230. results = self._get(self.kind, name=self.name, selector=self.selector, field_selector=self.field_selector)
  1231. if (results['returncode'] != 0 and 'stderr' in results and
  1232. '\"{}\" not found'.format(self.name) in results['stderr']):
  1233. results['returncode'] = 0
  1234. return results
  1235. def delete(self):
  1236. '''delete the object'''
  1237. results = self._delete(self.kind, name=self.name, selector=self.selector)
  1238. if (results['returncode'] != 0 and 'stderr' in results and
  1239. '\"{}\" not found'.format(self.name) in results['stderr']):
  1240. results['returncode'] = 0
  1241. return results
  1242. def create(self, files=None, content=None):
  1243. '''
  1244. Create a config
  1245. NOTE: This creates the first file OR the first conent.
  1246. TODO: Handle all files and content passed in
  1247. '''
  1248. if files:
  1249. return self._create(files[0])
  1250. # pylint: disable=no-member
  1251. # The purpose of this change is twofold:
  1252. # - we need a check to only use the ruamel specific dumper if ruamel is loaded
  1253. # - the dumper or the flow style change is needed so openshift is able to parse
  1254. # the resulting yaml, at least until gopkg.in/yaml.v2 is updated
  1255. if hasattr(yaml, 'RoundTripDumper'):
  1256. content['data'] = yaml.dump(content['data'], Dumper=yaml.RoundTripDumper)
  1257. else:
  1258. content['data'] = yaml.safe_dump(content['data'], default_flow_style=False)
  1259. content_file = Utils.create_tmp_files_from_contents(content)[0]
  1260. return self._create(content_file['path'])
  1261. # pylint: disable=too-many-function-args
  1262. def update(self, files=None, content=None, force=False):
  1263. '''update a current openshift object
  1264. This receives a list of file names or content
  1265. and takes the first and calls replace.
  1266. TODO: take an entire list
  1267. '''
  1268. if files:
  1269. return self._replace(files[0], force)
  1270. if content and 'data' in content:
  1271. content = content['data']
  1272. return self.update_content(content, force)
  1273. def update_content(self, content, force=False):
  1274. '''update an object through using the content param'''
  1275. return self._replace_content(self.kind, self.name, content, force=force)
  1276. def needs_update(self, files=None, content=None, content_type='yaml'):
  1277. ''' check to see if we need to update '''
  1278. objects = self.get()
  1279. if objects['returncode'] != 0:
  1280. return objects
  1281. data = None
  1282. if files:
  1283. data = Utils.get_resource_file(files[0], content_type)
  1284. elif content and 'data' in content:
  1285. data = content['data']
  1286. else:
  1287. data = content
  1288. # if equal then no need. So not equal is True
  1289. return not Utils.check_def_equal(data, objects['results'][0], skip_keys=None, debug=False)
  1290. # pylint: disable=too-many-return-statements,too-many-branches
  1291. @staticmethod
  1292. def run_ansible(params, check_mode=False):
  1293. '''perform the ansible idempotent code'''
  1294. ocobj = OCObject(params['kind'],
  1295. params['namespace'],
  1296. params['name'],
  1297. params['selector'],
  1298. kubeconfig=params['kubeconfig'],
  1299. verbose=params['debug'],
  1300. all_namespaces=params['all_namespaces'],
  1301. field_selector=params['field_selector'])
  1302. state = params['state']
  1303. api_rval = ocobj.get()
  1304. #####
  1305. # Get
  1306. #####
  1307. if state == 'list':
  1308. if api_rval['returncode'] != 0:
  1309. return {'changed': False, 'failed': True, 'msg': api_rval}
  1310. return {'changed': False, 'results': api_rval, 'state': state}
  1311. ########
  1312. # Delete
  1313. ########
  1314. if state == 'absent':
  1315. # verify its not in our results
  1316. if (params['name'] is not None or params['selector'] is not None) and \
  1317. (len(api_rval['results']) == 0 or \
  1318. ('items' in api_rval['results'][0] and len(api_rval['results'][0]['items']) == 0)):
  1319. return {'changed': False, 'state': state}
  1320. if check_mode:
  1321. return {'changed': True, 'msg': 'CHECK_MODE: Would have performed a delete'}
  1322. api_rval = ocobj.delete()
  1323. if api_rval['returncode'] != 0:
  1324. return {'failed': True, 'msg': api_rval}
  1325. return {'changed': True, 'results': api_rval, 'state': state}
  1326. # create/update: Must define a name beyond this point
  1327. if not params['name']:
  1328. return {'failed': True, 'msg': 'Please specify a name when state is present.'}
  1329. if state == 'present':
  1330. ########
  1331. # Create
  1332. ########
  1333. if not Utils.exists(api_rval['results'], params['name']):
  1334. if check_mode:
  1335. return {'changed': True, 'msg': 'CHECK_MODE: Would have performed a create'}
  1336. # Create it here
  1337. api_rval = ocobj.create(params['files'], params['content'])
  1338. if api_rval['returncode'] != 0:
  1339. return {'failed': True, 'msg': api_rval}
  1340. # return the created object
  1341. api_rval = ocobj.get()
  1342. if api_rval['returncode'] != 0:
  1343. return {'failed': True, 'msg': api_rval}
  1344. # Remove files
  1345. if params['files'] and params['delete_after']:
  1346. Utils.cleanup(params['files'])
  1347. return {'changed': True, 'results': api_rval, 'state': state}
  1348. ########
  1349. # Update
  1350. ########
  1351. # if a file path is passed, use it.
  1352. update = ocobj.needs_update(params['files'], params['content'])
  1353. if not isinstance(update, bool):
  1354. return {'failed': True, 'msg': update}
  1355. # No changes
  1356. if not update:
  1357. if params['files'] and params['delete_after']:
  1358. Utils.cleanup(params['files'])
  1359. return {'changed': False, 'results': api_rval['results'][0], 'state': state}
  1360. if check_mode:
  1361. return {'changed': True, 'msg': 'CHECK_MODE: Would have performed an update.'}
  1362. api_rval = ocobj.update(params['files'],
  1363. params['content'],
  1364. params['force'])
  1365. if api_rval['returncode'] != 0:
  1366. return {'failed': True, 'msg': api_rval}
  1367. # return the created object
  1368. api_rval = ocobj.get()
  1369. if api_rval['returncode'] != 0:
  1370. return {'failed': True, 'msg': api_rval}
  1371. return {'changed': True, 'results': api_rval, 'state': state}
  1372. # -*- -*- -*- End included fragment: class/oc_obj.py -*- -*- -*-
  1373. # -*- -*- -*- Begin included fragment: ansible/oc_obj.py -*- -*- -*-
  1374. # pylint: disable=too-many-branches
  1375. def main():
  1376. '''
  1377. ansible oc module for services
  1378. '''
  1379. module = AnsibleModule(
  1380. argument_spec=dict(
  1381. kubeconfig=dict(default='/etc/origin/master/admin.kubeconfig', type='str'),
  1382. state=dict(default='present', type='str',
  1383. choices=['present', 'absent', 'list']),
  1384. debug=dict(default=False, type='bool'),
  1385. namespace=dict(default='default', type='str'),
  1386. all_namespaces=dict(defaul=False, type='bool'),
  1387. name=dict(default=None, type='str'),
  1388. files=dict(default=None, type='list'),
  1389. kind=dict(required=True, type='str'),
  1390. delete_after=dict(default=False, type='bool'),
  1391. content=dict(default=None, type='dict'),
  1392. force=dict(default=False, type='bool'),
  1393. selector=dict(default=None, type='str'),
  1394. field_selector=dict(default=None, type='str'),
  1395. ),
  1396. mutually_exclusive=[["content", "files"], ["selector", "name"], ["field_selector", "name"]],
  1397. supports_check_mode=True,
  1398. )
  1399. rval = OCObject.run_ansible(module.params, module.check_mode)
  1400. if 'failed' in rval:
  1401. module.fail_json(**rval)
  1402. module.exit_json(**rval)
  1403. if __name__ == '__main__':
  1404. main()
  1405. # -*- -*- -*- End included fragment: ansible/oc_obj.py -*- -*- -*-