actionRoute actions are the "writes" to route loader "reads". They provide a way for apps to perform data mutations with simple HTML and HTTP semantics while React Router abstracts away the complexity of asynchronous UI and revalidation. This gives you the simple mental model of HTML + HTTP (where the browser handles the asynchrony and revalidation) with the behavior and UX capabilities of modern SPAs.
createBrowserRouter
<Route
  path="/song/:songId/edit"
  element={<EditSong />}
  action={async ({ params, request }) => {
    let formData = await request.formData();
    return fakeUpdateSong(params.songId, formData);
  }}
  loader={({ params }) => {
    return fakeGetSong(params.songId);
  }}
/>
Actions are called whenever the app sends a non-get submission ("post", "put", "patch", "delete") to your route. This can happen in a few ways:
// forms
<Form method="post" action="/songs" />;
<fetcher.Form method="put" action="/songs/123/edit" />;
// imperative submissions
let submit = useSubmit();
submit(data, {
  method: "delete",
  action: "/songs/123",
});
fetcher.submit(data, {
  method: "patch",
  action: "/songs/123/edit",
});
paramsRoute params are parsed from dynamic segments and passed to your action. This is useful for figuring out which resource to mutate:
<Route
  path="/projects/:projectId/delete"
  action={({ params }) => {
    return fakeDeleteProject(params.projectId);
  }}
/>
requestThis is a Fetch Request instance being sent to your route. The most common use case is to parse the FormData from the request
<Route
  action={async ({ request }) => {
    let formData = await request.formData();
    // ...
  }}
/>
A Request?!
It might seem odd at first that actions receive a "request". Have you ever written this line of code?
<form
  onSubmit={(event) => {
    event.preventDefault();
    // ...
  }}
/>
What exactly are you preventing?
Without JavaScript, just plain HTML and an HTTP web server, that default event that was prevented is actually pretty great. Browsers will serialize all the data in the form into FormData and send it as the body of a new request to your server. Like the code above, React Router <Form> prevents the browser from sending that request and instead sends the request to your route action! This enables highly dynamic web apps with the simple model of HTML and HTTP.
Remember that the values in the formData are automatically serialized from the form submission, so your inputs need a name.
<Form method="post">
  <input name="songTitle" />
  <textarea name="lyrics" />
  <button type="submit">Save</button>
</Form>;
// accessed by the same names
formData.get("songTitle");
formData.get("lyrics");
For more information on formData see Working with FormData.
While you can return anything you want from an action and get access to it from useActionData, you can also return a web Response.
For more information, see the loader documentation.
You can throw in your action to break out of the current call stack (stop running the current code) and React Router will start over down the "error path".
<Route
  action={async ({ params, request }) => {
    const res = await fetch(
      `/api/properties/${params.id}`,
      {
        method: "put",
        body: await request.formData(),
      }
    );
    if (!res.ok) throw res;
    return { ok: true };
  }}
/>
For more details and expanded use cases, read the errorElement documentation.