Skip to main content

Migration from v6 to v7

What's new?

Well, pretty much everything. Version 7 now works using methods; you might be familiar with this design if you have worked with tRPC or Kysely. A complete rewrite of the library in this direction was needed to vastly improve next-safe-action's APIs, and ensure that future versions will not break them (unless React/Next.js APIs change under the hood). The new design is much more resilient, powerful and flexible.

TL;DR

But please still read this migration guide carefully before upgrading to v7.

Assuming you're using Zod, in previous versions, you'd define an auth action client and then an action like this:

action-client-v6.ts
import { createSafeActionClient } from "next-safe-action";
import { cookies } from "next/headers";

// Base client
export const baseActionClient = createSafeActionClient();

// Auth client
export const authActionClient = createSafeActionClient({
async middleware(parsedInput) {
const session = cookies().get("session")?.value;

if (!session) {
throw new Error("Session not found!");
}

const userId = await getUserIdFromSessionId(session);

if (!userId) {
throw new Error("Session is not valid!");
}

return { userId };
},
});
action-v6.ts
"use server";

import { authActionClient } from "@/lib/safe-action";
import { z } from "zod";

export const editProfile = authActionClient(z.object({ username: z.string() }), async ({ username }, { ctx: { userId } }) => {
await saveNewUsernameInDb(userId, username);

return {
updated: true,
}
})

The same behavior can be achieved in v7 with the following refectored code:

action-client-v7.ts
import { createSafeActionClient } from "next-safe-action";
import { cookies } from "next/headers";

// Base client
export const actionClient = createSafeActionClient();

// Auth client
export const authActionClient = actionClient.use(async ({ next, ctx }) => {
const session = cookies().get("session")?.value;

if (!session) {
throw new Error("Session not found!");
}

const userId = await getUserIdFromSessionId(session);

if (!userId) {
throw new Error("Session is not valid!");
}

return next({ ctx: { userId } });
});
action-v7.ts
"use server";

import { authActionClient } from "@/lib/safe-action";
import { z } from "zod";

export const editProfile = authActionClient
.schema(z.object({ username: z.string() }))
.action(async ({ parsedInput: { username }, ctx: { userId } }) => {
await saveNewusernameInDb(userId, username)

return {
updated: true,
}
});

New features

Allow setting validation errors in action server code function

Sometimes it's useful to set custom validation errors in the action server code function, for example when the user wants to log in, but there was a problem with the email or password fields. next-safe-action v7 introduces a new function called returnValidationErrors that allows you to do that.

Support schema nested objects validation

Before v7, next-safe-action allowed you to define schemas with nested objects, but validation errors were not correctly set for nested fields. Version 7 of the library changes the returned errors to be an object with nested fields, that emulates Zod's format method.

Support middleware chaining

This is a core change in next-safe-action v7. In previous versions, you could define just one "monolithic" middleware at the instance level. So, the previous workflow was to define multiple safe action clients, each one with its own middleware.

With version 7, you can chain multiple middleware functions using the use method, both at the instance level and at the action level. This is explained in detail in the middleware page of the documentation. The new design is much more flexible and powerful, allowing you to do things that just couldn't be done before, such as extending context, logging action execution, integrating with third party systems for error reporting, etc.

Generic type for serverError

The serverError property of the action result object is now of generic type. By default it's a string with a default value of "Something went wrong while executing the operation.". You can customize error value and type using the handleReturnedServerError initialization function, just like pre-v7. Basically, what you return from that function is what serverError will be on the client.

Support binding additional arguments

Next.js allows you to pass additional arguments to the action using JavaScript bind method. This approach has the advantage of supporting progressive enhancement.

next-safe-action v7 supports bind arguments via the bindArgsSchemas method.

Support setting default validation errors shape per instance

