Vue/TypeScript
Testing
The Vue/TypeScript part of the application under the frontend
directory has two types of tests. The unit tests, which test functions and components, use vitest
and vue-test-utils
. You can run them with:
pnpm run --filter rotki test:unit
These are small tests ensuring that parts of the code work well in isolation.
The second type of tests is an end-to-end (e2e
) test suite using cypress
. The e2e tests require the Python virtual environment because they depend on the actual Python backend. These tests ensure proper e2e functionality and application integration and try to replicate scenarios of real user interaction through the application.
To run the e2e tests, use the following command inside the frontend directory:
pnpm run --filter rotki test:integration-ci
The above command will run the e2e tests in headless mode. If you want to debug specific tests, you can also run:
pnpm run --filter rotki test:integration
This command will open the Cypress Test Runner window where you can select specific suites to execute.
Linting
If you are doing frontend development, it is highly recommended to enable the available hooks:
pnpm run setup:hooks
You will see a message saying husky - Git hooks installed
. Now you should have a pre-commit hook that runs whenever you commit a file and lints the staged files.
Before committing and pushing your commits, ensure that you fix any lint issues. You can do this by running:
pnpm run lint:fix
Note: While lint warnings are not fatal and will not fail the CI pipeline, it would be better if a PR reduces the number of warnings and doesn't introduce new ones. Warnings are things that need to be fixed, and they will be converted to errors in the future.
Vue
Setup Script Macros
When using the defineProps
or defineEmits
macros in the setup script, the defineX<{}>()
format should be used instead of defineX({})
.
Any instances of defineX({})
should eventually be replaced with defineX<{}>()
.
For defineEmits<{...}>()
before the migration to vue 3 the verbose style was used. When you encounter such entries, try to replace with the short style instead.
e.g.
Before
const emit = defineEmits<{
(e: 'update:msg', msg: string): void;
}>();
After
const emit = defineEmits<{
'update:msg': [msg: string];
}>();
Setup script order
The preferred order of a setup script in should be the following:
// 1. Imports
import { get } from '@vueuse/core';
import { defineExpose } from '@vue/runtime-core';
// 2. Definitions (defineX)
defineOptions({
inheritAttrs: false,
});
const props = defineProps<{
msg: string;
}>();
const emit = defineEmits<{
'update:msg': [msg: string];
}>();
const { msg } = toRefs(props);
// 3. I18n & vue-router
const { t } = useI18n();
const router = useRouter();
const route = useRoute();
// 4. Reactive State variables
const counter = ref(0);
// 5. Use Pinia stors
const { todos } = toRefs(useTodoStore());
// 6. Composables
const { title } = useTitle();
// 7. Computed
const titleNumber = computed(() => `${get(title)} ${get(counter)}`);
// 8. Define Methods
function increaseCounter() {
set(counter, get(counter) + 1);
}
// 9. Watchers
watch(title, (title) => {
emit('update:msg', title);
});
// 10. Lifecycle
onMounted(() => {
increaseCounter();
});
// 11. Exposed
defineExpose({
increaseCounter,
});
Pinia Store
For pinia stores the suggested order for elements is the following.
// stores/counter.ts
import { defineStore } from 'pinia';
import { computed, ref } from 'vue';
export const useCounterStore = defineStore('counter', () => {
// 1. State
const count = ref<number>(0);
// 2. Getters
const doubleCount = computed(() => count.value * 2);
// 3. Actions
function increment(): void {
count.value++;
}
function decrement(): void {
count.value--;
}
// 4. Watchers
watch(count, (count) => {
// do something
});
return {
count,
doubleCount,
increment,
decrement,
};
});
useCssModules
It should be only used if there is some access to the css module class names inside the script tag. If the only usage is in the template then use $style
instead
useAttrs
It should be only used if there is some access to the attrs
inside the script tag. If the only usage is in the template part then use $attrs
instead.
Style Tag
Initially, the style tag was using scoped SCSS with BEM for naming. Any scoped style should eventually be replaced with CSS Modules, and we should simplify naming and move away from BEM.
Dependencies
Adding New Dependencies
As a rule of thumb, we should pick dependencies that come from well-known trusted sources, e.g., known Vue ecosystem/Nuxt maintainers with a good track record. From experience, these dependencies tend to have better support and more regular updates.
If the functionality implemented is simple enough and doesn't add a big maintenance overhead to the team, it would be preferable to skip the extra dependency and implement it as part of our codebase.
Versions
We always pin strict versions of our first-party dependencies, e.g.:
{
"dependencies": {
"package": "1.0.0"
}
}
instead of:
{
"dependencies": {
"package": "^1.0.0"
}
}