yedit.py 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983
  1. #!/usr/bin/env python
  2. # pylint: disable=missing-docstring
  3. # ___ ___ _ _ ___ ___ _ _____ ___ ___
  4. # / __| __| \| | __| _ \ /_\_ _| __| \
  5. # | (_ | _|| .` | _|| / / _ \| | | _|| |) |
  6. # \___|___|_|\_|___|_|_\/_/_\_\_|_|___|___/_ _____
  7. # | \ / _ \ | \| |/ _ \_ _| | __| \_ _|_ _|
  8. # | |) | (_) | | .` | (_) || | | _|| |) | | | |
  9. # |___/ \___/ |_|\_|\___/ |_| |___|___/___| |_|
  10. #
  11. # Copyright 2016 Red Hat, Inc. and/or its affiliates
  12. # and other contributors as indicated by the @author tags.
  13. #
  14. # Licensed under the Apache License, Version 2.0 (the "License");
  15. # you may not use this file except in compliance with the License.
  16. # You may obtain a copy of the License at
  17. #
  18. # http://www.apache.org/licenses/LICENSE-2.0
  19. #
  20. # Unless required by applicable law or agreed to in writing, software
  21. # distributed under the License is distributed on an "AS IS" BASIS,
  22. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  23. # See the License for the specific language governing permissions and
  24. # limitations under the License.
  25. #
  26. # -*- -*- -*- Begin included fragment: lib/import.py -*- -*- -*-
  27. # pylint: disable=wrong-import-order,wrong-import-position,unused-import
  28. from __future__ import print_function # noqa: F401
  29. import copy # noqa: F401
  30. import fcntl # noqa: F401
  31. import json # noqa: F401
  32. import os # noqa: F401
  33. import re # noqa: F401
  34. import shutil # noqa: F401
  35. import tempfile # noqa: F401
  36. import time # noqa: F401
  37. try:
  38. import ruamel.yaml as yaml # noqa: F401
  39. except ImportError:
  40. import yaml # noqa: F401
  41. from ansible.module_utils.basic import AnsibleModule
  42. # -*- -*- -*- End included fragment: lib/import.py -*- -*- -*-
  43. # -*- -*- -*- Begin included fragment: doc/yedit -*- -*- -*-
  44. DOCUMENTATION = '''
  45. ---
  46. module: yedit
  47. short_description: Create, modify, and idempotently manage yaml files.
  48. description:
  49. - Modify yaml files programmatically.
  50. options:
  51. state:
  52. description:
  53. - State represents whether to create, modify, delete, or list yaml
  54. required: true
  55. default: present
  56. choices: ["present", "absent", "list"]
  57. aliases: []
  58. debug:
  59. description:
  60. - Turn on debug information.
  61. required: false
  62. default: false
  63. aliases: []
  64. src:
  65. description:
  66. - The file that is the target of the modifications.
  67. required: false
  68. default: None
  69. aliases: []
  70. content:
  71. description:
  72. - Content represents the yaml content you desire to work with. This
  73. - could be the file contents to write or the inmemory data to modify.
  74. required: false
  75. default: None
  76. aliases: []
  77. content_type:
  78. description:
  79. - The python type of the content parameter.
  80. required: false
  81. default: 'dict'
  82. aliases: []
  83. key:
  84. description:
  85. - The path to the value you wish to modify. Emtpy string means the top of
  86. - the document.
  87. required: false
  88. default: ''
  89. aliases: []
  90. value:
  91. description:
  92. - The incoming value of parameter 'key'.
  93. required: false
  94. default:
  95. aliases: []
  96. value_type:
  97. description:
  98. - The python type of the incoming value.
  99. required: false
  100. default: ''
  101. aliases: []
  102. update:
  103. description:
  104. - Whether the update should be performed on a dict/hash or list/array
  105. - object.
  106. required: false
  107. default: false
  108. aliases: []
  109. append:
  110. description:
  111. - Whether to append to an array/list. When the key does not exist or is
  112. - null, a new array is created. When the key is of a non-list type,
  113. - nothing is done.
  114. required: false
  115. default: false
  116. aliases: []
  117. index:
  118. description:
  119. - Used in conjunction with the update parameter. This will update a
  120. - specific index in an array/list.
  121. required: false
  122. default: false
  123. aliases: []
  124. curr_value:
  125. description:
  126. - Used in conjunction with the update parameter. This is the current
  127. - value of 'key' in the yaml file.
  128. required: false
  129. default: false
  130. aliases: []
  131. curr_value_format:
  132. description:
  133. - Format of the incoming current value.
  134. choices: ["yaml", "json", "str"]
  135. required: false
  136. default: false
  137. aliases: []
  138. backup_ext:
  139. description:
  140. - The backup file's appended string.
  141. required: false
  142. default: .orig
  143. aliases: []
  144. backup:
  145. description:
  146. - Whether to make a backup copy of the current file when performing an
  147. - edit.
  148. required: false
  149. default: true
  150. aliases: []
  151. separator:
  152. description:
  153. - The separator being used when parsing strings.
  154. required: false
  155. default: '.'
  156. aliases: []
  157. author:
  158. - "Kenny Woodson <kwoodson@redhat.com>"
  159. extends_documentation_fragment: []
  160. '''
  161. EXAMPLES = '''
  162. # Simple insert of key, value
  163. - name: insert simple key, value
  164. yedit:
  165. src: somefile.yml
  166. key: test
  167. value: somevalue
  168. state: present
  169. # Results:
  170. # test: somevalue
  171. # Multilevel insert of key, value
  172. - name: insert simple key, value
  173. yedit:
  174. src: somefile.yml
  175. key: a#b#c
  176. value: d
  177. state: present
  178. # Results:
  179. # a:
  180. # b:
  181. # c: d
  182. #
  183. # multiple edits at the same time
  184. - name: perform multiple edits
  185. yedit:
  186. src: somefile.yml
  187. edits:
  188. - key: a#b#c
  189. value: d
  190. - key: a#b#c#d
  191. value: e
  192. state: present
  193. # Results:
  194. # a:
  195. # b:
  196. # c:
  197. # d: e
  198. '''
  199. # -*- -*- -*- End included fragment: doc/yedit -*- -*- -*-
  200. # -*- -*- -*- Begin included fragment: class/yedit.py -*- -*- -*-
  201. class YeditException(Exception):
  202. ''' Exception class for Yedit '''
  203. pass
  204. # pylint: disable=too-many-public-methods,too-many-instance-attributes
  205. class Yedit(object):
  206. ''' Class to modify yaml files '''
  207. re_valid_key = r"(((\[-?\d+\])|([0-9a-zA-Z%s/_-]+)).?)+$"
  208. re_key = r"(?:\[(-?\d+)\])|([0-9a-zA-Z{}/_-]+)"
  209. com_sep = set(['.', '#', '|', ':'])
  210. # pylint: disable=too-many-arguments
  211. def __init__(self,
  212. filename=None,
  213. content=None,
  214. content_type='yaml',
  215. separator='.',
  216. backup_ext=None,
  217. backup=False):
  218. self.content = content
  219. self._separator = separator
  220. self.filename = filename
  221. self.__yaml_dict = content
  222. self.content_type = content_type
  223. self.backup = backup
  224. if backup_ext is None:
  225. self.backup_ext = ".{}".format(time.strftime("%Y%m%dT%H%M%S"))
  226. else:
  227. self.backup_ext = backup_ext
  228. self.load(content_type=self.content_type)
  229. if self.__yaml_dict is None:
  230. self.__yaml_dict = {}
  231. @property
  232. def separator(self):
  233. ''' getter method for separator '''
  234. return self._separator
  235. @separator.setter
  236. def separator(self, inc_sep):
  237. ''' setter method for separator '''
  238. self._separator = inc_sep
  239. @property
  240. def yaml_dict(self):
  241. ''' getter method for yaml_dict '''
  242. return self.__yaml_dict
  243. @yaml_dict.setter
  244. def yaml_dict(self, value):
  245. ''' setter method for yaml_dict '''
  246. self.__yaml_dict = value
  247. @staticmethod
  248. def parse_key(key, sep='.'):
  249. '''parse the key allowing the appropriate separator'''
  250. common_separators = list(Yedit.com_sep - set([sep]))
  251. return re.findall(Yedit.re_key.format(''.join(common_separators)), key)
  252. @staticmethod
  253. def valid_key(key, sep='.'):
  254. '''validate the incoming key'''
  255. common_separators = list(Yedit.com_sep - set([sep]))
  256. if not re.match(Yedit.re_valid_key.format(''.join(common_separators)), key):
  257. return False
  258. return True
  259. # pylint: disable=too-many-return-statements,too-many-branches
  260. @staticmethod
  261. def remove_entry(data, key, index=None, value=None, sep='.'):
  262. ''' remove data at location key '''
  263. if key == '' and isinstance(data, dict):
  264. if value is not None:
  265. data.pop(value)
  266. elif index is not None:
  267. raise YeditException("remove_entry for a dictionary does not have an index {}".format(index))
  268. else:
  269. data.clear()
  270. return True
  271. elif key == '' and isinstance(data, list):
  272. ind = None
  273. if value is not None:
  274. try:
  275. ind = data.index(value)
  276. except ValueError:
  277. return False
  278. elif index is not None:
  279. ind = index
  280. else:
  281. del data[:]
  282. if ind is not None:
  283. data.pop(ind)
  284. return True
  285. if not (key and Yedit.valid_key(key, sep)) and \
  286. isinstance(data, (list, dict)):
  287. return None
  288. key_indexes = Yedit.parse_key(key, sep)
  289. for arr_ind, dict_key in key_indexes[:-1]:
  290. if dict_key and isinstance(data, dict):
  291. data = data.get(dict_key)
  292. elif (arr_ind and isinstance(data, list) and
  293. int(arr_ind) <= len(data) - 1):
  294. data = data[int(arr_ind)]
  295. else:
  296. return None
  297. # process last index for remove
  298. # expected list entry
  299. if key_indexes[-1][0]:
  300. if isinstance(data, list) and int(key_indexes[-1][0]) <= len(data) - 1: # noqa: E501
  301. del data[int(key_indexes[-1][0])]
  302. return True
  303. # expected dict entry
  304. elif key_indexes[-1][1]:
  305. if isinstance(data, dict):
  306. del data[key_indexes[-1][1]]
  307. return True
  308. @staticmethod
  309. def add_entry(data, key, item=None, sep='.'):
  310. ''' Get an item from a dictionary with key notation a.b.c
  311. d = {'a': {'b': 'c'}}}
  312. key = a#b
  313. return c
  314. '''
  315. if key == '':
  316. pass
  317. elif (not (key and Yedit.valid_key(key, sep)) and
  318. isinstance(data, (list, dict))):
  319. return None
  320. key_indexes = Yedit.parse_key(key, sep)
  321. for arr_ind, dict_key in key_indexes[:-1]:
  322. if dict_key:
  323. if isinstance(data, dict) and dict_key in data and data[dict_key]: # noqa: E501
  324. data = data[dict_key]
  325. continue
  326. elif data and not isinstance(data, dict):
  327. raise YeditException("Unexpected item type found while going through key " +
  328. "path: {} (at key: {})".format(key, dict_key))
  329. data[dict_key] = {}
  330. data = data[dict_key]
  331. elif (arr_ind and isinstance(data, list) and
  332. int(arr_ind) <= len(data) - 1):
  333. data = data[int(arr_ind)]
  334. else:
  335. raise YeditException("Unexpected item type found while going through key path: {}".format(key))
  336. if key == '':
  337. data = item
  338. # process last index for add
  339. # expected list entry
  340. elif key_indexes[-1][0] and isinstance(data, list) and int(key_indexes[-1][0]) <= len(data) - 1: # noqa: E501
  341. data[int(key_indexes[-1][0])] = item
  342. # expected dict entry
  343. elif key_indexes[-1][1] and isinstance(data, dict):
  344. data[key_indexes[-1][1]] = item
  345. # didn't add/update to an existing list, nor add/update key to a dict
  346. # so we must have been provided some syntax like a.b.c[<int>] = "data" for a
  347. # non-existent array
  348. else:
  349. raise YeditException("Error adding to object at path: {}".format(key))
  350. return data
  351. @staticmethod
  352. def get_entry(data, key, sep='.'):
  353. ''' Get an item from a dictionary with key notation a.b.c
  354. d = {'a': {'b': 'c'}}}
  355. key = a.b
  356. return c
  357. '''
  358. if key == '':
  359. pass
  360. elif (not (key and Yedit.valid_key(key, sep)) and
  361. isinstance(data, (list, dict))):
  362. return None
  363. key_indexes = Yedit.parse_key(key, sep)
  364. for arr_ind, dict_key in key_indexes:
  365. if dict_key and isinstance(data, dict):
  366. data = data.get(dict_key)
  367. elif (arr_ind and isinstance(data, list) and
  368. int(arr_ind) <= len(data) - 1):
  369. data = data[int(arr_ind)]
  370. else:
  371. return None
  372. return data
  373. @staticmethod
  374. def _write(filename, contents):
  375. ''' Actually write the file contents to disk. This helps with mocking. '''
  376. tmp_filename = filename + '.yedit'
  377. with open(tmp_filename, 'w') as yfd:
  378. fcntl.flock(yfd, fcntl.LOCK_EX | fcntl.LOCK_NB)
  379. yfd.write(contents)
  380. fcntl.flock(yfd, fcntl.LOCK_UN)
  381. os.rename(tmp_filename, filename)
  382. def write(self):
  383. ''' write to file '''
  384. if not self.filename:
  385. raise YeditException('Please specify a filename.')
  386. if self.backup and self.file_exists():
  387. shutil.copy(self.filename, '{}{}'.format(self.filename, self.backup_ext))
  388. # Try to set format attributes if supported
  389. try:
  390. self.yaml_dict.fa.set_block_style()
  391. except AttributeError:
  392. pass
  393. # Try to use RoundTripDumper if supported.
  394. if self.content_type == 'yaml':
  395. try:
  396. Yedit._write(self.filename, yaml.dump(self.yaml_dict, Dumper=yaml.RoundTripDumper))
  397. except AttributeError:
  398. Yedit._write(self.filename, yaml.safe_dump(self.yaml_dict, default_flow_style=False))
  399. elif self.content_type == 'json':
  400. Yedit._write(self.filename, json.dumps(self.yaml_dict, indent=4, sort_keys=True))
  401. else:
  402. raise YeditException('Unsupported content_type: {}.'.format(self.content_type) +
  403. 'Please specify a content_type of yaml or json.')
  404. return (True, self.yaml_dict)
  405. def read(self):
  406. ''' read from file '''
  407. # check if it exists
  408. if self.filename is None or not self.file_exists():
  409. return None
  410. contents = None
  411. with open(self.filename) as yfd:
  412. contents = yfd.read()
  413. return contents
  414. def file_exists(self):
  415. ''' return whether file exists '''
  416. if os.path.exists(self.filename):
  417. return True
  418. return False
  419. def load(self, content_type='yaml'):
  420. ''' return yaml file '''
  421. contents = self.read()
  422. if not contents and not self.content:
  423. return None
  424. if self.content:
  425. if isinstance(self.content, dict):
  426. self.yaml_dict = self.content
  427. return self.yaml_dict
  428. elif isinstance(self.content, str):
  429. contents = self.content
  430. # check if it is yaml
  431. try:
  432. if content_type == 'yaml' and contents:
  433. # Try to set format attributes if supported
  434. try:
  435. self.yaml_dict.fa.set_block_style()
  436. except AttributeError:
  437. pass
  438. # Try to use RoundTripLoader if supported.
  439. try:
  440. self.yaml_dict = yaml.load(contents, yaml.RoundTripLoader)
  441. except AttributeError:
  442. self.yaml_dict = yaml.safe_load(contents)
  443. # Try to set format attributes if supported
  444. try:
  445. self.yaml_dict.fa.set_block_style()
  446. except AttributeError:
  447. pass
  448. elif content_type == 'json' and contents:
  449. self.yaml_dict = json.loads(contents)
  450. except yaml.YAMLError as err:
  451. # Error loading yaml or json
  452. raise YeditException('Problem with loading yaml file. {}'.format(err))
  453. return self.yaml_dict
  454. def get(self, key):
  455. ''' get a specified key'''
  456. try:
  457. entry = Yedit.get_entry(self.yaml_dict, key, self.separator)
  458. except KeyError:
  459. entry = None
  460. return entry
  461. def pop(self, path, key_or_item):
  462. ''' remove a key, value pair from a dict or an item for a list'''
  463. try:
  464. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  465. except KeyError:
  466. entry = None
  467. if entry is None:
  468. return (False, self.yaml_dict)
  469. if isinstance(entry, dict):
  470. # AUDIT:maybe-no-member makes sense due to fuzzy types
  471. # pylint: disable=maybe-no-member
  472. if key_or_item in entry:
  473. entry.pop(key_or_item)
  474. return (True, self.yaml_dict)
  475. return (False, self.yaml_dict)
  476. elif isinstance(entry, list):
  477. # AUDIT:maybe-no-member makes sense due to fuzzy types
  478. # pylint: disable=maybe-no-member
  479. ind = None
  480. try:
  481. ind = entry.index(key_or_item)
  482. except ValueError:
  483. return (False, self.yaml_dict)
  484. entry.pop(ind)
  485. return (True, self.yaml_dict)
  486. return (False, self.yaml_dict)
  487. def delete(self, path, index=None, value=None):
  488. ''' remove path from a dict'''
  489. try:
  490. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  491. except KeyError:
  492. entry = None
  493. if entry is None:
  494. return (False, self.yaml_dict)
  495. result = Yedit.remove_entry(self.yaml_dict, path, index, value, self.separator)
  496. if not result:
  497. return (False, self.yaml_dict)
  498. return (True, self.yaml_dict)
  499. def exists(self, path, value):
  500. ''' check if value exists at path'''
  501. try:
  502. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  503. except KeyError:
  504. entry = None
  505. if isinstance(entry, list):
  506. if value in entry:
  507. return True
  508. return False
  509. elif isinstance(entry, dict):
  510. if isinstance(value, dict):
  511. rval = False
  512. for key, val in value.items():
  513. if entry[key] != val:
  514. rval = False
  515. break
  516. else:
  517. rval = True
  518. return rval
  519. return value in entry
  520. return entry == value
  521. def append(self, path, value):
  522. '''append value to a list'''
  523. try:
  524. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  525. except KeyError:
  526. entry = None
  527. if entry is None:
  528. self.put(path, [])
  529. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  530. if not isinstance(entry, list):
  531. return (False, self.yaml_dict)
  532. # AUDIT:maybe-no-member makes sense due to loading data from
  533. # a serialized format.
  534. # pylint: disable=maybe-no-member
  535. entry.append(value)
  536. return (True, self.yaml_dict)
  537. # pylint: disable=too-many-arguments
  538. def update(self, path, value, index=None, curr_value=None):
  539. ''' put path, value into a dict '''
  540. try:
  541. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  542. except KeyError:
  543. entry = None
  544. if isinstance(entry, dict):
  545. # AUDIT:maybe-no-member makes sense due to fuzzy types
  546. # pylint: disable=maybe-no-member
  547. if not isinstance(value, dict):
  548. raise YeditException('Cannot replace key, value entry in dict with non-dict type. ' +
  549. 'value=[{}] type=[{}]'.format(value, type(value)))
  550. entry.update(value)
  551. return (True, self.yaml_dict)
  552. elif isinstance(entry, list):
  553. # AUDIT:maybe-no-member makes sense due to fuzzy types
  554. # pylint: disable=maybe-no-member
  555. ind = None
  556. if curr_value:
  557. try:
  558. ind = entry.index(curr_value)
  559. except ValueError:
  560. return (False, self.yaml_dict)
  561. elif index is not None:
  562. ind = index
  563. if ind is not None and entry[ind] != value:
  564. entry[ind] = value
  565. return (True, self.yaml_dict)
  566. # see if it exists in the list
  567. try:
  568. ind = entry.index(value)
  569. except ValueError:
  570. # doesn't exist, append it
  571. entry.append(value)
  572. return (True, self.yaml_dict)
  573. # already exists, return
  574. if ind is not None:
  575. return (False, self.yaml_dict)
  576. return (False, self.yaml_dict)
  577. def put(self, path, value):
  578. ''' put path, value into a dict '''
  579. try:
  580. entry = Yedit.get_entry(self.yaml_dict, path, self.separator)
  581. except KeyError:
  582. entry = None
  583. if entry == value:
  584. return (False, self.yaml_dict)
  585. # deepcopy didn't work
  586. # Try to use ruamel.yaml and fallback to pyyaml
  587. try:
  588. tmp_copy = yaml.load(yaml.round_trip_dump(self.yaml_dict,
  589. default_flow_style=False),
  590. yaml.RoundTripLoader)
  591. except AttributeError:
  592. tmp_copy = copy.deepcopy(self.yaml_dict)
  593. # set the format attributes if available
  594. try:
  595. tmp_copy.fa.set_block_style()
  596. except AttributeError:
  597. pass
  598. result = Yedit.add_entry(tmp_copy, path, value, self.separator)
  599. if result is None:
  600. return (False, self.yaml_dict)
  601. # When path equals "" it is a special case.
  602. # "" refers to the root of the document
  603. # Only update the root path (entire document) when its a list or dict
  604. if path == '':
  605. if isinstance(result, list) or isinstance(result, dict):
  606. self.yaml_dict = result
  607. return (True, self.yaml_dict)
  608. return (False, self.yaml_dict)
  609. self.yaml_dict = tmp_copy
  610. return (True, self.yaml_dict)
  611. def create(self, path, value):
  612. ''' create a yaml file '''
  613. if not self.file_exists():
  614. # deepcopy didn't work
  615. # Try to use ruamel.yaml and fallback to pyyaml
  616. try:
  617. tmp_copy = yaml.load(yaml.round_trip_dump(self.yaml_dict,
  618. default_flow_style=False),
  619. yaml.RoundTripLoader)
  620. except AttributeError:
  621. tmp_copy = copy.deepcopy(self.yaml_dict)
  622. # set the format attributes if available
  623. try:
  624. tmp_copy.fa.set_block_style()
  625. except AttributeError:
  626. pass
  627. result = Yedit.add_entry(tmp_copy, path, value, self.separator)
  628. if result is not None:
  629. self.yaml_dict = tmp_copy
  630. return (True, self.yaml_dict)
  631. return (False, self.yaml_dict)
  632. @staticmethod
  633. def get_curr_value(invalue, val_type):
  634. '''return the current value'''
  635. if invalue is None:
  636. return None
  637. curr_value = invalue
  638. if val_type == 'yaml':
  639. curr_value = yaml.safe_load(str(invalue))
  640. elif val_type == 'json':
  641. curr_value = json.loads(invalue)
  642. return curr_value
  643. @staticmethod
  644. def parse_value(inc_value, vtype=''):
  645. '''determine value type passed'''
  646. true_bools = ['y', 'Y', 'yes', 'Yes', 'YES', 'true', 'True', 'TRUE',
  647. 'on', 'On', 'ON', ]
  648. false_bools = ['n', 'N', 'no', 'No', 'NO', 'false', 'False', 'FALSE',
  649. 'off', 'Off', 'OFF']
  650. # It came in as a string but you didn't specify value_type as string
  651. # we will convert to bool if it matches any of the above cases
  652. if isinstance(inc_value, str) and 'bool' in vtype:
  653. if inc_value not in true_bools and inc_value not in false_bools:
  654. raise YeditException('Not a boolean type. str=[{}] vtype=[{}]'.format(inc_value, vtype))
  655. elif isinstance(inc_value, bool) and 'str' in vtype:
  656. inc_value = str(inc_value)
  657. # There is a special case where '' will turn into None after yaml loading it so skip
  658. if isinstance(inc_value, str) and inc_value == '':
  659. pass
  660. # If vtype is not str then go ahead and attempt to yaml load it.
  661. elif isinstance(inc_value, str) and 'str' not in vtype:
  662. try:
  663. inc_value = yaml.safe_load(inc_value)
  664. except Exception:
  665. raise YeditException('Could not determine type of incoming value. ' +
  666. 'value=[{}] vtype=[{}]'.format(type(inc_value), vtype))
  667. return inc_value
  668. @staticmethod
  669. def process_edits(edits, yamlfile):
  670. '''run through a list of edits and process them one-by-one'''
  671. results = []
  672. for edit in edits:
  673. value = Yedit.parse_value(edit['value'], edit.get('value_type', ''))
  674. if edit.get('action') == 'update':
  675. # pylint: disable=line-too-long
  676. curr_value = Yedit.get_curr_value(
  677. Yedit.parse_value(edit.get('curr_value')),
  678. edit.get('curr_value_format'))
  679. rval = yamlfile.update(edit['key'],
  680. value,
  681. edit.get('index'),
  682. curr_value)
  683. elif edit.get('action') == 'append':
  684. rval = yamlfile.append(edit['key'], value)
  685. else:
  686. rval = yamlfile.put(edit['key'], value)
  687. if rval[0]:
  688. results.append({'key': edit['key'], 'edit': rval[1]})
  689. return {'changed': len(results) > 0, 'results': results}
  690. # pylint: disable=too-many-return-statements,too-many-branches
  691. @staticmethod
  692. def run_ansible(params):
  693. '''perform the idempotent crud operations'''
  694. yamlfile = Yedit(filename=params['src'],
  695. backup=params['backup'],
  696. content_type=params['content_type'],
  697. backup_ext=params['backup_ext'],
  698. separator=params['separator'])
  699. state = params['state']
  700. if params['src']:
  701. rval = yamlfile.load()
  702. if yamlfile.yaml_dict is None and state != 'present':
  703. return {'failed': True,
  704. 'msg': 'Error opening file [{}]. Verify that the '.format(params['src']) +
  705. 'file exists, that it is has correct permissions, and is valid yaml.'}
  706. if state == 'list':
  707. if params['content']:
  708. content = Yedit.parse_value(params['content'], params['content_type'])
  709. yamlfile.yaml_dict = content
  710. if params['key']:
  711. rval = yamlfile.get(params['key'])
  712. return {'changed': False, 'result': rval, 'state': state}
  713. elif state == 'absent':
  714. if params['content']:
  715. content = Yedit.parse_value(params['content'], params['content_type'])
  716. yamlfile.yaml_dict = content
  717. if params['update']:
  718. rval = yamlfile.pop(params['key'], params['value'])
  719. else:
  720. rval = yamlfile.delete(params['key'], params['index'], params['value'])
  721. if rval[0] and params['src']:
  722. yamlfile.write()
  723. return {'changed': rval[0], 'result': rval[1], 'state': state}
  724. elif state == 'present':
  725. # check if content is different than what is in the file
  726. if params['content']:
  727. content = Yedit.parse_value(params['content'], params['content_type'])
  728. # We had no edits to make and the contents are the same
  729. if yamlfile.yaml_dict == content and \
  730. params['value'] is None:
  731. return {'changed': False, 'result': yamlfile.yaml_dict, 'state': state}
  732. yamlfile.yaml_dict = content
  733. # If we were passed a key, value then
  734. # we enapsulate it in a list and process it
  735. # Key, Value passed to the module : Converted to Edits list #
  736. edits = []
  737. _edit = {}
  738. if params['value'] is not None:
  739. _edit['value'] = params['value']
  740. _edit['value_type'] = params['value_type']
  741. _edit['key'] = params['key']
  742. if params['update']:
  743. _edit['action'] = 'update'
  744. _edit['curr_value'] = params['curr_value']
  745. _edit['curr_value_format'] = params['curr_value_format']
  746. _edit['index'] = params['index']
  747. elif params['append']:
  748. _edit['action'] = 'append'
  749. edits.append(_edit)
  750. elif params['edits'] is not None:
  751. edits = params['edits']
  752. if edits:
  753. results = Yedit.process_edits(edits, yamlfile)
  754. # if there were changes and a src provided to us we need to write
  755. if results['changed'] and params['src']:
  756. yamlfile.write()
  757. return {'changed': results['changed'], 'result': results['results'], 'state': state}
  758. # no edits to make
  759. if params['src']:
  760. # pylint: disable=redefined-variable-type
  761. rval = yamlfile.write()
  762. return {'changed': rval[0],
  763. 'result': rval[1],
  764. 'state': state}
  765. # We were passed content but no src, key or value, or edits. Return contents in memory
  766. return {'changed': False, 'result': yamlfile.yaml_dict, 'state': state}
  767. return {'failed': True, 'msg': 'Unkown state passed'}
  768. # -*- -*- -*- End included fragment: class/yedit.py -*- -*- -*-
  769. # -*- -*- -*- Begin included fragment: ansible/yedit.py -*- -*- -*-
  770. # pylint: disable=too-many-branches
  771. def main():
  772. ''' ansible oc module for secrets '''
  773. module = AnsibleModule(
  774. argument_spec=dict(
  775. state=dict(default='present', type='str',
  776. choices=['present', 'absent', 'list']),
  777. debug=dict(default=False, type='bool'),
  778. src=dict(default=None, type='str'),
  779. content=dict(default=None),
  780. content_type=dict(default='yaml', choices=['yaml', 'json']),
  781. key=dict(default='', type='str'),
  782. value=dict(),
  783. value_type=dict(default='', type='str'),
  784. update=dict(default=False, type='bool'),
  785. append=dict(default=False, type='bool'),
  786. index=dict(default=None, type='int'),
  787. curr_value=dict(default=None, type='str'),
  788. curr_value_format=dict(default='yaml',
  789. choices=['yaml', 'json', 'str'],
  790. type='str'),
  791. backup=dict(default=False, type='bool'),
  792. backup_ext=dict(default=".{}".format(time.strftime("%Y%m%dT%H%M%S")), type='str'),
  793. separator=dict(default='.', type='str'),
  794. edits=dict(default=None, type='list'),
  795. ),
  796. mutually_exclusive=[["curr_value", "index"], ['update', "append"]],
  797. required_one_of=[["content", "src"]],
  798. )
  799. # Verify we recieved either a valid key or edits with valid keys when receiving a src file.
  800. # A valid key being not None or not ''.
  801. if module.params['src'] is not None:
  802. key_error = False
  803. edit_error = False
  804. if module.params['key'] in [None, '']:
  805. key_error = True
  806. if module.params['edits'] in [None, []]:
  807. edit_error = True
  808. else:
  809. for edit in module.params['edits']:
  810. if edit.get('key') in [None, '']:
  811. edit_error = True
  812. break
  813. if key_error and edit_error:
  814. module.fail_json(failed=True, msg='Empty value for parameter key not allowed.')
  815. rval = Yedit.run_ansible(module.params)
  816. if 'failed' in rval and rval['failed']:
  817. module.fail_json(**rval)
  818. module.exit_json(**rval)
  819. if __name__ == '__main__':
  820. main()
  821. # -*- -*- -*- End included fragment: ansible/yedit.py -*- -*- -*-