xref: /openbmc/openbmc-test-automation/CONTRIBUTING.md (revision ef00306e35d5ad131753a823d41e19960abcc66b)
1Contributing to OpenBMC Test Automation
2=======================================
3Guide to working on OpenBMC test automation. This document will always be a
4work-in-progress, feel free to propose changes.
5
6Submitting changes via Gerrit server
7------------------------------------
8-   Reference [OpenBMC CLA signers](https://github.com/openbmc/openbmc-tools/blob/master/emilyshaffer/cla-signers/cla-signers)
9-   Reference [OpenBMC docs](https://github.com/openbmc/docs/blob/master/CONTRIBUTING.md#submitting-changes-via-gerrit-server)
10
11Links
12-----
13
14- [Redfish Coding Guidelines](docs/redfish_coding_guidelines.md)
15
16Robot Coding Guidelines
17-----------------------
18-   For this project, we will write Robot keyword definitions in either Robot
19    or Python.  Robot code should be quite simple.  Therefore, if the algorithm
20    in question is the least bit complex, please write it in Python.
21
22    See the following for support on writing most keywords in python.
23      - [Quit Writing Ugly Robot Code](https://esalagea.wordpress.com/2014/11/24/robot-framework-quit-writing-ugly-robot-code-just-write-proper-python/)
24      - [Robot Dos and Don'ts](https://www.slideshare.net/pekkaklarck/robot-framework-dos-and-donts)
25
26-   Observe a maximum line length of 110 characters.
27-   Avoid trailing space at the end of any line of Robot code.
28-   Avoid the use of tabs.
29-   Robot supports delimiting cells with either two or more spaces or with a
30    pipe symbol (e.g. "\|"). Our team has chosen to use spaces rather than the
31    pipe character. Make sure all space delimiters in Robot code are the
32    **minimum** of two spaces. There may be some exceptions to this rule.
33
34    Exceptions to two-space delimiter rule:
35    - When you wish to line up resource, library or variable values:
36      ```
37      Library         Lib1
38      Resource        Resource1
39      *** Variables ***
40      ${var1}         ${EMPTY}
41      ```
42    - When you wish to line up fields for test templates:
43      ```
44      [Template]  Set System LED State
45      # LED Name  LED State
46      power       On
47      power       Off
48      ```
49    - When you wish to indent if/else or loop bodies for visual effect:
50      ```
51      Run Keyword If  '${this}' == '${that}'
52      ...    Log  Bla, bla...
53      ...  ELSE
54      ...    Run Keywords  Key1  parms
55      ...    AND  Key2  parms
56      ```
57-   Use single spaces to make conditions more readable:
58
59    Correct example:
60    ```
61    Run Keyword If  '${var1}' == '${0}'  My Keyword
62    ```
63    Incorrect example:
64    ```
65    Run Keyword If  '${var1}'=='${0}'  My Keyword
66    ```
67-   When you define or call a Robot keyword, Robot pays no attention to spaces,
68    underscores or case.  However, our team will observe the following
69    conventions in both our definitions and our calls:
70    - Separate words with single spaces.
71    - Capitalize the first character of each word.
72    - Capitalize all characters in any word that is an acronym (e.g. JSON, BMC,
73      etc).
74
75    Examples:
76    ```
77    *** Keywords ***
78
79    This Is Correct
80
81        # This keyword name is correct.
82
83    this_is_incorrect
84
85        # This keyword name is incorrect because of 1) the
86        # underscores instead of spaces and 2) the failure to
87        # capitalize each word in the keyword.
88
89    soisthis
90
91        # This keyword name is incorrect because of 1) a failure to
92        # separate words with single spaces and 2) a failure to capitalize
93        # each word in the keyword.
94
95    BMC Is An Acronym
96
97        # This keyword name is correct.  Note that "BMC" is an
98        # acronym and as such is entirely uppercase.
99    ```
100-   Documentation strings:
101    -  Each documentation string should be phrased as an **English command**.
102       Punctuate it correctly with the first word capitalized and a period at
103       the end.
104
105       Correct example:
106        ```
107        Boot BMC
108            [Documentation]  Boot the BMC.
109        ```
110        Incorrect example:
111        ```
112        Boot BMC
113            [Documentation]  This keyword boots the BMC.
114
115            # The doc string above is not phrased as a command.
116        ```
117    -   Doc strings should be just one terse, descriptive sentence.
118        Remember that this doc string shows up in the HTML log file.  Put
119        additional commentary below in standard comment lines.
120
121        Correct example:
122        ```
123        Stop SOL Console Logging
124
125            [Documentation]  Stop system console logging and return log output.
126        ```
127        Incorrect example:
128        ```
129        Stop SOL Console Logging
130
131            [Documentation]  Stop system console logging.  If there are multiple
132            ...              system console processes, they will all be
133            ...              stopped.  If there is no existing log file this
134            ...              keyword will return an error message to that
135            ...              effect (and write that message to targ_file_path,
136            ...              if specified).  NOTE: This keyword will not fail
137            ...              if there is no running system console process.
138
139            # This doc string is way too long.
140        ```
141-   Tags:
142    -   Create a tag for every test case with a tag name that mirrors the test case
143        name as follows:
144        ```
145        Create Intermediate File
146
147            [Tags]  Create_Intermediate_File
148        ```
149-   Description of argument(s):
150    -   As shown in the following example, if your keyword has any arguments, include
151        a "**Description of argument(s)**" section.  This effectively serves as the
152        help text for anyone wanting to use or understand your keyword.  Include
153        real data examples wherever possible and applicable.  Leave at least 2 spaces
154        between the argument name and the description.  Align all description text as
155        shown in the example below.
156
157        Example:
158        ```
159        Get URL List
160            [Documentation]  Return list of URLs under given URL.
161            [Arguments]  ${openbmc_url}  ${policy}   ${verify}
162
163            # Description of argument(s):
164            # openbmc_url  URL for list operation (e.g.
165            #              "/xyz/openbmc_project/inventory").
166            # policy       Power restore policy (e.g "RESTORE_LAST_STATE",
167            #              ${RESTORE_LAST_STATE}, etc.).
168            # verify       Verify the results (${TRUE}/${FALSE}).
169        ```
170        Additional rules for example text in descriptions:
171        - Put parentheses around your examples.  Leave one space prior to the
172          left parenthesis.
173        - Use "e.g." (which effectively means "for example") to set introduce
174          your examples.
175        - Quote string examples.
176        - Include ", etc." after multiple examples.
177        - For cases where you're providing an complete list of possible values
178          (${TRUE}/${FALSE} or "PASS"/"FAIL"), do NOT use "e.g." and do NOT
179          use "etc.".  Separate such values with a slash.
180-   Variable assignments:
181
182    When assigning a variable as output from a keyword, do not precede the
183    equal sign with a space.
184
185    Correct examples:
186    ```
187    ${var1}=  Set Variable  ${1}
188    ${var1}=  My Keyword
189    ```
190    Incorrect examples:
191
192    ```
193    ${var1} =  Set Variable  ${1}
194    ${var1} =  My Keyword
195    ```
196-   General variable naming conventions:
197    -   Variable names should be lower case with few exceptions (listed here):
198        -   Environment variables should be all upper case.
199        -   Variables intended to be set by Robot -v parameters may be all
200            upper case.
201    -   Words within a variable name should be separated by underscores:
202
203        Correct examples:
204        ```
205        ${host_name}
206        ${program_pid}
207        ```
208        Incorrect examples:
209        ```
210        ${HostName}
211        ${ProgramPid}
212        ```
213    -   Use good judgement in choosing variable names that are neither way too
214        short nor way too long.
215
216        Correct examples:
217        ```
218        ${host_name}
219        ${program_pid}
220        ```
221        Incorrect examples:
222        ```
223        ${HostName}
224        ${ProgramPid}
225        ```
226    -   Avoid having the variable's type as a suffix portion of the name:
227
228        Incorrect examples:
229        ```
230        ${inventory_dict}
231        ${led_list}
232        ```
233
234        Incorrect examples:
235        ```
236        ${inventory}
237        # Use plural name to indicate that it is a list.
238        ${leds}
239        ```
240
241        A possible exception to this rule is when your keyword or function has
242        an ongoing need to refer to one piece of data in more than one format
243        (e.g. date_str, date_epoch, etc.).
244
245    -   Consistency of variable names:
246
247        Try to avoid referring to the same data entity by multiple different
248        names.  It creates a lot of confusion for the reader.
249
250        Incorrect example:
251        ```
252        # Why call the receiving variable rc (return code) when the keyword is
253        # supposed to return status.
254        ${rc}=  Run Keyword And Return Status  Bla Bla
255        ```
256
257        Correct example:
258        ```
259        ${status}=  Run Keyword And Return Status  Bla Bla
260        ```
261
262-   Special variable naming conventions.
263
264    For certain very commonly used kinds of variables, please observe these
265    conventions in order to achieve consistency throughout the code.
266
267    -   hosts
268
269        When a variable is intended to contain **either** an IP address **or**
270        a host name (whether long or short), please give it a suffix of "_host".
271
272        Examples:
273        ```
274        openbmc_host
275        os_host
276        pdu_host
277        openbmc_serial_host
278        ```
279    -   host names
280
281        For host names (long or short, e.g. "bmc1" or "bmc1.example.com"), use
282        a suffix of _host_name.
283
284        Examples:
285        ```
286        openbmc_host_name
287        os_host_name
288        pdu_host_name
289        openbmc_serial_host_name
290        ```
291    -   Short host names
292
293        For short host names (e.g. "bmc1"), use a suffix of _host_short_name.
294
295        Examples:
296        ```
297        openbmc_host_short_name
298        os_host_short_name
299        pdu_host_short_name
300        openbmc_serial_host_short_name
301        ```
302    -   IP addresses
303
304        For IP addresses, use a suffix of _ip.
305
306        Example:
307        ```
308        openbmc_ip
309        os_ip
310        pdu_ip
311        openbmc_serial_ip
312        ```
313    -   Files and directories:
314        -   Files:
315            -   If your variable is to contain only the file's name, use a suffix
316                of _file_name.
317
318                Examples:
319                ```
320                ffdc_file_name = "bmc1.170428.120200.ffdc"
321                ```
322            -   If your variable is to contain the path to a file, use a suffix of
323                _file_path.  Bear in mind that a file path can be relative or
324                absolute, so that should not be a consideration in whether to use
325                the "_file_path" suffix.
326
327                Examples:
328                ```
329                status_file_path = "bmc1.170428.120200.status"
330                status_file_path = "subdir/bmc1.170428.120200.status"
331                status_file_path = "./bmc1.170428.120200.status"
332                status_file_path = "../bmc1.170428.120200.status"
333                status_file_path = "/home/user1/status/bmc1.170428.120200.status"
334                ```
335                To re-iterate, it doesn't matter whether the contents of the
336                variable are a relative or absolute path (as shown in the
337                examples above).  A file path is simply a value with enough
338                information in it for the program to find the file.
339
340            -   If the variable **must** contain an absolute path (which should be
341                the rare case), use a suffix _abs_file_path.
342
343        -   Directories:
344            -   Directory variables should follow the same conventions as file
345                variables.
346
347            -   If your variable is to contain only the directory's name, use a
348                suffix of _dir_name.
349
350                Example:
351                ```
352                ffdc_dir_name = "ffdc"
353                ```
354            -   If your variable is to contain the path to a directory, use a
355                suffix of _dir_path.  Bear in mind that a dir path can be
356                relative or absolute, so that should not be a consideration in
357                whether to use _dir_path.
358
359                Examples:
360                ```
361                status_dir_path = "status/"
362                status_dir_path = "subdir/status"
363                status_dir_path = "./status/"
364                status_dir_path = "../status/"
365                status_dir_path = "/home/user1/status/"
366                ```
367                To re-iterate, it doesn't matter whether the contents of
368                the variable are a relative or absolute path (as shown in
369                the examples above).  A dir path is simply a value with
370                enough information in it for the program to find the
371                directory.
372
373            -   If the variable **must** contain an absolute path (which
374                should be the rare case), use a suffix _abs_dir_path.
375            -   IMPORTANT:  As a programming convention, do pre-
376                processing on all dir_path variables to ensure that they
377                contain a trailing slash.  If we follow that convention
378                religiously, then when changes are made in other parts of
379                the program, the programmer can count on the value having
380                a trailing slash.  Therefore, they can safely do this kind
381                of thing:
382                ```
383                my_file_path = my_dir_path + my_file_name
384                ```
385    -   Setup/Teardown keywords
386
387        Use standardized names for setup and teardown keywords:
388        - Suite Setup Execution
389        - Suite Teardown Execution
390        - Test Setup Execution
391        - Test Teardown Execution
392-   Traditional comments (i.e. using the hashtag style comments)
393    -   Please leave one space following the hashtag.
394        ```
395        #wrong
396
397        # Right
398        ```
399    -   Please use proper English punctuation:
400        -   Capitalize the first word in the sentence or phrase.
401        -   End sentences (or stand-alone phrases) with a period.
402
403    -   Do not keep commented-out code in your program.  Instead, remove it
404        entirely.
405-   Robot Template Test Cases
406    -   Follow this format for Robot template test cases:
407
408        Note: Documentation, Tags and Template lines are all required and should be coded in the order shown.
409        ```
410        Test Case Name
411            [Documentation]
412            [Tags]
413            [Template]
414            # arg1  arg2  etc.
415            <arg1>  <arg2>
416
417        Example:
418
419        Get Response Codes
420            [Documentation]  REST "Get" response status test.
421            [Tags]  Get_Response_Codes
422            [Template]  Execute Get And Check Response
423
424            # expected_response_code  url_path
425            ${HTTP_OK}                /org/
426            ${HTTP_OK}                /xyz/
427            ${HTTP_OK}                /xyz/openbmc_project/
428            ${HTTP_OK}                /xyz/openbmc_project/state/enumerate
429            ${HTTP_NOT_FOUND}         /xyz/i/dont/exist/
430        ```
431
432        Note: Normally, a template test case would have many rows of data arguments as in the example above.
433              However, contributors frequently define multiple template test cases that each have only
434              one row of data which may seem to defeat the value of using templates in the first place. However,
435              it is done for these reasons:
436              1) Template tests are counted as a single test. The user may wish to have separate results for
437                 each call to the template function.
438              2) If any call to the template function fails, one would like FFDC data collected immediately
439                 and would like one set of FFDC data for EACH such failure.
440
441
442Python Coding Guidelines
443-----------------------
444-   The minimum required Python version is 2.7.x.  In the very near future, we
445    will stop supporting python 2 and will require python 3.
446-   Run pycodestyle on all Python files and correct errors to follow the
447    guidelines in https://www.python.org/dev/peps/pep-0008/.  Note that when
448    python code is checked into gerrit, pycodestyle is run automatically on it.
449
450    Example as run from a Linux command line:
451    ```
452    pycodestyle my_pgm.py
453
454    my_pgm.py:41:1: E302 expected 2 blank lines, found 1
455    my_pgm.py:58:52: W291 trailing whitespace
456    ```
457-   Include doc strings in every function and follow the guidelines in
458    https://www.python.org/dev/peps/pep-0257/.
459
460    Example:
461    ```
462        r"""
463        Return the function name associated with the indicated stack frame.
464
465        Description of argument(s):
466        stack_frame_ix                  The index of the stack frame whose
467                                        function name should be returned.  If
468                                        the caller does not specify a value,
469                                        this function will set the value to 1
470                                        which is the index of the caller's
471                                        stack frame.  If the caller is the
472                                        wrapper function "print_func_name",
473                                        this function will bump it up by 1.
474        """
475    ```
476-   As shown in the prior example, if your function has any arguments, include
477    a "Description of argument(s)" section.  This effectively serves as the
478    help text for anyone wanting to use or understand your function.  Include
479    real data examples wherever possible and applicable.
480-   Function definitions:
481    -   Put each function parameter on its own line:
482        ```
483        def func1(parm1,
484
485                  parm2):
486        ```
487-   Do not keep commented-out code in your program.  Instead, remove it
488    entirely.
489-   When you define a python function, observe the following
490    conventions:
491    - Separate words with single underscores.
492    - Use lower-case letters.
493
494    Examples:
495    ```
496
497    def this_is_correct():
498
499        # This function name is correct.
500
501    def This_Is_Incorrect():
502
503        # This keyword name is incorrect because of the upper case letters used.
504
505    def soisthis():
506
507        # This keyword name is incorrect because of 1) a failure to
508        # separate words with underscores.
509
510    ```
511-   Documentation strings:
512    -  Each documentation string should be phrased as an **English command**.
513       Punctuate it correctly with the first word capitalized and a period at
514       the end.
515
516       Correct example:
517        ```
518        def boot_bmc():
519            r"""
520            Boot the BMC.
521            """
522        ```
523        Incorrect example:
524        ```
525        def boot_bmc():
526            r"""
527            This function boots the BMC.
528            """
529
530            # The doc string above is not phrased as a command.
531        ```
532    -   Doc strings should begin with a summary line which is one terse,
533        descriptive sentence.  Put additional commentary below.
534
535        Correct example:
536        ```
537        def stop_sol_console_logging():
538            r"""
539            Stop system console logging and return log output.
540
541            Additional comments...
542            """
543        ```
544        Incorrect example:
545        ```
546        def stop_sol_console_logging():
547            r"""
548            Stop system console logging.  If there are multiple system console
549            processes, they will all be stopped.  If there is no existing log file
550            this keyword will return an error message to that effect (and write that
551            message to targ_file_path, if specified).  NOTE: This keyword will not
552            fail if there is no running system console process.
553            """
554
555            # This summary is way too long.
556        ```
557-   General variable naming conventions:
558    -   Variable names should be lower case with few exceptions (listed here):
559        -   Environment variables should be all upper case.
560    -   Words within a variable name should be separated by underscores:
561
562        Correct examples:
563        ```
564        ${host_name}
565        ${program_pid}
566        ```
567        Incorrect examples:
568        ```
569        ${HostName}
570        ${ProgramPid}
571        ```
572-   Special variable naming conventions.
573
574    For certain very commonly used kinds of variables, please observe these
575    conventions in order to achieve consistency throughout the code.
576
577    -   hosts
578
579        When a variable is intended to contain **either** an IP address **or**
580        a host name (either long or short), please give it a suffix of "_host".
581
582        Examples:
583        ```
584        openbmc_host
585        os_host
586        pdu_host
587        openbmc_serial_host
588        ```
589    -   host names
590
591        For host names (long or short, e.g. "bmc1" or "bmc1.example.com"), use
592        a suffix of _host_name.
593
594        Examples:
595        ```
596        openbmc_host_name
597        os_host_name
598        pdu_host_name
599        openbmc_serial_host_name
600        ```
601    -   Short host names
602
603        For short host names (e.g. "bmc1"), use a suffix of _host_short_name.
604
605        Examples:
606        ```
607        openbmc_host_short_name
608        os_host_short_name
609        pdu_host_short_name
610        openbmc_serial_host_short_name
611        ```
612    -   IP addresses
613
614        For IP addresses, use a suffix of _ip.
615
616        Example:
617        ```
618        openbmc_ip
619        os_ip
620        pdu_ip
621        openbmc_serial_ip
622        ```
623-   Files and directories:
624    -   Files:
625        -   If your variable is to contain only the file's name, use a suffix
626            of _file_name.
627
628            Examples:
629            ```
630            ffdc_file_name = "bmc1.170428.120200.ffdc"
631            ```
632        -   If your variable is to contain the path to a file, use a suffix of
633            _file_path.  Bear in mind that a file path can be relative or
634            absolute, so that should not be a consideration in whether to use
635            the "_file_path" suffix.
636
637            Examples:
638            ```
639            status_file_path = "bmc1.170428.120200.status"
640            status_file_path = "subdir/bmc1.170428.120200.status"
641            status_file_path = "./bmc1.170428.120200.status"
642            status_file_path = "../bmc1.170428.120200.status"
643            status_file_path = "/home/user1/status/bmc1.170428.120200.status"
644            ```
645            To re-iterate, it doesn't matter whether the contents of the
646            variable are a relative or absolute path (as shown in the
647            examples above).  A file path is simply a value with enough
648            information in it for the program to find the file.
649
650        -   If the variable **must** contain an absolute path (which should be
651            the rare case), use a suffix _abs_file_path.
652
653    -   Directories:
654        -   Directory variables should follow the same conventions as file
655            variables.
656
657        -   If your variable is to contain only the directory's name, use a
658            suffix of _dir_name.
659
660            Example:
661            ```
662            ffdc_dir_name = "ffdc"
663            ```
664        -   If your variable is to contain the path to a directory, use a
665            suffix of _dir_path.  Bear in mind that a dir path can be
666            relative or absolute so, that should not be a consideration in
667            whether to use _dir_path.
668
669            Examples:
670            ```
671            status_dir_path = "status/"
672            status_dir_path = "subdir/status"
673            status_dir_path = "./status/"
674            status_dir_path = "../status/"
675            status_dir_path = "/home/user1/status/"
676            ```
677            To re-iterate, it doesn't matter whether the contents of
678            the variable are a relative or absolute path (as shown in
679            the examples above).  A dir path is simply a value with
680            enough information in it for the program to find the
681            directory.
682
683        -   If the variable **must** contain an absolute path (which
684            should be the rare case), use a suffix _abs_dir_path.
685        -   IMPORTANT:  As a programming convention, do pre-
686            processing on all dir_path variables to ensure that they
687            contain a trailing slash.  If we follow that convention
688            religiously, that when changes are made in other parts of
689            the program, the programmer can count on the value having
690            a trailing slash.  Therefore they can safely do this kind
691            of thing:
692            ```
693            my_file_path = my_dir_path + my_file_name
694            ```
695-   Traditional comments (i.e. using the hashtag style comments)
696    -   Please leave one space following the hashtag.
697        ```
698        #wrong
699
700        # Right
701        ```
702    -   Please use proper English punction:
703        -   Capitalize the first word in the sentence or phrase.
704        -   End sentences (or stand-alone phrases) with a period.
705
706    -   Do not keep commented-out code in your program.  Instead, remove it
707        entirely.
708
709Template Usage Requirements
710---------------------------
711We have several templates in the templates/ sub-directory. If there is a
712template that applies to your programming situation (Python, bash, etc.),
713it should be used to create new programs as in the following example
714
715- Example:
716
717    ```
718    $ cd templates
719    $ cp python_pgm_template] ../bin/my_new_program.py
720    ```
721
722These templates have much of your preliminary work done for you and will help
723us all follow a similar structure.
724
725See [python_pgm_template](templates/python_pgm_template) as an example.
726
727- Features:
728    - Help text and arg parsing started for you.
729    - Support for "stock" parameters like "quiet", "debug", "test_mode".
730    - "exit_function" pre-defined.
731    - "validate_parms" function pre-defined.
732    - "main" function follows conventional startup sequence:
733
734        ```
735            gen_get_options(parser, stock_list)
736
737            validate_parms()
738
739            qprint_pgm_header()
740
741            # Your code here.
742        ```
743