Querying Data
For each model, the store a set of functions to query the data. It can be accessed by using the model name: store.<model_name>.<method>
.
Here are some examples:
const store = await createStore({
models: [
{ name: 'Todo' },
{ name: 'users' },
],
plugins: [],
})
const { data: todos } = store.Todo.queryMany()
const { data: users } = store.users.queryMany()
Query composables
The query composables are the recommended way to fetch data from the server. They are designed to be used in a Vue component and return a reactive result to be used in the components.
The queryFirst
and queryMany
composables return an object with the following properties:
data
: a ref that contains the data.loading
: a ref that indicates whether the data is being fetched.error
: a ref that contains the error if the data could not be fetched.refresh
: a function that can be called to refresh the data.
The composables also return a promise so they can be used with async setup.
<script setup>
const store = useStore()
// Plays nice with Nuxt SSR!
const { data: todos } = await store.Todo.queryMany()
</script>
Query first
Composable function that reads the first item that matches the key or the filter in the cache and fetches it if not found.
const someKey = ref('some-key')
const { data: todo } = store.Todo.queryFirst(someKey)
With filtering:
const email = ref('cat@acme.com')
const { data: someUsers } = store.User.queryFirst(() => ({
filter: item => item.email === email.value,
params: {
email: email.value,
},
}))
INFO
Why specify the params
in the query? The params
are used to fetch the data from the server. By design, we want to compute the result also on the client side -- hence the filter
function. The params
are used to fetch the data from the server and are accessible in the Plugin hooks.
We can also create our own convention for the params
and filter
. For example, we could turn the filter
option into an object that is both used to filter the data in the cache and to fetch the data from the server, using plugin hooks.
Query many
Composable function that reads all items that match the (optional) filter in the cache and fetches them if not found.
const { data: todos } = store.Todo.queryMany()
const email = ref('@acme.com')
const { data: someUsers } = store.User.queryMany(() => ({
filter: item => item.email.endsWith(email.value),
params: {
email: {
$like: `%${email.value}`,
},
},
}))
Disabling a query
You can disable a query by returning the { enabled: false }
flag as the find options:
const enabled = ref(false)
const { data: todos } = store.Todo.queryMany(() => ({
enabled: enabled.value,
}))
const enabled = ref(false)
const { data: todos } = store.Todo.queryFirst(
() => enabled.value ? 'some-id' : { enabled: false }
)
This syntax is useful when you use TS as it will allow guarding against nullish values at the same time:
const someItem = ref<Record<string, any> | null>(null)
const { data: parent } = store.Item.queryFirst(() => someItem.value
? {
params: {
// `someItem.value` is not nullish here
id: someItem.value.parentId,
}
}
: { enabled: false }
)
Dependent queries
// Get the user
const { data: user } = store.User.queryFirst({ /* ... */ })
// Then get the user's projects
const { data: projects } = store.Project.queryMany(() => user.value
? {
params: {
userId: user.value.id,
}
}
: { enabled: false })
Cache read
Peek first
Read the first item that matches the key or the filter in the cache without doing any fetching.
const result = store.Todo.peekFirst('some-key')
With filtering:
const result = store.User.peekFirst({
filter: item => item.email === 'cat@acme.com',
})
Peek many
Read all items that match the (optional) filter in the cache without doing any fetching.
const todos = store.Todo.peekMany()
const someUsers = store.User.peekMany({
filter: item => item.email.endsWith('@acme.com'),
})
Simple queries
Find first
Find the first item that matches the key or the filter in the cache or fetches it if not found.
const todo = store.Todo.findFirst('some-key')
With filtering:
const someUsers = store.User.findFirst({
filter: item => item.email === 'cat@acme.com',
params: {
email: 'cat@acme.com',
},
})
Find many
Find all items that match the (optional) filter in the cache or fetches them if not found.
const todos = store.Todo.findMany()
const someUsers = store.User.findMany({
filter: item => item.email.endsWith('@acme.com'),
params: {
email: {
$like: '%@acme.com',
},
},
})
Fetch Policy
The fetch policy determines how the data is fetched from the server. The default fetch policy is cache-first
, which means that the data is fetched from the cache first and then from the server if not found.
The fetch policy can be set globally in the store or per query.
Global fetch policy
The global fetch policy can be set in the store configuration using findDefaults
:
const store = createStore({
models,
plugins,
findDefaults: {
fetchPolicy: 'cache-first',
},
})
Per query fetch policy
The fetch policy can be set per query using the fetchPolicy
option:
const { data: todos } = store.Todo.queryMany({
fetchPolicy: 'fetch-only',
})
Possible fetch policies
cache-first
(default) means that the query will first try to fetch the data from the cache. If the data is not found in the cache, it will fetch the data from the adapter plugins.cache-and-fetch
means that the query will first try to fetch the data from the cache. It will then fetch the data from the adapter plugins and update the cache.fetch-only
means that the query will only fetch the data from the adapter plugins. The data will be stored in the cache when the query is resolved.cache-only
means that the query will only fetch the data from the cache.no-cache
means that the query will not use the cache and only fetch the data from the adapter plugins. No data will be stored in the cache.
Items list
Usually we write list item components that are used to display a list of items. It is recommended to only pass the key to the list item component and then use queryFirst
to fetch the data in the list item component. By default the fetch policy is cache-first
, so the data will be red from the cache in the item component and no unnecessary requests will be made for each items.
The benefits are that the data is co-located with the component that uses it and there is no need to specify the types of the item prop again (usually a simple id: string | number
is enough). The item component is also easier to potentially reuse in totally different contexts.
<script lang="ts" setup>
const store = useStore()
const { data: todos } = await store.Todo.queryMany()
</script>
<template>
<!-- Only pass the id here -->
<TodoItem
v-for="{ id } in todos"
:id
:key="id"
/>
</template>
<script lang="ts" setup>
const props = defineProps<{
// Easy to type prop
id: string
}>()
const store = useStore()
// No additional request is made here, the data is read from the cache
const { data: todo } = await store.Todo.queryFirst(props.id)
// Enjoy inferred types here too!
</script>
Co-locating queries
The best of both worlds! You can co-locate the queries with the components that use them, while still enjoying the benefits of a centralized store thanks to the cache -- letting it deduplicating and synchronizing all components. This is a powerful pattern that improves the independence of components and thus their maintainability.
<script lang="ts" setup>
const store = useStore()
// The query is co-located with the component, not in a store somewhere else
const { data: user } = await store.User.queryFirst('user-id')
</script>
<template>
<div>
<h1>{{ user.name }}</h1>
</div>
</template>
<script lang="ts" setup>
const store = useStore()
// The centralized cache ensures that the data is up to date here too
const { data: user } = await store.User.queryFirst('user-id')
</script>
<template>
<div>
<h1>{{ user.name }}</h1>
<p>{{ user.email }}</p>
</div>
</template>
Fetching relations
You can pass an include
option to the query to fetch related data. The include
option is an object where the keys are the names of the relations. Learn more about how to define relations here.
const { data: comments } = store.comments.queryMany({
include: {
author: true,
},
})
The cache will automatically resolve the relations as soon as the data is available in the cache.
Plugins hooked on the fetchRelations
hook will also be called to potentially fetch the data of the relations. See Plugin hooks for more details.