repoquery.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644
  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/repoquery -*- -*- -*-
  44. DOCUMENTATION = '''
  45. ---
  46. module: repoquery
  47. short_description: Query package information from Yum repositories
  48. description:
  49. - Query package information from Yum repositories.
  50. options:
  51. state:
  52. description:
  53. - The expected state. Currently only supports list.
  54. required: false
  55. default: list
  56. choices: ["list"]
  57. aliases: []
  58. name:
  59. description:
  60. - The name of the package to query
  61. required: true
  62. default: None
  63. aliases: []
  64. query_type:
  65. description:
  66. - Narrows the packages queried based off of this value.
  67. - If repos, it narrows the query to repositories defined on the machine.
  68. - If installed, it narrows the query to only packages installed on the machine.
  69. - If available, it narrows the query to packages that are available to be installed.
  70. - If recent, it narrows the query to only recently edited packages.
  71. - If updates, it narrows the query to only packages that are updates to existing installed packages.
  72. - If extras, it narrows the query to packages that are not present in any of the available repositories.
  73. - If all, it queries all of the above.
  74. required: false
  75. default: repos
  76. aliases: []
  77. verbose:
  78. description:
  79. - Shows more detail for the requested query.
  80. required: false
  81. default: false
  82. aliases: []
  83. show_duplicates:
  84. description:
  85. - Shows multiple versions of a package.
  86. required: false
  87. default: false
  88. aliases: []
  89. match_version:
  90. description:
  91. - Match the specific version given to the package.
  92. required: false
  93. default: None
  94. aliases: []
  95. author:
  96. - "Matt Woodson <mwoodson@redhat.com>"
  97. extends_documentation_fragment: []
  98. '''
  99. EXAMPLES = '''
  100. # Example 1: Get bash versions
  101. - name: Get bash version
  102. repoquery:
  103. name: bash
  104. show_duplicates: True
  105. register: bash_out
  106. # Results:
  107. # ok: [localhost] => {
  108. # "bash_out": {
  109. # "changed": false,
  110. # "results": {
  111. # "cmd": "/usr/bin/repoquery --quiet --pkgnarrow=repos --queryformat=%{version}|%{release}|%{arch}|%{repo}|%{version}-%{release} --show-duplicates bash",
  112. # "package_found": true,
  113. # "package_name": "bash",
  114. # "returncode": 0,
  115. # "versions": {
  116. # "available_versions": [
  117. # "4.2.45",
  118. # "4.2.45",
  119. # "4.2.45",
  120. # "4.2.46",
  121. # "4.2.46",
  122. # "4.2.46",
  123. # "4.2.46"
  124. # ],
  125. # "available_versions_full": [
  126. # "4.2.45-5.el7",
  127. # "4.2.45-5.el7_0.2",
  128. # "4.2.45-5.el7_0.4",
  129. # "4.2.46-12.el7",
  130. # "4.2.46-19.el7",
  131. # "4.2.46-20.el7_2",
  132. # "4.2.46-21.el7_3"
  133. # ],
  134. # "latest": "4.2.46",
  135. # "latest_full": "4.2.46-21.el7_3"
  136. # }
  137. # },
  138. # "state": "present"
  139. # }
  140. # }
  141. # Example 2: Get bash versions verbosely
  142. - name: Get bash versions verbosely
  143. repoquery:
  144. name: bash
  145. show_duplicates: True
  146. verbose: True
  147. register: bash_out
  148. # Results:
  149. # ok: [localhost] => {
  150. # "bash_out": {
  151. # "changed": false,
  152. # "results": {
  153. # "cmd": "/usr/bin/repoquery --quiet --pkgnarrow=repos --queryformat=%{version}|%{release}|%{arch}|%{repo}|%{version}-%{release} --show-duplicates bash",
  154. # "package_found": true,
  155. # "package_name": "bash",
  156. # "raw_versions": {
  157. # "4.2.45-5.el7": {
  158. # "arch": "x86_64",
  159. # "release": "5.el7",
  160. # "repo": "rhel-7-server-rpms",
  161. # "version": "4.2.45",
  162. # "version_release": "4.2.45-5.el7"
  163. # },
  164. # "4.2.45-5.el7_0.2": {
  165. # "arch": "x86_64",
  166. # "release": "5.el7_0.2",
  167. # "repo": "rhel-7-server-rpms",
  168. # "version": "4.2.45",
  169. # "version_release": "4.2.45-5.el7_0.2"
  170. # },
  171. # "4.2.45-5.el7_0.4": {
  172. # "arch": "x86_64",
  173. # "release": "5.el7_0.4",
  174. # "repo": "rhel-7-server-rpms",
  175. # "version": "4.2.45",
  176. # "version_release": "4.2.45-5.el7_0.4"
  177. # },
  178. # "4.2.46-12.el7": {
  179. # "arch": "x86_64",
  180. # "release": "12.el7",
  181. # "repo": "rhel-7-server-rpms",
  182. # "version": "4.2.46",
  183. # "version_release": "4.2.46-12.el7"
  184. # },
  185. # "4.2.46-19.el7": {
  186. # "arch": "x86_64",
  187. # "release": "19.el7",
  188. # "repo": "rhel-7-server-rpms",
  189. # "version": "4.2.46",
  190. # "version_release": "4.2.46-19.el7"
  191. # },
  192. # "4.2.46-20.el7_2": {
  193. # "arch": "x86_64",
  194. # "release": "20.el7_2",
  195. # "repo": "rhel-7-server-rpms",
  196. # "version": "4.2.46",
  197. # "version_release": "4.2.46-20.el7_2"
  198. # },
  199. # "4.2.46-21.el7_3": {
  200. # "arch": "x86_64",
  201. # "release": "21.el7_3",
  202. # "repo": "rhel-7-server-rpms",
  203. # "version": "4.2.46",
  204. # "version_release": "4.2.46-21.el7_3"
  205. # }
  206. # },
  207. # "results": "4.2.45|5.el7|x86_64|rhel-7-server-rpms|4.2.45-5.el7\n4.2.45|5.el7_0.2|x86_64|rhel-7-server-rpms|4.2.45-5.el7_0.2\n4.2.45|5.el7_0.4|x86_64|rhel-7-server-rpms|4.2.45-5.el7_0.4\n4.2.46|12.el7|x86_64|rhel-7-server-rpms|4.2.46-12.el7\n4.2.46|19.el7|x86_64|rhel-7-server-rpms|4.2.46-19.el7\n4.2.46|20.el7_2|x86_64|rhel-7-server-rpms|4.2.46-20.el7_2\n4.2.46|21.el7_3|x86_64|rhel-7-server-rpms|4.2.46-21.el7_3\n",
  208. # "returncode": 0,
  209. # "versions": {
  210. # "available_versions": [
  211. # "4.2.45",
  212. # "4.2.45",
  213. # "4.2.45",
  214. # "4.2.46",
  215. # "4.2.46",
  216. # "4.2.46",
  217. # "4.2.46"
  218. # ],
  219. # "available_versions_full": [
  220. # "4.2.45-5.el7",
  221. # "4.2.45-5.el7_0.2",
  222. # "4.2.45-5.el7_0.4",
  223. # "4.2.46-12.el7",
  224. # "4.2.46-19.el7",
  225. # "4.2.46-20.el7_2",
  226. # "4.2.46-21.el7_3"
  227. # ],
  228. # "latest": "4.2.46",
  229. # "latest_full": "4.2.46-21.el7_3"
  230. # }
  231. # },
  232. # "state": "present"
  233. # }
  234. # }
  235. # Example 3: Match a specific version
  236. - name: matched versions repoquery test
  237. repoquery:
  238. name: atomic-openshift
  239. show_duplicates: True
  240. match_version: 3.3
  241. register: openshift_out
  242. # Result:
  243. # ok: [localhost] => {
  244. # "openshift_out": {
  245. # "changed": false,
  246. # "results": {
  247. # "cmd": "/usr/bin/repoquery --quiet --pkgnarrow=repos --queryformat=%{version}|%{release}|%{arch}|%{repo}|%{version}-%{release} --show-duplicates atomic-openshift",
  248. # "package_found": true,
  249. # "package_name": "atomic-openshift",
  250. # "returncode": 0,
  251. # "versions": {
  252. # "available_versions": [
  253. # "3.2.0.43",
  254. # "3.2.1.23",
  255. # "3.3.0.32",
  256. # "3.3.0.34",
  257. # "3.3.0.35",
  258. # "3.3.1.3",
  259. # "3.3.1.4",
  260. # "3.3.1.5",
  261. # "3.3.1.7",
  262. # "3.4.0.39"
  263. # ],
  264. # "available_versions_full": [
  265. # "3.2.0.43-1.git.0.672599f.el7",
  266. # "3.2.1.23-1.git.0.88a7a1d.el7",
  267. # "3.3.0.32-1.git.0.37bd7ea.el7",
  268. # "3.3.0.34-1.git.0.83f306f.el7",
  269. # "3.3.0.35-1.git.0.d7bd9b6.el7",
  270. # "3.3.1.3-1.git.0.86dc49a.el7",
  271. # "3.3.1.4-1.git.0.7c8657c.el7",
  272. # "3.3.1.5-1.git.0.62700af.el7",
  273. # "3.3.1.7-1.git.0.0988966.el7",
  274. # "3.4.0.39-1.git.0.5f32f06.el7"
  275. # ],
  276. # "latest": "3.4.0.39",
  277. # "latest_full": "3.4.0.39-1.git.0.5f32f06.el7",
  278. # "matched_version_found": true,
  279. # "matched_version_full_latest": "3.3.1.7-1.git.0.0988966.el7",
  280. # "matched_version_latest": "3.3.1.7",
  281. # "matched_versions": [
  282. # "3.3.0.32",
  283. # "3.3.0.34",
  284. # "3.3.0.35",
  285. # "3.3.1.3",
  286. # "3.3.1.4",
  287. # "3.3.1.5",
  288. # "3.3.1.7"
  289. # ],
  290. # "matched_versions_full": [
  291. # "3.3.0.32-1.git.0.37bd7ea.el7",
  292. # "3.3.0.34-1.git.0.83f306f.el7",
  293. # "3.3.0.35-1.git.0.d7bd9b6.el7",
  294. # "3.3.1.3-1.git.0.86dc49a.el7",
  295. # "3.3.1.4-1.git.0.7c8657c.el7",
  296. # "3.3.1.5-1.git.0.62700af.el7",
  297. # "3.3.1.7-1.git.0.0988966.el7"
  298. # ],
  299. # "requested_match_version": "3.3"
  300. # }
  301. # },
  302. # "state": "present"
  303. # }
  304. # }
  305. '''
  306. # -*- -*- -*- End included fragment: doc/repoquery -*- -*- -*-
  307. # -*- -*- -*- Begin included fragment: lib/repoquery.py -*- -*- -*-
  308. '''
  309. class that wraps the repoquery commands in a subprocess
  310. '''
  311. # pylint: disable=too-many-lines,wrong-import-position,wrong-import-order
  312. from collections import defaultdict # noqa: E402
  313. # pylint: disable=no-name-in-module,import-error
  314. # Reason: pylint errors with "No name 'version' in module 'distutils'".
  315. # This is a bug: https://github.com/PyCQA/pylint/issues/73
  316. from distutils.version import LooseVersion # noqa: E402
  317. import subprocess # noqa: E402
  318. class RepoqueryCLIError(Exception):
  319. '''Exception class for repoquerycli'''
  320. pass
  321. def _run(cmds):
  322. ''' Actually executes the command. This makes mocking easier. '''
  323. proc = subprocess.Popen(cmds,
  324. stdin=subprocess.PIPE,
  325. stdout=subprocess.PIPE,
  326. stderr=subprocess.PIPE)
  327. stdout, stderr = proc.communicate()
  328. return proc.returncode, stdout, stderr
  329. # pylint: disable=too-few-public-methods
  330. class RepoqueryCLI(object):
  331. ''' Class to wrap the command line tools '''
  332. def __init__(self,
  333. verbose=False):
  334. ''' Constructor for RepoqueryCLI '''
  335. self.verbose = verbose
  336. self.verbose = True
  337. def _repoquery_cmd(self, cmd, output=False, output_type='json'):
  338. '''Base command for repoquery '''
  339. cmds = ['/usr/bin/repoquery', '--plugins', '--quiet']
  340. cmds.extend(cmd)
  341. rval = {}
  342. results = ''
  343. err = None
  344. if self.verbose:
  345. print(' '.join(cmds))
  346. returncode, stdout, stderr = _run(cmds)
  347. rval = {
  348. "returncode": returncode,
  349. "results": results,
  350. "cmd": ' '.join(cmds),
  351. }
  352. if returncode == 0:
  353. if output:
  354. if output_type == 'raw':
  355. rval['results'] = stdout
  356. if self.verbose:
  357. print(stdout)
  358. print(stderr)
  359. if err:
  360. rval.update({
  361. "err": err,
  362. "stderr": stderr,
  363. "stdout": stdout,
  364. "cmd": cmds
  365. })
  366. else:
  367. rval.update({
  368. "stderr": stderr,
  369. "stdout": stdout,
  370. "results": {},
  371. })
  372. return rval
  373. # -*- -*- -*- End included fragment: lib/repoquery.py -*- -*- -*-
  374. # -*- -*- -*- Begin included fragment: class/repoquery.py -*- -*- -*-
  375. class Repoquery(RepoqueryCLI):
  376. ''' Class to wrap the repoquery
  377. '''
  378. # pylint: disable=too-many-arguments,too-many-instance-attributes
  379. def __init__(self, name, query_type, show_duplicates,
  380. match_version, ignore_excluders, verbose):
  381. ''' Constructor for YumList '''
  382. super(Repoquery, self).__init__(None)
  383. self.name = name
  384. self.query_type = query_type
  385. self.show_duplicates = show_duplicates
  386. self.match_version = match_version
  387. self.ignore_excluders = ignore_excluders
  388. self.verbose = verbose
  389. if self.match_version:
  390. self.show_duplicates = True
  391. self.query_format = "%{version}|%{release}|%{arch}|%{repo}|%{version}-%{release}"
  392. self.tmp_file = None
  393. def build_cmd(self):
  394. ''' build the repoquery cmd options '''
  395. repo_cmd = []
  396. repo_cmd.append("--pkgnarrow=" + self.query_type)
  397. repo_cmd.append("--queryformat=" + self.query_format)
  398. if self.show_duplicates:
  399. repo_cmd.append('--show-duplicates')
  400. if self.ignore_excluders:
  401. repo_cmd.append('--config=' + self.tmp_file.name)
  402. repo_cmd.append(self.name)
  403. return repo_cmd
  404. @staticmethod
  405. def process_versions(query_output):
  406. ''' format the package data into something that can be presented '''
  407. version_dict = defaultdict(dict)
  408. for version in query_output.decode().split('\n'):
  409. pkg_info = version.split("|")
  410. pkg_version = {}
  411. pkg_version['version'] = pkg_info[0]
  412. pkg_version['release'] = pkg_info[1]
  413. pkg_version['arch'] = pkg_info[2]
  414. pkg_version['repo'] = pkg_info[3]
  415. pkg_version['version_release'] = pkg_info[4]
  416. version_dict[pkg_info[4]] = pkg_version
  417. return version_dict
  418. def format_versions(self, formatted_versions):
  419. ''' Gather and present the versions of each package '''
  420. versions_dict = {}
  421. versions_dict['available_versions_full'] = list(formatted_versions.keys())
  422. # set the match version, if called
  423. if self.match_version:
  424. versions_dict['matched_versions_full'] = []
  425. versions_dict['requested_match_version'] = self.match_version
  426. versions_dict['matched_versions'] = []
  427. # get the "full version (version - release)
  428. versions_dict['available_versions_full'].sort(key=LooseVersion)
  429. versions_dict['latest_full'] = versions_dict['available_versions_full'][-1]
  430. # get the "short version (version)
  431. versions_dict['available_versions'] = []
  432. for version in versions_dict['available_versions_full']:
  433. versions_dict['available_versions'].append(formatted_versions[version]['version'])
  434. if self.match_version:
  435. if version.startswith(self.match_version):
  436. versions_dict['matched_versions_full'].append(version)
  437. versions_dict['matched_versions'].append(formatted_versions[version]['version'])
  438. versions_dict['available_versions'].sort(key=LooseVersion)
  439. versions_dict['latest'] = versions_dict['available_versions'][-1]
  440. # finish up the matched version
  441. if self.match_version:
  442. if versions_dict['matched_versions_full']:
  443. versions_dict['matched_version_found'] = True
  444. versions_dict['matched_versions'].sort(key=LooseVersion)
  445. versions_dict['matched_version_latest'] = versions_dict['matched_versions'][-1]
  446. versions_dict['matched_version_full_latest'] = versions_dict['matched_versions_full'][-1]
  447. else:
  448. versions_dict['matched_version_found'] = False
  449. versions_dict['matched_versions'] = []
  450. versions_dict['matched_version_latest'] = ""
  451. versions_dict['matched_version_full_latest'] = ""
  452. return versions_dict
  453. def repoquery(self):
  454. '''perform a repoquery '''
  455. if self.ignore_excluders:
  456. # Duplicate yum.conf and reset exclude= line to an empty string
  457. # to clear a list of all excluded packages
  458. self.tmp_file = tempfile.NamedTemporaryFile()
  459. with open("/etc/yum.conf", "r") as file_handler:
  460. yum_conf_lines = file_handler.readlines()
  461. yum_conf_lines = ["exclude=" if l.startswith("exclude=") else l for l in yum_conf_lines]
  462. with open(self.tmp_file.name, "w") as file_handler:
  463. file_handler.writelines(yum_conf_lines)
  464. file_handler.flush()
  465. repoquery_cmd = self.build_cmd()
  466. rval = self._repoquery_cmd(repoquery_cmd, True, 'raw')
  467. # check to see if there are actual results
  468. if rval['results']:
  469. processed_versions = Repoquery.process_versions(rval['results'].strip())
  470. formatted_versions = self.format_versions(processed_versions)
  471. rval['package_found'] = True
  472. rval['versions'] = formatted_versions
  473. rval['package_name'] = self.name
  474. if self.verbose:
  475. rval['raw_versions'] = processed_versions
  476. else:
  477. del rval['results']
  478. # No packages found
  479. else:
  480. rval['package_found'] = False
  481. if self.ignore_excluders:
  482. self.tmp_file.close()
  483. return rval
  484. @staticmethod
  485. def run_ansible(params, check_mode):
  486. '''run the ansible idempotent code'''
  487. repoquery = Repoquery(
  488. params['name'],
  489. params['query_type'],
  490. params['show_duplicates'],
  491. params['match_version'],
  492. params['ignore_excluders'],
  493. params['verbose'],
  494. )
  495. state = params['state']
  496. if state == 'list':
  497. results = repoquery.repoquery()
  498. if results['returncode'] != 0:
  499. return {'failed': True,
  500. 'msg': results}
  501. return {'changed': False, 'results': results, 'state': 'list', 'check_mode': check_mode}
  502. return {'failed': True,
  503. 'changed': False,
  504. 'msg': 'Unknown state passed. %s' % state,
  505. 'state': 'unknown'}
  506. # -*- -*- -*- End included fragment: class/repoquery.py -*- -*- -*-
  507. # -*- -*- -*- Begin included fragment: ansible/repoquery.py -*- -*- -*-
  508. def main():
  509. '''
  510. ansible repoquery module
  511. '''
  512. module = AnsibleModule(
  513. argument_spec=dict(
  514. state=dict(default='list', type='str', choices=['list']),
  515. name=dict(default=None, required=True, type='str'),
  516. query_type=dict(default='repos', required=False, type='str',
  517. choices=[
  518. 'installed', 'available', 'recent',
  519. 'updates', 'extras', 'all', 'repos'
  520. ]),
  521. verbose=dict(default=False, required=False, type='bool'),
  522. show_duplicates=dict(default=False, required=False, type='bool'),
  523. match_version=dict(default=None, required=False, type='str'),
  524. ignore_excluders=dict(default=False, required=False, type='bool'),
  525. retries=dict(default=4, required=False, type='int'),
  526. retry_interval=dict(default=5, required=False, type='int'),
  527. ),
  528. supports_check_mode=False,
  529. required_if=[('show_duplicates', True, ['name'])],
  530. )
  531. tries = 1
  532. while True:
  533. rval = Repoquery.run_ansible(module.params, module.check_mode)
  534. if 'failed' not in rval:
  535. module.exit_json(**rval)
  536. elif tries > module.params['retries']:
  537. module.fail_json(**rval)
  538. tries += 1
  539. time.sleep(module.params['retry_interval'])
  540. if __name__ == "__main__":
  541. main()
  542. # -*- -*- -*- End included fragment: ansible/repoquery.py -*- -*- -*-