Skip to content

Latest commit

 

History

History
235 lines (183 loc) · 7.11 KB

README.md

File metadata and controls

235 lines (183 loc) · 7.11 KB

debtdao.finance

Contributing

Code style follows prettier conventions (yarn prettier). Commit messages follow the Conventional Commits spec.

Initial Setup

  • Fork the original repo into your GitHub account

  • Clone the forked repo from your GitHub account to your local machine

    git clone https://github.com/<your-gh>/yearn-finance-v3.git
    
  • Set origin to your fork. This is where you push your changes to. This is done automatically by the step above.

    git remote add origin https://github.com/<your-gh>/yearn-finance-v3
    
  • Set upstream to original repo.

    git remote add upstream https://github.com/yearn/yearn-finance-v3.git
    
  • Create .env file in root directory of repo then copy contents of .env.example to .env

    cp .env.example .env
    
    • REACT_APP_INFURA_PROJECT_ID should remain blank because we are currently using Alchemy as our provider
    • REACT_APP_ALCHEMY_API_KEY alchemy api key should be provided by the contributor if he hits api limits

Making Changes

  • Create a new local branch from upstream/develop for each PR that you will submit
    git fetch
    git checkout -b <your branch name> upstream/develop
    
  • Commit your changes as you work
    git add .
    git commit -S -m "message"
    

Pushing Changes to your Repo

  • Commits are squashed when PR is merged so rebasing is optional
  • When ready to push
    git fetch
    git merge upstream/develop
    git push origin <branch-name>
    

Submitting Pull Request

  • Go to your GitHub and navigate to your forked repo
  • Click on Pull requests and then click on New pull request
  • Click on compare across forks
  • Click on compare: and select branch that you want to create a pull request for then click on create pull request

Development

yarn dev

or for Windows:

yarn dev-win
  • To enable Dev Mode set REACT_APP_ALLOW_DEV_MODE=true in your .env
  • Wallet Address Override can be activated by navigating to Settings in the app and clicking Enable Dev Mode

Production

yarn build
yarn start

Translations

We use i18n react with locize cli to update/download translations.

Refer to main repo for documentation: https://github.com/locize/locize-cli

Sync with: yarn syncDevLocales must provide api key

Check sync changes with: yarn checkDevLocales

Download prod locales with: yarn downloadProdLocales

Repo Structure

|
|- public
	|- locales = Where we store all copy translations that you see on the page
| src
	|- config = all env vars and constants
	|- client = all React/HTML/CSS
	|- tests = testing services
	|- utils = small helper functinos throughout the codebase 
	|- core
		|- types = data structs for entire codebase
			|- State = for redux state data structs
			|- Service = Credit + Collateral + other services
			|- CreditLine = anything related to credit specifically
		|- frameworks = 3rd party dependency integrations (ethersjs, subgraph, blocknative)
		|- services = Libs for composing frameworks in actions for users
			|- Tx = sending txs
			|- CreditLine = interacting with credit/debt positions
			|- Collateral = interacting with line collateral (Spigot + Escrow)
			|- OnchainMetadata = ens + abis + ?
		|- store = Redux. dynamic state about marketplace, user, tokens, etc. + actions to query/interact

Code Practices / Styleguide

  1. AWLAYS use translated text for anything user facing
    1. using i18n and public.locales
  2. prefer constant immutable vars and functional programming
  3. Type EVERYTHING
    1. We have types for every API call arg, expected API responses, and formatted API responses once processed.

Code Structure

  1. Always try to utilize state vars instead of local vars. Allows connecting workflows between ux components a lot more seamless across the app and its more proper modern react using hooks and selectors.

BAD Practice

const [activeAddress,setAddress] = useState();
setAddress(window.location.path.split('/')[2])
return <h1>User Address: {activeAddress} </h1>

GOOD Practice

const activeAddress = useAppSelector(WalletSelectors.selectSelectedAddress)
const setAddress = () => dispatch(WalletActions.setSelectedAddress)
setAddress(window.location.path.split('/')[2])
return <h1>User Address: {activeAddress} </h1>
  1. Try to organize code by how/when you use it not by arbitrary taxonomies. E.g. when making services, cbe originally had 1 service file for each contract and mapped functions 1<>1 between them. I refactored and combined anything related to borrowing/lending into CreditService and anything related to spigot/escrow to CollateralService (including liquidate() from LoC contract). Or i made “OnchainMetadataService” for ENS + ABIs + who knows what else we might want to get context about random hashes

Error Handling

  1. Good error messages and UX
    1. should only check 1-2 conditions per if to have specific error messages.
    2. Good error message: https://cdn.discordapp.com/attachments/1009551123339825163/1034974530881470494/Screen_Shot_2022-10-26_at_7.39.00_PM.png image
  2. handling undefined vars a. if(!something)
    // good
    if (!something) {...}
    //bad
    if (something === undefined) {...}
    if (something != null) { ... }
    
  3. handling errors in calls
    1. API GET request
      1. return null/undefined as data. ideally error string to users
    2. API POST
      1. return null/undefined as data. error string to users
// Good GET 
const getAPI = () => {
	const response = await axios.get('my.api')
	if(!response || response.error) {
		return {
			data: null,
			error: response?.error || 'Could not fetch API'
		}
	}
}

// Bad GET

const getAPI = () => {
	const response = await axios.get('my.api')
	// ALWAYS validate response + data
	return response.data
}

const getAPI = () => {
	const response = await axios.get('my.api')
	if(!response || response.error) {
		throw new Error('aaaaaaaaah');
	}
}

const getAPI = () => {
	const response = await axios.get('my.api')
	if(!response || response.error) {
		return Promise.resolve();
	}
}

// Good POST
const getAPI = () => {
	const response = await axios.post('my.api', {data: myData})
	if(!response || response.error) {
		return {
			data: null,
			error: response?.error || 'Could not fetch API'
		}
	}
}

// Bad POST 
const getAPI = () => {
	const response = await axios.post('my.api', {data: myData})
	// DONT do side effects regardless of results
	setTransactionStatus(1);
	return response.data;
}