2019-11-14 01:23:59 +08:00
# CONTRIBUTION GUIDELINES
## Before contributing
2020-10-19 20:34:13 +08:00
2022-12-17 04:49:48 +08:00
Welcome to [TheAlgorithms/C-Plus-Plus ](https://github.com/TheAlgorithms/C-Plus-Plus )! Before submitting pull requests, please make sure that you have **read the whole guidelines** . If you have any doubts about this contribution guide, please open [an issue ](https://github.com/TheAlgorithms/C-Plus-Plus/issues/new/choose ) or ask on our [Discord server ](https://the-algorithms.com/discord/ ), and clearly state your concerns.
2019-11-14 01:23:59 +08:00
## Contributing
2020-10-19 20:34:13 +08:00
2021-07-06 11:02:33 +08:00
### Maintainer/reviewer
**Please check the [reviewer code ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/REVIEWER_CODE.md ) file for maintainers and reviewers.**
2019-12-04 11:47:11 +08:00
### Contributor
2020-10-19 20:34:13 +08:00
2021-06-19 03:55:52 +08:00
Being a contributor at The Algorithms, we request you to follow the points mentioned below:
2020-10-19 20:34:13 +08:00
2019-12-04 11:47:11 +08:00
- You did your own work.
2022-09-26 05:14:10 +08:00
- No plagiarism is allowed. Any plagiarized work will not be merged.
2022-10-03 00:40:31 +08:00
- Your work will be distributed under the [MIT License ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/LICENSE ) once your pull request has been merged.
2021-06-19 03:55:52 +08:00
- Please follow the repository guidelines and standards mentioned below.
2019-11-14 01:23:59 +08:00
2021-06-19 03:55:52 +08:00
**New implementation** New implementations are welcome!
2020-05-24 14:52:09 +08:00
2022-09-26 05:14:10 +08:00
You can add new algorithms or data structures that are **not present in the repository** or that can **improve** the old implementations (**documentation**, **improving test cases** , removing bugs, or in any other reasonable sense)
2020-05-24 14:52:09 +08:00
2021-06-19 03:55:52 +08:00
**Issues** Please avoid opening issues asking to be "assigned” to a particular algorithm. This merely creates unnecessary noise for maintainers. Instead, please submit your implementation in a pull request, and it will be evaluated by project maintainers.
2019-11-14 01:23:59 +08:00
### Making Changes
#### Code
2020-10-19 20:34:13 +08:00
2019-12-04 11:47:11 +08:00
- Please use the directory structure of the repository.
2022-09-26 05:14:10 +08:00
- Make sure the file extensions are `*.hpp` , `*.h` or `*.cpp` .
2021-06-19 03:55:52 +08:00
- Don't use ** `bits/stdc++.h` ** because this is quite Linux-specific and slows down the compilation process.
2021-10-21 04:13:35 +08:00
- Organize your code using ** `struct` **, ** `class` **, and/or ** `namespace` ** keywords.
2023-04-22 00:10:54 +08:00
- If an implementation of the algorithm already exists, please refer to the [file-name section below ](#file-name-guidelines ).
2019-12-04 11:47:11 +08:00
- You can suggest reasonable changes to existing algorithms.
- Strictly use snake_case (underscore_separated) in filenames.
- If you have added or modified code, please make sure the code compiles before submitting.
2022-09-30 03:26:13 +08:00
- Our automated testing runs [**CMake** ](https://cmake.org/ ) on all the pull requests, so please be sure that your code passes before submitting.
2022-09-26 05:14:10 +08:00
- Please conform to [Doxygen ](https://www.doxygen.nl/manual/docblocks.html ) standards and document the code as much as possible. This not only facilitates the readers but also generates the correct info on the website.
2021-06-19 03:55:52 +08:00
- **Be consistent in the use of these guidelines.**
2019-11-14 01:23:59 +08:00
2020-07-10 06:15:05 +08:00
#### Documentation
2020-10-19 20:34:13 +08:00
2021-06-19 03:55:52 +08:00
- Make sure you put useful comments in your code. Do not comment on obvious things.
2020-07-10 06:15:05 +08:00
- Please avoid creating new directories if at all possible. Try to fit your work into the existing directory structure. If you want to create a new directory, then please check if a similar category has been recently suggested or created by other pull requests.
2021-06-19 03:55:52 +08:00
- If you have modified/added documentation, please ensure that your language is concise and must not contain grammatical errors.
- Do not update [`README.md` ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/README.md ) along with other changes. First, create an issue and then link to that issue in your pull request to suggest specific changes required to [`README.md` ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/README.md ).
- The repository follows [Doxygen ](https://www.doxygen.nl/manual/docblocks.html ) standards and auto-generates the [repository website ](https://thealgorithms.github.io/C-Plus-Plus ). Please ensure the code is documented in this structure. A sample implementation is given below.
2020-07-10 06:15:05 +08:00
#### Test
2020-10-19 20:34:13 +08:00
2021-06-19 03:55:52 +08:00
- Make sure to add examples and test cases in your `main()` function.
- If you find an algorithm or document without tests, please feel free to create a pull request or issue describing suggested changes.
2021-07-06 11:02:33 +08:00
- Please try to add one or more `test()` functions that will invoke the algorithm implementation on random test data with the expected output. Use the `assert()` function to confirm that the tests will pass. Requires including the `cassert` library.
2023-04-22 00:10:54 +08:00
- Test cases should fully verify that your program works as expected. Rather than asking the user for input, it's best to make sure the given output is the correct output.
##### Self-test examples
1. [Quick sort ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/sorting/quick_sort.cpp#L137 ) testing (complex).
```cpp
// Let's make sure the array of numbers is ordered after calling the function.
std::vector< uint64_t > arr = {5, 3, 8, 12, 14, 16, 28, 96, 2, 5977};
std::vector< uint64_t > arr_sorted = sorting::quick_sort::quick_sort(
arr, 0, int(std::end(arr) - std::begin(arr)) - 1);
assert(std::is_sorted(std::begin(arr_sorted), std::end(arr_sorted)));
```
2. [Subset Sum ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/backtracking/subset_sum.cpp#L58 ) testing (medium).
```cpp
std::vector< int32_t > array1 = {-7, -3, -2, 5, 8}; // input array
assert(backtracking::subset_sum::number_of_subsets(0, array1) ==
2); // first argument in subset_sum function is the required sum and
// second is the input array
```
3. Small C++ program that showcases and explains the use of tests.
```cpp
2023-04-27 09:08:19 +08:00
#include <iostream> /// for IO operations
#include <vector> /// for std::vector
#include <cassert> /// for assert
2023-04-22 00:10:54 +08:00
/**
* @brief Verifies if the given array
* contains the given number on it.
* @tparam T the type of array (e.g., `int` , `float` , etc.)
* @param arr the array to be used for checking
* @param number the number to check if it's inside the array
* @return false if the number was NOT found in the array
* @return true if the number WAS found in the array
*/
template < typename T >
bool is_number_on_array(const std::vector< T > & arr, const int & number) {
for (int i = 0; i < sizeof ( arr ) / sizeof ( int ) ; i + + ) {
if (arr[i] == number) {
return true;
}
else {
2023-04-27 09:08:19 +08:00
// Number not in the current index, keep searching.
2023-04-22 00:10:54 +08:00
}
}
return false;
}
/**
* @brief Self-test implementations
* @returns void
*/
static void tests() {
std::vector< int > arr = { 9, 14, 21, 98, 67 };
assert(is_number_on_array(arr, 9) == true);
assert(is_number_on_array(arr, 4) == false);
2023-04-27 08:59:57 +08:00
assert(is_number_on_array(arr, 98) == true);
assert(is_number_on_array(arr, 512) == false);
2023-04-27 01:26:03 +08:00
std::cout << "All tests have successfully passed!\n";
2023-04-22 00:10:54 +08:00
}
/**
* @brief Main function
* @returns 0 on exit
*/
int main() {
tests(); // run self-test implementations
2023-04-22 00:12:03 +08:00
return 0;
2023-04-22 00:10:54 +08:00
}
```
2020-10-19 20:34:13 +08:00
#### Typical structure of a program
2020-07-10 06:15:05 +08:00
```cpp
/**
2020-10-19 20:34:13 +08:00
* @file
2023-04-22 00:10:54 +08:00
* @brief Add one-line description here. Should contain a Wikipedia
2022-09-26 05:14:10 +08:00
* link or another source explaining the algorithm/implementation.
2020-10-19 20:34:13 +08:00
* @details
2021-07-06 11:02:33 +08:00
* This is a multi-line
2020-07-10 06:15:05 +08:00
* description containing links, references,
2021-07-06 11:02:33 +08:00
* math equations, etc.
2020-07-10 06:15:05 +08:00
* @author [Name ](https://github.com/handle )
* @see related_file.cpp, another_file.cpp
*/
2021-07-06 11:02:33 +08:00
#include <cassert> /// for assert
#include /// for `some function here`
2020-07-10 06:15:05 +08:00
2020-08-29 00:47:02 +08:00
/**
2023-04-22 00:10:54 +08:00
* @namespace
2022-09-26 05:14:10 +08:00
* @brief < namespace description >
2020-07-10 06:15:05 +08:00
*/
namespace name {
/**
2021-07-06 11:02:33 +08:00
* @brief Class documentation
2020-07-10 06:15:05 +08:00
*/
2020-08-29 00:47:02 +08:00
class class_name {
2020-07-10 06:15:05 +08:00
private:
2022-09-26 05:14:10 +08:00
int variable; ///< short info of this variable
2020-08-29 00:47:02 +08:00
char *message; ///< short info
2020-07-10 06:15:05 +08:00
public:
2022-09-26 05:14:10 +08:00
// other members should be also documented as below
2020-07-10 06:15:05 +08:00
}
/**
2021-07-06 11:02:33 +08:00
* @brief Function documentation
2020-07-10 06:15:05 +08:00
* @tparam T this is a one-line info about T
* @param param1 on-line info about param1
* @param param2 on-line info about param2
* @returns `true` if ...
2020-08-29 00:47:02 +08:00
* @returns `false` if ...
2020-07-10 06:15:05 +08:00
*/
template< class T >
bool func(int param1, T param2) {
2020-08-29 00:47:02 +08:00
// function statements here
if (/*something bad*/) {
2020-07-10 06:15:05 +08:00
return false;
2020-08-29 00:47:02 +08:00
}
2020-07-10 06:15:05 +08:00
return true;
}
2022-09-26 05:14:10 +08:00
} // namespace name
2020-07-10 06:15:05 +08:00
2020-10-19 20:34:13 +08:00
/**
2021-07-06 11:02:33 +08:00
* @brief Self-test implementations
2020-10-19 20:34:13 +08:00
* @returns void
*/
2020-08-29 00:47:02 +08:00
static void test() {
2021-07-10 03:03:57 +08:00
/* descriptions of the following test */
2020-08-29 00:47:02 +08:00
assert(func(...) == ...); // this ensures that the algorithm works as expected
2020-07-10 06:15:05 +08:00
2020-08-29 00:47:02 +08:00
// can have multiple checks
2023-04-27 01:26:03 +08:00
// this lets the user know that the tests have passed
std::cout << "All tests have successfully passed!\n";
2020-07-10 06:15:05 +08:00
}
2020-10-19 20:34:13 +08:00
/**
* @brief Main function
* @param argc commandline argument count (ignored)
* @param argv commandline array of arguments (ignored)
* @returns 0 on exit
*/
2020-07-10 06:15:05 +08:00
int main(int argc, char *argv[]) {
2021-07-06 11:02:33 +08:00
test(); // run self-test implementations
2020-07-10 06:15:05 +08:00
// code here
return 0;
}
```
2022-12-22 00:37:27 +08:00
#### File Name guidelines
2020-10-19 20:34:13 +08:00
2021-06-19 03:55:52 +08:00
- Use lowercase words with ``"_"`` as a separator
2020-08-29 00:47:02 +08:00
- For instance
2020-10-19 20:34:13 +08:00
```markdown
2019-11-27 23:26:36 +08:00
MyNewCppClass.CPP is incorrect
2019-11-25 23:49:48 +08:00
my_new_cpp_class.cpp is correct format
```
2020-10-19 20:34:13 +08:00
2019-11-25 23:49:48 +08:00
- It will be used to dynamically create a directory of files and implementation.
2021-07-06 11:02:33 +08:00
- File name validation will run on Docker to ensure validity.
2022-09-26 05:14:10 +08:00
- If an implementation of the algorithm already exists and your version is different from that implemented, please use incremental numeric digit as a suffix. For example: if `median_search.cpp` already exists in the `search` folder, and you are contributing a new implementation, the filename should be `median_search2.cpp` . For a third implementation, `median_search3.cpp` , and so on.
2019-11-25 23:49:48 +08:00
2022-12-22 00:37:27 +08:00
#### Directory guidelines
2020-10-19 20:34:13 +08:00
2019-12-03 02:07:27 +08:00
- We recommend adding files to existing directories as much as possible.
2019-12-03 02:03:29 +08:00
- Use lowercase words with ``"_"`` as separator ( no spaces or ```"-"``` allowed )
2019-11-27 23:43:53 +08:00
- For instance
2020-10-19 20:34:13 +08:00
```markdown
2019-11-27 23:43:53 +08:00
SomeNew Fancy-Category is incorrect
some_new_fancy_category is correct
2019-12-05 15:45:53 +08:00
```
2020-10-19 20:34:13 +08:00
2019-12-05 15:45:53 +08:00
- Filepaths will be used to dynamically create a directory of our algorithms.
- Filepath validation will run on GitHub Actions to ensure compliance.
2019-11-27 23:43:53 +08:00
2022-12-22 00:37:27 +08:00
##### Integrating CMake in a new directory
In case a new directory is 100% required, `CMakeLists.txt` file in the root directory needs to be updated, and a new `CMakeLists.txt` file needs to be created within the new directory.
An example of how your new `CMakeLists.txt` file should look like. Note that if there are any extra libraries/setup required, you must include that in this file as well.
```cmake
# If necessary, use the RELATIVE flag, otherwise each source file may be listed
# with full pathname. The RELATIVE flag makes it easier to extract an executable's name
# automatically.
file( GLOB APP_SOURCES RELATIVE ${CMAKE_CURRENT_SOURCE_DIR} *.cpp )
foreach( testsourcefile ${APP_SOURCES} )
string( REPLACE ".cpp" "" testname ${testsourcefile} ) # File type. Example: `.cpp`
add_executable( ${testname} ${testsourcefile} )
set_target_properties(${testname} PROPERTIES LINKER_LANGUAGE CXX)
if(OpenMP_CXX_FOUND)
target_link_libraries(${testname} OpenMP::OpenMP_CXX)
endif()
install(TARGETS ${testname} DESTINATION "bin/< foldername > ") # Folder name. Do NOT include `<>`
endforeach( testsourcefile ${APP_SOURCES} )
```
The `CMakeLists.txt` file in the root directory should be updated to include the new directory.\
Include your new directory after the last subdirectory. Example:
```cmake
...
add_subdirectory(divide_and_conquer)
add_subdirectory(< foldername > )
```
2019-11-14 01:23:59 +08:00
#### Commit Guidelines
2020-10-19 20:34:13 +08:00
2021-06-19 03:55:52 +08:00
- It is recommended to keep your changes grouped logically within individual commits. Maintainers find it easier to understand changes that are logically spilled across multiple commits. Try to modify just one or two files in the same directory. Pull requests that span multiple directories are often rejected.
2020-10-19 20:34:13 +08:00
```bash
2019-11-14 01:23:59 +08:00
git add file_xyz.cpp
git commit -m "your message"
```
2020-10-19 20:34:13 +08:00
2019-11-14 01:23:59 +08:00
Examples of commit messages with semantic prefixes:
2020-10-19 20:34:13 +08:00
```markdown
2019-11-14 01:23:59 +08:00
fix: xyz algorithm bug
feat: add xyx algorithm, class xyz
test: add test for xyz algorithm
2022-09-26 05:14:10 +08:00
docs: add comments and explanation to xyz algorithm/improve contributing guidelines
chore: update Gitpod badge
2019-11-14 01:23:59 +08:00
```
2020-10-19 20:34:13 +08:00
2019-11-14 01:23:59 +08:00
Common prefixes:
2020-10-19 20:34:13 +08:00
2019-11-14 01:23:59 +08:00
- fix: A bug fix
- feat: A new feature
- docs: Documentation changes
2019-12-04 11:47:11 +08:00
- test: Correct existing tests or add new ones
2022-09-26 05:14:10 +08:00
- chore: Miscellaneous changes that do not match any of the above.
2019-11-14 01:23:59 +08:00
### Pull Requests
2020-10-19 20:34:13 +08:00
2019-11-14 01:34:12 +08:00
- Checkout our [pull request template ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/.github/pull_request_template.md )
2019-11-14 01:23:59 +08:00
2020-08-31 01:38:49 +08:00
#### Building Locally
2020-10-19 20:34:13 +08:00
Before submitting a pull request,
build the code locally or using the convenient [![Gitpod Ready-to-Code ](https://img.shields.io/badge/Gitpod-Ready--to--Code-blue?logo=gitpod )](https://gitpod.io/#https://github.com/TheAlgorithms/C-Plus-Plus) service.
```bash
2020-08-31 01:38:49 +08:00
cmake -B build -S .
```
#### Static Code Analyzer
2020-10-19 20:34:13 +08:00
2021-07-06 11:02:33 +08:00
We use [`clang-tidy` ](https://clang.llvm.org/extra/clang-tidy/ ) as a static code analyzer with a configuration in [`.clang-tidy` ](https://github.com/TheAlgorithms/C-Plus-Plus/blob/master/.clang-tidy ).
2020-10-19 20:34:13 +08:00
```bash
2020-08-31 01:38:49 +08:00
clang-tidy --fix --quiet -p build subfolder/file_to_check.cpp --
```
#### Code Formatter
2020-10-19 20:34:13 +08:00
2022-09-30 03:26:13 +08:00
[`clang-format` ](https://clang.llvm.org/docs/ClangFormat.html ) is used for code formatting.
2020-10-19 20:34:13 +08:00
- Installation (only needs to be installed once.)
- Mac (using home-brew): `brew install clang-format`
- Mac (using macports): `sudo port install clang-10 +analyzer`
- Windows (MSYS2 64-bit): `pacman -S mingw-w64-x86_64-clang-tools-extra`
- Linux (Debian): `sudo apt-get install clang-format-10 clang-tidy-10`
- Running (all platforms): `clang-format -i -style="file" my_file.cpp`
2020-08-31 01:38:49 +08:00
#### GitHub Actions
2020-10-19 20:34:13 +08:00
- Enable GitHub Actions on your fork of the repository.
2021-06-19 03:55:52 +08:00
After enabling, it will execute `clang-tidy` and `clang-format` after every push (not a commit).
2020-10-19 20:34:13 +08:00
- Click on the tab "Actions", then click on the big green button to enable it.
![GitHub Actions ](https://user-images.githubusercontent.com/51391473/94609466-6e925100-0264-11eb-9d6f-3706190eab2b.png )
- The result can create another commit if the actions made any changes on your behalf.
- Hence, it is better to wait and check the results of GitHub Actions after every push.
2021-06-19 03:55:52 +08:00
- Run `git pull` in your local clone if these actions made many changes to avoid merge conflicts.
2019-12-05 16:21:08 +08:00
2019-12-06 02:12:07 +08:00
Most importantly,
2020-10-19 20:34:13 +08:00
2019-12-06 02:12:07 +08:00
- Happy coding!