Code Pal for ABAP > How to Contribute
We are a team with limited capacity dedicated to several projects. Every single contribution is valid and more than welcome.
Feel free to help us in developing new ABAP Checks and keeping this project up & running!
💡 Keep it Simple: One check validates one single behavior.
Fork our repository, then clone/pull the forked repo via abapGit into your system.
Follow the Fork a repo guide.
Create a new global class under the package *_checks
. It should have a constructor
and redefine the inspect_tokens
methods at least.
CLASS y_check_<its_name> DEFINITION PUBLIC INHERITING FROM y_check_base CREATE PUBLIC .
PUBLIC SECTION.
METHODS constructor.
PROTECTED SECTION.
METHODS inspect_tokens REDEFINITION.
ENDCLASS.
The constructor
method will inherit the superclass, define the check default customization, and set the code inspector message:
METHOD constructor.
super->constructor( ).
settings-prio = <its_priority>.
settings-threshold = <its_threshold>.
settings-documentation = |{ c_docs_path-checks }<its_documentation>.md|.
settings-pseudo_comment = '"#EC <its_pseudo_comment>' ##NO_TEXT.
" settings...
set_check_message( '<its_message>' ).
ENDMETHOD.
The inspect_tokens
method will detect the issue, validate the customizing, and raise the check.
Here, you can use the imported structures
and statements
to loop the tokens
.
LOOP AT ref_scan_manager->get_tokens( ) ASSIGNING FIELD-SYMBOL(<token>)
FROM statement-from TO statement-to.
" ...
ENDLOOP.
💡 Before start implementing the
inspect_tokens
method, we recommend you to create the unit tests to follow the Test Driven Development (TDD) approach.
The detect_check_configuration
method validates the check customizing.
If an empty structure is received, it means the check should not be raise.
DATA(check_configuration) = detect_check_configuration( error_count = <error_count>
statement = <statement_structure> ).
IF configuration IS INITIAL.
RETURN.
ENDIF.
The raise_error
method raises the check.
raise_error( statement_level = statement-level
statement_index = index
statement_from = statement-from
error_priority = configuration-prio ).
The execute_check
method can be redefined when the check searches for an issue in a non-default statement type. The default types are defined in the y_check_base
->execute_check
.
Create a local test class under the global test class created above. It should inherit and implement the abstract methods from the y_unit_test_base
class.
We will use the y_check_prefer_is_not
check as an example.
💡 Create multiples local test classes to validate distinct behaviors.
CLASS ltc_not_is_initial DEFINITION INHERITING FROM y_unit_test_base FOR TESTING RISK LEVEL HARMLESS DURATION SHORT.
PROTECTED SECTION.
METHODS get_cut REDEFINITION.
METHODS get_code_with_issue REDEFINITION.
METHODS get_code_without_issue REDEFINITION.
METHODS get_code_with_exemption REDEFINITION.
ENDCLASS.
The get_cut
method must return an instance of the class under test.
METHOD get_cut.
result ?= NEW y_check_prefer_is_not( ).
ENDMETHOD.
The get_code_with_issue
method must return a snippet of code which raises the check.
METHOD get_code_with_issue.
result = VALUE #(
( ' REPORT y_example. ' )
( ' START-OF-SELECTION. ' )
( ' DATA(count) = 0. ' )
( ' IF NOT count IS INITIAL. ' )
( ' count = 1. ' )
( ' ENDIF. ' )
).
ENDMETHOD.
The get_code_without_issue
method must return a snippet of code which do not raise the check, or how the fixed code should be.
METHOD get_code_with_issue.
result = VALUE #(
( 'REPORT y_example. ' )
( ' START-OF-SELECTION. ' )
( ' DATA(count) = 0. ' )
( ' IF count IS NOT INITIAL. ' )
( ' count = 1. ' )
( ' ENDIF. ' )
).
ENDMETHOD.
The get_code_with_exemption
method must return a snippet of code which usage of the pseudo comment.
METHOD get_code_with_exemption.
result = VALUE #(
( 'REPORT y_example. ' )
( ' START-OF-SELECTION. ' )
( ' DATA(count) = 0. ' )
( ' IF NOT count IS INITIAL. "#EC PREFER_IS_NOT ' )
( ' count = 1. ' )
( ' ENDIF. ' )
).
ENDMETHOD.
Start the transaction SCI
, and go to the Code Inspector > Management of > Checks
menu.
Then, select the new check class and save it.
Start the transaction SCI
again, and change the global check variant.
Then, select the new check class and save it.
Extend the y_demo_failures
class with an example for the new check.
Then, run the code inspector or ATC using the global check variant.
When it is done, please stage
--> commit
--> push
the files to your fork.
In the github.com, create a pull request from your fork to our base repo.
At this point of time, we will verify your code and authorize/approve the merge (if applicable).
Please create the pull request to merge it with our master
branch.
Thank you in advance for contributing and sharing your ideas within our community!
We really appreciate this! 😍