State Management
In a Tauri application, you often need to keep track of the current state of your application or manage the lifecycle of things associated with it. Tauri provides an easy way to manage the state of your application using the Manager
API, and read it when commands are called.
Here is a simple example:
You can later access your state with any type that implements the Manager
trait, for example the App
instance:
For more info, including accessing state in commands, see the Accessing State section.
Mutability
In Rust, you cannot directly mutate values which are shared between multiple threads or when ownership is controlled through a shared pointer such as Arc
(or Tauri’s State
). Doing so could cause data races (for example, two writes happening simultaneously).
To work around this, you can use a concept known as interior mutability. For example, the standard library’s Mutex
can be used to wrap your state. This allows you to lock the value when you need to modify it, and unlock it when you are done.
The state can now be modified by locking the mutex:
At the end of the scope, or when the MutexGuard
is otherwise dropped, the mutex is unlocked automatically so that other parts of your application can access and mutate the data within.
When to use an async mutex
To quote the Tokio documentation, it’s often fine to use the standard library’s Mutex
instead of an async mutex such as the one Tokio provides:
Contrary to popular belief, it is ok and often preferred to use the ordinary Mutex from the standard library in asynchronous code … The primary use case for the async mutex is to provide shared mutable access to IO resources such as a database connection.
It’s a good idea to read the linked documentation fully to understand the trade-offs between the two. One reason you would need an async mutex is if you need to hold the MutexGuard
across await points.
Do you need Arc
?
It’s common to see Arc
used in Rust to share ownership of a value across multiple threads (usually paired with a Mutex
in the form of Arc<Mutex<T>>
). However, you don’t need to use Arc
for things stored in State
because Tauri will do this for you.
In case State
’s lifetime requirements prevent you from moving your state into a new thread you can instead move an AppHandle
into the thread and then retrieve your state as shown below in the “Access state with the Manager trait” section. AppHandle
s are deliberately cheap to clone for use-cases like this.
Accessing State
Access state in commands
For more information on commands, see Calling Rust from the Frontend.
Async commands
If you are using async
commands and want to use Tokio’s async Mutex
, you can set it up the same way and access the state like this:
Note that the return type must be Result
if you use asynchronous commands.
Access state with the Manager
trait
Sometimes you may need to access the state outside of commands, such as in a different thread or in an event handler like on_window_event
. In such cases, you can use the state()
method of types that implement the Manager
trait (such as the AppHandle
) to get the state:
This method is useful when you cannot rely on command injection. For example, if you need to move the state into a thread where using an AppHandle
is easier, or if you are not in a command context.
Mismatching Types
If you prefer, you can wrap your state with a type alias to prevent this mistake:
However, make sure to use the type alias as it is, and not wrap it in a Mutex
a second time, otherwise you will run into the same issue.
© 2025 Tauri Contributors. CC-BY / MIT