best_practices_guide.adoc 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538
  1. // vim: ft=asciidoc
  2. = openshift-ansible Best Practices Guide
  3. The purpose of this guide is to describe the preferred patterns and best practices used in this repository (both in ansible and python).
  4. It is important to note that this repository may not currently comply with all best practices, but the intention is that it will.
  5. All new pull requests created against this repository MUST comply with this guide.
  6. This guide complies with https://www.ietf.org/rfc/rfc2119.txt[RFC2119].
  7. == Pull Requests
  8. [[All-pull-requests-MUST-pass-the-build-bot-before-they-are-merged]]
  9. [cols="2v,v"]
  10. |===
  11. | <<All-pull-requests-MUST-pass-the-build-bot-before-they-are-merged, Rule>>
  12. | All pull requests MUST pass the build bot *before* they are merged.
  13. |===
  14. The purpose of this rule is to avoid cases where the build bot will fail pull requests for code modified in a previous pull request.
  15. The tooling is flexible enough that exceptions can be made so that the tool the build bot is running will ignore certain areas or certain checks, but the build bot itself must pass for the pull request to be merged.
  16. == Python
  17. === Python Source Files
  18. '''
  19. [[Python-source-files-MUST-contain-the-following-vim-mode-line]]
  20. [cols="2v,v"]
  21. |===
  22. | <<Python-source-files-MUST-contain-the-following-vim-mode-line, Rule>>
  23. | Python source files MUST contain the following vim mode line.
  24. |===
  25. [source]
  26. ----
  27. # vim: expandtab:tabstop=4:shiftwidth=4
  28. ----
  29. Since most developers contributing to this repository use vim, this rule helps to promote consistency.
  30. If mode lines for other editors are needed, please open a GitHub issue.
  31. === Method Signatures
  32. '''
  33. [[When-adding-a-new-paramemter-to-an-existing-method-a-default-value-SHOULD-be-used]]
  34. [cols="2v,v"]
  35. |===
  36. | <<When-adding-a-new-paramemter-to-an-existing-method-a-default-value-SHOULD-be-used, Rule>>
  37. | When adding a new paramemter to an existing method, a default value SHOULD be used
  38. |===
  39. The purpose of this rule is to make it so that method signatures are backwards compatible.
  40. If this rule isn't followed, it will be necessary for the person who changed the method to search out all callers and make sure that they're able to use the new method signature.
  41. .Before:
  42. [source,python]
  43. ----
  44. def add_person(first_name, last_name):
  45. ----
  46. .After:
  47. [source,python]
  48. ----
  49. def add_person(first_name, last_name, age=None):
  50. ----
  51. === PyLint
  52. http://www.pylint.org/[PyLint] is used in an attempt to keep the python code as clean and as manageable as possible. The build bot runs each pull request through PyLint and any warnings or errors cause the build bot to fail the pull request.
  53. '''
  54. [[PyLint-rules-MUST-NOT-be-disabled-on-a-whole-file]]
  55. [cols="2v,v"]
  56. |===
  57. | <<PyLint-rules-MUST-NOT-be-disabled-on-a-whole-file, Rule>>
  58. | PyLint rules MUST NOT be disabled on a whole file.
  59. |===
  60. Instead, http://docs.pylint.org/faq.html#is-it-possible-to-locally-disable-a-particular-message[disable the PyLint check on the line where PyLint is complaining].
  61. '''
  62. [[PyLint-rules-MUST-NOT-be-disabled-unless-they-meet-one-of-the-following-exceptions]]
  63. [cols="2v,v"]
  64. |===
  65. | <<PyLint-rules-MUST-NOT-be-disabled-unless-they-meet-one-of-the-following-exceptions, Rule>>
  66. | PyLint rules MUST NOT be disabled unless they meet one of the following exceptions
  67. |===
  68. .Exceptions:
  69. 1. When PyLint fails because of a dependency that can't be installed on the build bot
  70. 1. When PyLint fails because of including a module that is outside of control (like Ansible)
  71. 1. When PyLint fails, but the code makes more sense the way it is formatted (stylistic exception). For this exception, the description of the PyLint disable MUST state why the code is more clear, AND the person reviewing the PR will decide if they agree or not. The reviewer may reject the PR if they disagree with the reason for the disable.
  72. '''
  73. [[All-PyLint-rule-disables-MUST-be-documented-in-the-code]]
  74. [cols="2v,v"]
  75. |===
  76. | <<All-PyLint-rule-disables-MUST-be-documented-in-the-code, Rule>>
  77. | All PyLint rule disables MUST be documented in the code.
  78. |===
  79. The purpose of this rule is to inform future developers about the disable.
  80. .Specifically, the following MUST accompany every PyLint disable:
  81. 1. Why is the check being disabled?
  82. 1. Is disabling this check meant to be permanent or temporary?
  83. .Example:
  84. [source,python]
  85. ----
  86. # Reason: disable pylint maybe-no-member because overloaded use of
  87. # the module name causes pylint to not detect that 'results'
  88. # is an array or hash
  89. # Status: permanently disabled unless a way is found to fix this.
  90. # pylint: disable=maybe-no-member
  91. metadata[line] = results.pop()
  92. ----
  93. == Ansible
  94. === Yaml Files (Playbooks, Roles, Vars, etc)
  95. '''
  96. [[Ansible-files-SHOULD-NOT-use-JSON-use-pure-YAML-instead]]
  97. [cols="2v,v"]
  98. |===
  99. | <<Ansible-files-SHOULD-NOT-use-JSON-use-pure-YAML-instead, Rule>>
  100. | Ansible files SHOULD NOT use JSON (use pure YAML instead).
  101. |===
  102. YAML is a superset of JSON, which means that Ansible allows JSON syntax to be interspersed. Even though YAML (and by extension Ansible) allows for this, JSON SHOULD NOT be used.
  103. .Reasons:
  104. * Ansible is able to give clearer error messages when the files are pure YAML
  105. * YAML reads nicer (preference held by several team members)
  106. * YAML makes for nicer diffs as YAML tends to be multi-line, whereas JSON tends to be more concise
  107. .Exceptions:
  108. * Ansible static inventory files are INI files. To pass in variables for specific hosts, Ansible allows for these variables to be put inside of the static inventory files. These variables can be in JSON format, but can't be in YAML format. This is an acceptable use of JSON, as YAML is not allowed in this case.
  109. Every effort should be made to keep our Ansible YAML files in pure YAML.
  110. === Modules
  111. '''
  112. [[Custom-Ansible-modules-SHOULD-be-embedded-in-a-role]]
  113. [cols="2v,v"]
  114. |===
  115. | <<Custom-Ansible-modules-SHOULD-be-embedded-in-a-role, Rule>>
  116. | Custom Ansible modules SHOULD be embedded in a role.
  117. |===
  118. .Context
  119. * http://docs.ansible.com/ansible/playbooks_roles.html#embedding-modules-in-roles[Ansible doc on how to embed modules in roles]
  120. The purpose of this rule is to make it easy to include custom modules in our playbooks and share them on Ansible Galaxy.
  121. .Custom module `openshift_facts.py` is embedded in the `openshift_facts` role.
  122. ----
  123. > ll openshift-ansible/roles/openshift_facts/library/
  124. -rwxrwxr-x. 1 user group 33616 Jul 22 09:36 openshift_facts.py
  125. ----
  126. .Custom module `openshift_facts` can be used after `openshift_facts` role has been referenced.
  127. [source,yaml]
  128. ----
  129. - hosts: openshift_hosts
  130. gather_facts: no
  131. roles:
  132. - role: openshift_facts
  133. post_tasks:
  134. - openshift_facts
  135. role: common
  136. hostname: host
  137. public_hostname: host.example.com
  138. ----
  139. '''
  140. [[Parameters-to-Ansible-modules-SHOULD-use-the-Yaml-dictionary-format-when-3-or-more-parameters-are-being-passed]]
  141. [cols="2v,v"]
  142. |===
  143. | <<Parameters-to-Ansible-modules-SHOULD-use-the-Yaml-dictionary-format-when-3-or-more-parameters-are-being-passed, Rule>>
  144. | Parameters to Ansible modules SHOULD use the Yaml dictionary format when 3 or more parameters are being passed
  145. |===
  146. When a module has several parameters that are being passed in, it's hard to see exactly what value each parameter is getting. It is preferred to use the Ansible Yaml syntax to pass in parameters so that it's more clear what values are being passed for each paramemter.
  147. .Bad:
  148. [source,yaml]
  149. ----
  150. - file: src=/file/to/link/to dest=/path/to/symlink owner=foo group=foo state=link
  151. ----
  152. .Good:
  153. [source,yaml]
  154. ----
  155. - file:
  156. src: /file/to/link/to
  157. dest: /path/to/symlink
  158. owner: foo
  159. group: foo
  160. state: link
  161. ----
  162. '''
  163. [[Parameters-to-Ansible-modules-SHOULD-use-the-Yaml-dictionary-format-when-the-line-length-exceeds-120-characters]]
  164. [cols="2v,v"]
  165. |===
  166. | <<Parameters-to-Ansible-modules-SHOULD-use-the-Yaml-dictionary-format-when-the-line-length-exceeds-120-characters, Rule>>
  167. | Parameters to Ansible modules SHOULD use the Yaml dictionary format when the line length exceeds 120 characters
  168. |===
  169. Lines that are long quickly become a wall of text that isn't easily parsable. It is preferred to use the Ansible Yaml syntax to pass in parameters so that it's more clear what values are being passed for each paramemter.
  170. .Bad:
  171. [source,yaml]
  172. ----
  173. - get_url: url=http://example.com/path/file.conf dest=/etc/foo.conf sha256sum=b5bb9d8014a0f9b1d61e21e796d78dccdf1352f23cd32812f4850b878ae4944c
  174. ----
  175. .Good:
  176. [source,yaml]
  177. ----
  178. - get_url:
  179. url: http://example.com/path/file.conf
  180. dest: /etc/foo.conf
  181. sha256sum: b5bb9d8014a0f9b1d61e21e796d78dccdf1352f23cd32812f4850b878ae4944c
  182. ----
  183. '''
  184. [[The-Ansible-command-module-SHOULD-be-used-instead-of-the-Ansible-shell-module]]
  185. [cols="2v,v"]
  186. |===
  187. | <<The-Ansible-command-module-SHOULD-be-used-instead-of-the-Ansible-shell-module, Rule>>
  188. | The Ansible `command` module SHOULD be used instead of the Ansible `shell` module.
  189. |===
  190. .Context
  191. * http://docs.ansible.com/shell_module.html#notes[Ansible doc on why using the command module is a best practice]
  192. The Ansible `shell` module can run most commands that can be run from a bash CLI. This makes it extremely powerful, but it also opens our playbooks up to being exploited by attackers.
  193. .Bad:
  194. [source,yaml]
  195. ----
  196. - shell: "/bin/echo {{ cli_var }}"
  197. ----
  198. .Better:
  199. [source,yaml]
  200. ----
  201. - command: "/bin/echo {{ cli_var }}"
  202. ----
  203. '''
  204. [[The-Ansible-quote-filter-MUST-be-used-with-any-variable-passed-into-the-shell-module]]
  205. [cols="2v,v"]
  206. |===
  207. | <<The-Ansible-quote-filter-MUST-be-used-with-any-variable-passed-into-the-shell-module, Rule>>
  208. | The Ansible `quote` filter MUST be used with any variable passed into the shell module.
  209. |===
  210. .Context
  211. * http://docs.ansible.com/shell_module.html#notes[Ansible doc describing why to use the quote filter]
  212. It is recommended not to use the `shell` module. However, if it absolutely must be used, all variables passed into the `shell` module MUST use the `quote` filter to ensure they are shell safe.
  213. .Bad:
  214. [source,yaml]
  215. ----
  216. - shell: "/bin/echo {{ cli_var }}"
  217. ----
  218. .Good:
  219. [source,yaml]
  220. ----
  221. - shell: "/bin/echo {{ cli_var | quote }}"
  222. ----
  223. === Defensive Programming
  224. .Context
  225. * http://docs.ansible.com/fail_module.html[Ansible Fail Module]
  226. '''
  227. [[Ansible-playbooks-MUST-begin-with-checks-for-any-variables-that-they-require]]
  228. [cols="2v,v"]
  229. |===
  230. | <<Ansible-playbooks-MUST-begin-with-checks-for-any-variables-that-they-require, Rule>>
  231. | Ansible playbooks MUST begin with checks for any variables that they require.
  232. |===
  233. If an Ansible playbook requires certain variables to be set, it's best to check for these up front before any other actions have been performed. In this way, the user knows exactly what needs to be passed into the playbook.
  234. .Example:
  235. [source,yaml]
  236. ----
  237. ---
  238. - hosts: localhost
  239. gather_facts: no
  240. tasks:
  241. - fail: msg="This playbook requires g_environment to be set and non empty"
  242. when: g_environment is not defined or g_environment == ''
  243. ----
  244. '''
  245. [[Ansible-roles-tasks-main-yml-file-MUST-begin-with-checks-for-any-variables-that-they-require]]
  246. [cols="2v,v"]
  247. |===
  248. | <<Ansible-roles-tasks-main-yml-file-MUST-begin-with-checks-for-any-variables-that-they-require, Rule>>
  249. | Ansible roles tasks/main.yml file MUST begin with checks for any variables that they require.
  250. |===
  251. If an Ansible role requires certain variables to be set, it's best to check for these up front before any other actions have been performed. In this way, the user knows exactly what needs to be passed into the role.
  252. .Example:
  253. [source,yaml]
  254. ----
  255. ---
  256. # tasks/main.yml
  257. - fail: msg="This role requires arl_environment to be set and non empty"
  258. when: arl_environment is not defined or arl_environment == ''
  259. ----
  260. === Tasks
  261. '''
  262. [[Ansible-tasks-SHOULD-NOT-be-used-in-ansible-playbooks-Instead-use-pre_tasks-and-post_tasks]]
  263. [cols="2v,v"]
  264. |===
  265. | <<Ansible-tasks-SHOULD-NOT-be-used-in-ansible-playbooks-Instead-use-pre_tasks-and-post_tasks, Rule>>
  266. | Ansible tasks SHOULD NOT be used in ansible playbooks. Instead, use pre_tasks and post_tasks.
  267. |===
  268. An Ansible play is defined as a Yaml dictionary. Because of that, ansible doesn't know if the play's tasks list or roles list was specified first. Therefore Ansible always runs tasks after roles.
  269. This can be quite confusing if the tasks list is defined in the playbook before the roles list because people assume in order execution in Ansible.
  270. Therefore, we SHOULD use pre_tasks and post_tasks to make it more clear when the tasks will be run.
  271. .Context
  272. * https://docs.ansible.com/playbooks_roles.html[Ansible documentation on pre_tasks and post_tasks]
  273. .Bad:
  274. [source,yaml]
  275. ----
  276. ---
  277. # playbook.yml
  278. - hosts: localhost
  279. gather_facts: no
  280. tasks:
  281. - name: This will execute AFTER the example_role, so it's confusing
  282. debug: msg="in tasks list"
  283. roles:
  284. - role: example_role
  285. # roles/example_role/tasks/main.yml
  286. - debug: msg="in example_role"
  287. ----
  288. .Good:
  289. [source,yaml]
  290. ----
  291. ---
  292. # playbook.yml
  293. - hosts: localhost
  294. gather_facts: no
  295. pre_tasks:
  296. - name: This will execute BEFORE the example_role, so it makes sense
  297. debug: msg="in pre_tasks list"
  298. roles:
  299. - role: example_role
  300. # roles/example_role/tasks/main.yml
  301. - debug: msg="in example_role"
  302. ----
  303. === Roles
  304. '''
  305. [[All-tasks-in-a-role-SHOULD-be-tagged-with-the-role-name]]
  306. [cols="2v,v"]
  307. |===
  308. | <<All-tasks-in-a-role-SHOULD-be-tagged-with-the-role-name, Rule>>
  309. | All tasks in a role SHOULD be tagged with the role name.
  310. |===
  311. .Context
  312. * http://docs.ansible.com/playbooks_tags.html[Ansible doc explaining tags]
  313. Ansible tasks can be tagged, and then these tags can be used to either _run_ or _skip_ the tagged tasks using the `--tags` and `--skip-tags` ansible-playbook options respectively.
  314. This is very useful when developing and debugging new tasks. It can also significantly speed up playbook runs if the user specifies only the roles that changed.
  315. .Example:
  316. [source,yaml]
  317. ----
  318. ---
  319. # roles/example_role/tasks/main.yml
  320. - debug: msg="in example_role"
  321. tags:
  322. - example_role
  323. ----
  324. '''
  325. [[The-Ansible-roles-directory-MUST-maintain-a-flat-structure]]
  326. [cols="2v,v"]
  327. |===
  328. | <<The-Ansible-roles-directory-MUST-maintain-a-flat-structure, Rule>>
  329. | The Ansible roles directory MUST maintain a flat structure.
  330. |===
  331. .Context
  332. * http://docs.ansible.com/playbooks_best_practices.html#directory-layout[Ansible Suggested Directory Layout]
  333. .The purpose of this rule is to:
  334. * Comply with the upstream best practices
  335. * Make it familiar for new contributors
  336. * Make it compatible with Ansible Galaxy
  337. '''
  338. [[Ansible-Roles-SHOULD-be-named-like-technology_component_subcomponent]]
  339. [cols="2v,v"]
  340. |===
  341. | [[Ansible-Roles-SHOULD-be-named-like-technology_component_subcomponent, Rule]]
  342. | Ansible Roles SHOULD be named like technology_component[_subcomponent].
  343. |===
  344. For consistency, role names SHOULD follow the above naming pattern. It is important to note that this is a recommendation for role naming, and follows the pattern used by upstream.
  345. Many times the `technology` portion of the pattern will line up with a package name. It is advised that whenever possible, the package name should be used.
  346. .Examples:
  347. * The role to configure a master is called `openshift_master`
  348. * The role to configure OpenShift specific yum repositories is called `openshift_repos`
  349. === Filters
  350. .Context:
  351. * https://docs.ansible.com/playbooks_filters.html[Ansible Playbook Filters]
  352. * http://jinja.pocoo.org/docs/dev/templates/#builtin-filters[Jinja2 Builtin Filters]
  353. '''
  354. [[The-default-filter-SHOULD-replace-empty-strings-lists-etc]]
  355. [cols="2v,v"]
  356. |===
  357. | <<The-default-filter-SHOULD-replace-empty-strings-lists-etc, Rule>>
  358. | The `default` filter SHOULD replace empty strings, lists, etc.
  359. |===
  360. When using the jinja2 `default` filter, unless the variable is a boolean, specify `true` as the second parameter. This will cause the default filter to replace empty strings, lists, etc with the provided default.
  361. This is because it is preferable to either have a sane default set than to have an empty string, list, etc. For example, it is preferable to have a config value set to a sane default than to have it simply set as an empty string.
  362. .From the http://jinja.pocoo.org/docs/dev/templates/[Jinja2 Docs]:
  363. [quote]
  364. If you want to use default with variables that evaluate to false you have to set the second parameter to true
  365. .Example:
  366. [source,yaml]
  367. ----
  368. ---
  369. - hosts: localhost
  370. gather_facts: no
  371. vars:
  372. somevar: ''
  373. tasks:
  374. - debug: var=somevar
  375. - name: "Will output 'somevar: []'"
  376. debug: "msg='somevar: [{{ somevar | default('the string was empty') }}]'"
  377. - name: "Will output 'somevar: [the string was empty]'"
  378. debug: "msg='somevar: [{{ somevar | default('the string was empty', true) }}]'"
  379. ----
  380. In other words, normally the `default` filter will only replace the value if it's undefined. By setting the second parameter to `true`, it will also replace the value if it defaults to a false value in python, so None, empty list, empty string, etc.
  381. This is almost always more desirable than an empty list, string, etc.
  382. === Yum and DNF
  383. '''
  384. [[Package-installation-MUST-use-ansible-action-module-to-abstract-away-dnf-yum]]
  385. [cols="2v,v"]
  386. |===
  387. | <<Package-installation-MUST-use-ansible-action-module-to-abstract-away-dnf-yum, Rule>>
  388. | Package installation MUST use ansible action module to abstract away dnf/yum.
  389. |===
  390. [[Package-installation-MUST-use-name-and-state-present-rather-than-pkg-and-state-installed-respectively]]
  391. [cols="2v,v"]
  392. |===
  393. | <<Package-installation-MUST-use-name-and-state-present-rather-than-pkg-and-state-installed-respectively, Rule>>
  394. | Package installation MUST use name= and state=present rather than pkg= and state=installed respectively.
  395. |===
  396. This is done primarily because if you're registering the result of the
  397. installation and you have two conditional tasks based on whether or not yum or
  398. dnf are in use you'll end up inadvertently overwriting the value. It also
  399. reduces duplication. name= and state=present are common between dnf and yum
  400. modules.
  401. .Bad:
  402. [source,yaml]
  403. ----
  404. ---
  405. # tasks.yml
  406. - name: Install etcd (for etcdctl)
  407. yum: name=etcd state=latest"
  408. when: "ansible_pkg_mgr == yum"
  409. register: install_result
  410. - name: Install etcd (for etcdctl)
  411. dnf: name=etcd state=latest"
  412. when: "ansible_pkg_mgr == dnf"
  413. register: install_result
  414. ----
  415. .Good:
  416. [source,yaml]
  417. ----
  418. ---
  419. # tasks.yml
  420. - name: Install etcd (for etcdctl)
  421. action: "{{ ansible_pkg_mgr }} name=etcd state=latest"
  422. register: install_result
  423. ----