By default, next-safe-action v7 returns validation errors in an object of the same shape as Zod's format method. You can override this behavior globally by setting the defaultValidationErrorsShape optional property to flattened in createSafeActionClient method. Doing so, the validation errors are returned in the shape of the Zod's format method. If you need a custom format for a specific action, you can override the default shape using the handleValidationErrorsShape and handleBindArgsValidationErrorsShape optional functions in schema and bindArgsSchemas methods, as explained below.

Support custom validation errors shape

As already said above, by default version 7 now returns validation errors in the same format of the Zod's format method.

This is customizable by using the handleValidationErrorsShape/handleBindArgsValidationErrorsShape optional functions in schema/bindArgsSchemas methods. Check out this page for more information. For example, if you need to work with flattened errors for a specific action, next-safe-action conveniently provides two functions to do that: flattenValidationErrors and flattenBindArgsValidationErrors.

Allow calling action method without schema

Sometimes it's not necessary to define an action with input. In this case, you can omit the schema method and use directly the action/stateAction method.

Support passing schema via async function

When working with i18n solutions, often you'll find implementations that require awaiting a getTranslations function in order to get the translations, that then get passed to the schema. Starting from version 7, next-safe-action allows you to pass an async function to the schema method, that returns a promise of type Schema. More information about this feature can be found in this discussion on GitHub and in the i18n recipe.

Support stateful actions using React useActionState hook

React added a hook called useActionState that replaces the previous useFormState hook and improves it. next-safe-action v7 uses it under the hood in the exported useStateAction hook, that keeps track of the state of the action execution.

Note that this hook expects as argument actions defined using the stateAction method, and not the usual action method. Find more information about these two methods here.

Return input from hooks

Sometimes it's useful to access the input passed to an action when using hooks. Starting from version 7, input property is returned from hooks.

Return shorthand statuses from hooks

Starting from version 7, isIdle, isExecuting, hasSucceeded and hasErrored are returned from hooks, in addition to the status property. This is the same behavior of next-safe-action pre-v4 and very similar to the TanStack Query API.

Return executeAsync from useAction and useOptimisticAction hooks

Sometimes it's useful to await the result of an action execution when using actions via hooks. Starting from version 7, executeAsync is returned from useAction and useOptimisticAction hooks. It's essentially the same as the original safe action function, with the added benefits from the hooks. It's currently not possible to add this function to the useStateAction hook, due to internal React limitations.

Refactors

serverCodeFn signature

Previously, serverCodeFn had two arguments: parsedInput and ctx. Now, it only has one argument, which is an object that contains parsedInput and ctx, and other useful properties. In the case of stateAction method, serverCodeFn also has an additional argument, which is an object that contains the previous result of the action. Find more information about serverCodeFn here.

useOptimisticAction signature

The function signature for useOptimisticAction has been updated to be much more clear and readable. Before, you had to pass currentState and updateFn as the second and third argument of the hook. Now, the first argument is the safe action, and additional required and optional properties are placed inside the second argument of the hook, which is an object.

Other than that, now currentState is unlinked from the safe action's return value. The action purpose in optimistic state updates is just to make mutations of data. Then, the fresh data is refetched from the parent Server Component, so it didn't make sense to lock the type of currentState to the action's return type. This is explained in detail here and here.

Find more information about the updated useOptimisticAction hook here.

Hook callbacks arguments

Previously, there were multiple arguments in hook callbacks. Now, metadata is passed inside a single object that is the first argument of each function. Find more information about the updated callbacks here.

Action metadata

In version 6, you could pass metadata to actions via the third argument of the safe action function, after serverCodeFn. In version 7, there's a dedicated metadata method that lets you define useful data for the action execution. This data can then be accessed in middleware functions and serverCodeFn. Find more information about the metadata method here.

Internal changes

TypeSchema update

TypeSchema was updated to v0.13, so now, if you want to use a validation library other than Zod, you also need to install the related TypeSchema adapter.

Requirements

next-safe-action version 7 requires Next.js 15 and React 19 or later to work.

What about v6?

You can still keep using version 6 and eventually upgrade to version 7. Note that version 6 is frozen and no new features will be released in the future for it. v6 documentation can still be found here.