- Fundamentals
- Table of contents
- Importance of reading the material
- Naming Conventions
- Dividing The Code Into Commits
- Imitate Existing Solutions
- Simplify Complex Expressions
- Delete Unused Material
- Don't Hoard Issues
- Don't Use Magic Numbers
- Avoid Code Repetition
- Avoid Tight Coupling
- Iterate Before Opening a PR
- Resolving Conversations
- When to Merge PR
- Most people who skim through the reading materials fail to become long term contributors. The key to passing is reading the developer docs during each PR for the first 2 weeks, until you've internalized them. And read them again each month to realign.
your_short_name/feature-description
- Ex:
goktug/main-ejs-env-system
- Follow https://mirrors.edge.kernel.org/pub/software/scm/git/docs/git-check-ref-format.html, and don't use any special characters like
'
.
your_short_name/<PR Name>
- Ex:
muhammad/Disable Spectator Trade Options
The correct file naming convention depends on the file type
.ejs
and.scss
files use snake_case naming (e.g.style.scss
,content_disable_ad_block.ejs
).ts
files use PascalCase naming (e.g.Api.ts
,UIProfileFriendsController.ts
)
Folders use snake_case naming
- Functions use camelCase naming.
- Make sure each function has a specific purpose. If you can't name a function easily, it is doing too many things
- Make sure function names capture the purpose of the function
Since functions perform an action, it's often useful to start a function name with a verb. Some common verbs used are add
, set
, update
, create
and remove
)
function addToList(list: number[], item: number) {
list.push(item)
}
function createList(item1, item2) {
return [item1, item2]
}
Classes use PascalCase naming, all properties and functions within the class should use camelCase.
import {PascalCaseClass} from './code/snake_case_folder/PascalCaseFile.ts'
class PascalCaseClass {
camelCaseVariable: string
camelCaseFunction(): void {
}
}
Each class is kept in its own file, which must have the same name as the class (e.g. class MyClass
saved to MyClass.ts
)
Don't save multiple classes in the same file. If you need to create many classes for a feature, they should all be saved in their own folder.
.
└── my_feature/
├── MyFeatureClass1.ts
└── MyFeatureClass2.ts
- Prefer specific names instead of typical loop variable names like x, i, j, k
- Shift names from CS naming to meaning (e.g: statusFlag to isValidDatabaseEntry)
- Make sure class names defined for HTML should always use kebab case. For example
singleton-page-container
is the correct class convetion whilesingleton_page_container
is not. If you find such cases, please create PRs with the fix - The generic/parent word should come first
Example:
MenuItemText
instead ofMenuTextItem
, menu has item, item has text
The way to name the variable:
- Explain what the variable does to someone
- Use that explanation as the name
Examples:
- https://curc.readthedocs.io/en/latest/programming/coding-best-practices.html
- https://www.loom.com/share/75692b3f08ce4d1da51e067268618d86
Break down your commits into the smallest commit that represents a single
change in a build-able state. Ideally only 1 operation should happen in a commit.
Creating big commits result in
- Extending review time
- Possibility of missing a problem during review
- Lower code quality
- Reviewer being overwhelmed and not merging your PR, asking you to do it again
This commit can be broken down into:
- Renaming the function
- Moving function elsewhere
- Changing where it was called from
- Cleaning it up
- Writing a patch-note
Never do such commits:
Always create and commit code elements in dependency order - dependent elements first. This keeps your code functional and simplifies code reviews.
Example: https://prnt.sc/vQZcWSvCkIcS
When you want to move a certain code from one place to another never separate the commit into add commit
and remove commit
. They should be in a single commit which is a move commit
When the developer does an add commit
in hopes of deleting the original code in the future, what they are actually doing is a duplicate commit
. This makes the reviewers job harder by making them need to jump back and forth increasing the likely hood of them missing a potential flaw. This results in a bad code quality overtime resulting in bad product.
Bad Example:
There is a high chance that a problem similar to the one you're currently trying to solve exists in the codebase. Try to find components that might have had similar problems, understand its logic, and adapt the existing solution into your specific problem.
Example: https://prnt.sc/oCdql_NaawK0
When creating complex expressions or calculations, it is recommended to divide the calculations or assignments into simpler lines. This enhances readability and makes it easier to understand and maintain the code.
Example:
- Before:
const response = await axios.get<DiscordPurchases[]>(`${DISCORD_API_URL}/${KatanServerConfigs.discordAuthClientId()}/entitlements?user_id=${userId}`, {headers: headers})
- After:
const goodVarName = `${DISCORD_API_URL}/${KatanServerConfigs.discordAuthClientId()}/entitlements?user_id=${userId}`
const response = await axios.get<DiscordPurchases[]> (goodVarName, {headers: headers})
After refactoring, some code or resources may become unused. It's important to check if this is the case, and if so, delete unused material to keep the code clean, organized, and easy for others to work with.
Example: https://www.loom.com/share/f99d5206b7394e96b132026863fe409a
- Assign one issue at a time. Once you open a PR and wait for review you can assign a new issue
- There is a possibility the issue you assigned can take 3-5x the time you envisioned. If you have assigned multiple issues, it means you're keeping the other ones from being completed.
Read: https://stackoverflow.com/questions/47882/what-is-a-magic-number-and-why-is-it-bad
- To prevent duplicating code, implement functions, loops, or modular components whenever possible. This approach simplifies our codebase and makes it more maintainable, efficient, and easier to understand for others.
Bad example: https://www.loom.com/share/6d4df641659940ab8f3ab33c2fb50eed
- Tight coupling occurs when changing a part of the code requires changes to the design of another part.
- It causes the code to break in once place due to changes made in another place.
- Avoid creating tight coupling, which may break in the future when someone modifies another part without realizing its consequences.
- Ensure that your code is loosely coupled, and it remains independent from changes in different modules.
Concept explanation: https://stackoverflow.com/a/37993102
Related part of suggested reading material: https://github.com/trantuyen1998/clean-code-javascript#solid
- Before submitting your first implementation as a PR, take the time to iterate and refine it. Aim for 2 or 3 iterations to achieve a better solution with each attempt.
- By iterating and refining your implementation, you increase the chances of your PR being approved faster and with fewer requested changes.
In your PRs, you'll get a bunch of reviews. There are a few actions you should take
- If you agree,
- Leave a thumbs up
- Add the conversation link to your commit message
- Resolve conversation
- Do not comment saying you did it, the reviewer gets extra notification https://i.imgur.com/Q4IRFsO.png
- If you disagree,
- Explain why you disagree and what you propose
- If you're unsure,
- Ask follow up questions
- Merge -> Get approval from 1 QA and 1 Dev
- Not Merge -> Get approval from 2 QAs
- Not Merge -> Get approval from 2 devs
- Not Merge -> Get approval from 1 designer, 1 dev