授权
简介
除了提供内置的身份验证服务外,Laravel 还提供了一种简单的方式来授权用户对给定资源执行操作。例如,即使用户已通过身份验证,他们也可能未被授权更新或删除应用程序管理的某些 Eloquent 模型或数据库记录。Laravel 的授权功能提供了一种简单、有组织的方式来管理这些类型的授权检查。
Laravel 提供了两种主要的授权操作方式:Gate(守卫) 和 策略。可以将 Gate(守卫)和策略视为路由和控制器。Gate(守卫)提供了一种基于闭包的简单授权方法,而策略(如控制器)则围绕特定模型或资源对逻辑进行分组。在本文档中,我们将首先探讨 Gate(守卫),然后再研究策略。
在构建应用程序时,您无需选择专门使用 Gate(守卫)或专门使用策略。大多数应用程序很可能包含 Gate(守卫)和策略的某种混合,这完全没有问题!Gate(守卫)最适用于与任何模型或资源无关的操作,例如查看管理员仪表板。相反,当您希望授权对特定模型或资源执行操作时,应使用策略。
Gate(守卫)
编写 Gate(守卫)
Gate(守卫)是学习 Laravel 授权功能基础知识的好方法;但是,在构建健壮的 Laravel 应用程序时,您应考虑使用策略来组织您的授权规则。
Gate(守卫)只是闭包,用于确定用户是否有权执行给定的操作。通常,Gate(守卫)是使用 Gate
外观在 App\Providers\AppServiceProvider
类的 boot
方法中定义的。Gate(守卫)始终接收用户实例作为其第一个参数,并且可以选择接收其他参数,例如相关的 Eloquent 模型。
在此示例中,我们将定义一个 Gate(守卫)来确定用户是否可以更新给定的 App\Models\Post
模型。该 Gate(守卫)将通过比较用户的 id
与创建帖子的用户的 user_id
来完成此操作
1use App\Models\Post; 2use App\Models\User; 3use Illuminate\Support\Facades\Gate; 4 5/** 6 * Bootstrap any application services. 7 */ 8public function boot(): void 9{10 Gate::define('update-post', function (User $user, Post $post) {11 return $user->id === $post->user_id;12 });13}
与控制器类似,Gate(守卫)也可以使用类回调数组定义
1use App\Policies\PostPolicy; 2use Illuminate\Support\Facades\Gate; 3 4/** 5 * Bootstrap any application services. 6 */ 7public function boot(): void 8{ 9 Gate::define('update-post', [PostPolicy::class, 'update']);10}
授权操作
要使用 Gate(守卫)授权操作,您应该使用 Gate
外观提供的 allows
或 denies
方法。请注意,您无需将当前经过身份验证的用户传递给这些方法。Laravel 将自动处理将用户传递到 Gate(守卫)闭包中。通常在应用程序的控制器中调用 Gate(守卫)授权方法,然后再执行需要授权的操作
1<?php 2 3namespace App\Http\Controllers; 4 5use App\Http\Controllers\Controller; 6use App\Models\Post; 7use Illuminate\Http\RedirectResponse; 8use Illuminate\Http\Request; 9use Illuminate\Support\Facades\Gate;10 11class PostController extends Controller12{13 /**14 * Update the given post.15 */16 public function update(Request $request, Post $post): RedirectResponse17 {18 if (! Gate::allows('update-post', $post)) {19 abort(403);20 }21 22 // Update the post...23 24 return redirect('/posts');25 }26}
如果您想确定除当前经过身份验证的用户之外的用户是否有权执行操作,则可以使用 Gate
外观上的 forUser
方法
1if (Gate::forUser($user)->allows('update-post', $post)) {2 // The user can update the post...3}4 5if (Gate::forUser($user)->denies('update-post', $post)) {6 // The user can't update the post...7}
您可以使用 any
或 none
方法一次授权多个操作
1if (Gate::any(['update-post', 'delete-post'], $post)) {2 // The user can update or delete the post...3}4 5if (Gate::none(['update-post', 'delete-post'], $post)) {6 // The user can't update or delete the post...7}
授权或抛出异常
如果您想尝试授权操作,并在用户不允许执行给定操作时自动抛出 Illuminate\Auth\Access\AuthorizationException
,则可以使用 Gate
外观的 authorize
方法。AuthorizationException
的实例会自动由 Laravel 转换为 403 HTTP 响应
1Gate::authorize('update-post', $post);2 3// The action is authorized...
提供额外的上下文
用于授权能力的 Gate(守卫)方法(allows
、denies
、check
、any
、none
、authorize
、can
、cannot
)和授权Blade 指令(@can
、@cannot
、@canany
)可以接收数组作为其第二个参数。这些数组元素作为参数传递给 Gate(守卫)闭包,并且可以在进行授权决策时用于额外的上下文
1use App\Models\Category; 2use App\Models\User; 3use Illuminate\Support\Facades\Gate; 4 5Gate::define('create-post', function (User $user, Category $category, bool $pinned) { 6 if (! $user->canPublishToGroup($category->group)) { 7 return false; 8 } elseif ($pinned && ! $user->canPinPosts()) { 9 return false;10 }11 12 return true;13});14 15if (Gate::check('create-post', [$category, $pinned])) {16 // The user can create the post...17}
Gate(守卫)响应
到目前为止,我们只检查了返回简单布尔值的 Gate(守卫)。但是,有时您可能希望返回更详细的响应,包括错误消息。为此,您可以从您的 Gate(守卫)返回 Illuminate\Auth\Access\Response
1use App\Models\User;2use Illuminate\Auth\Access\Response;3use Illuminate\Support\Facades\Gate;4 5Gate::define('edit-settings', function (User $user) {6 return $user->isAdmin7 ? Response::allow()8 : Response::deny('You must be an administrator.');9});
即使您从 Gate(守卫)返回授权响应,Gate::allows
方法仍将返回一个简单的布尔值;但是,您可以使用 Gate::inspect
方法来获取 Gate(守卫)返回的完整授权响应
1$response = Gate::inspect('edit-settings');2 3if ($response->allowed()) {4 // The action is authorized...5} else {6 echo $response->message();7}
当使用 Gate::authorize
方法时,如果操作未获得授权,该方法将抛出 AuthorizationException
,则授权响应提供的错误消息将传播到 HTTP 响应
1Gate::authorize('edit-settings');2 3// The action is authorized...
自定义 HTTP 响应状态
当通过 Gate(守卫)拒绝操作时,将返回 403
HTTP 响应;但是,有时返回替代 HTTP 状态代码可能很有用。您可以使用 Illuminate\Auth\Access\Response
类上的 denyWithStatus
静态构造函数自定义失败的授权检查返回的 HTTP 状态代码
1use App\Models\User;2use Illuminate\Auth\Access\Response;3use Illuminate\Support\Facades\Gate;4 5Gate::define('edit-settings', function (User $user) {6 return $user->isAdmin7 ? Response::allow()8 : Response::denyWithStatus(404);9});
由于通过 404
响应隐藏资源是 Web 应用程序的一种常见模式,因此为了方便起见,提供了 denyAsNotFound
方法
1use App\Models\User;2use Illuminate\Auth\Access\Response;3use Illuminate\Support\Facades\Gate;4 5Gate::define('edit-settings', function (User $user) {6 return $user->isAdmin7 ? Response::allow()8 : Response::denyAsNotFound();9});
拦截 Gate(守卫)检查
有时,您可能希望向特定用户授予所有能力。您可以使用 before
方法定义一个在所有其他授权检查之前运行的闭包
1use App\Models\User;2use Illuminate\Support\Facades\Gate;3 4Gate::before(function (User $user, string $ability) {5 if ($user->isAdministrator()) {6 return true;7 }8});
如果 before
闭包返回非 null 结果,则该结果将被视为授权检查的结果。
您可以使用 after
方法定义一个在所有其他授权检查之后执行的闭包
1use App\Models\User;2 3Gate::after(function (User $user, string $ability, bool|null $result, mixed $arguments) {4 if ($user->isAdministrator()) {5 return true;6 }7});
除非 Gate(守卫)或策略返回 null
,否则 after
闭包返回的值不会覆盖授权检查的结果。
内联授权
有时,您可能希望确定当前经过身份验证的用户是否有权执行给定的操作,而无需编写与该操作对应的专用 Gate(守卫)。Laravel 允许您通过 Gate::allowIf
和 Gate::denyIf
方法执行这些类型的“内联”授权检查。内联授权不会执行任何已定义的“before”或“after”授权钩子
1use App\Models\User;2use Illuminate\Support\Facades\Gate;3 4Gate::allowIf(fn (User $user) => $user->isAdministrator());5 6Gate::denyIf(fn (User $user) => $user->banned());
如果操作未获得授权,或者当前没有经过身份验证的用户,Laravel 将自动抛出 Illuminate\Auth\Access\AuthorizationException
异常。AuthorizationException
的实例会自动由 Laravel 的异常处理程序转换为 403 HTTP 响应。
创建策略
生成策略
策略是围绕特定模型或资源组织授权逻辑的类。例如,如果您的应用程序是一个博客,则您可能有一个 App\Models\Post
模型和一个对应的 App\Policies\PostPolicy
来授权用户操作,例如创建或更新帖子。
您可以使用 make:policy
Artisan 命令生成策略。生成的策略将放置在 app/Policies
目录中。如果此目录在您的应用程序中不存在,Laravel 将为您创建它
1php artisan make:policy PostPolicy
make:policy
命令将生成一个空的策略类。如果您想生成一个包含与查看、创建、更新和删除资源相关的示例策略方法的类,则可以在执行命令时提供 --model
选项
1php artisan make:policy PostPolicy --model=Post
注册策略
策略发现
默认情况下,只要模型和策略遵循标准的 Laravel 命名约定,Laravel 就会自动发现策略。具体来说,策略必须位于包含模型的目录或其上级目录的 Policies
目录中。因此,例如,模型可以放置在 app/Models
目录中,而策略可以放置在 app/Policies
目录中。在这种情况下,Laravel 将在 app/Models/Policies
然后 app/Policies
中检查策略。此外,策略名称必须与模型名称匹配,并具有 Policy
后缀。因此,User
模型将对应于 UserPolicy
策略类。
如果您想定义自己的策略发现逻辑,则可以使用 Gate::guessPolicyNamesUsing
方法注册自定义策略发现回调。通常,应从应用程序的 AppServiceProvider
的 boot
方法中调用此方法
1use Illuminate\Support\Facades\Gate;2 3Gate::guessPolicyNamesUsing(function (string $modelClass) {4 // Return the name of the policy class for the given model...5});
手动注册策略
使用 Gate
外观,您可以在应用程序的 AppServiceProvider
的 boot
方法中手动注册策略及其对应的模型
1use App\Models\Order; 2use App\Policies\OrderPolicy; 3use Illuminate\Support\Facades\Gate; 4 5/** 6 * Bootstrap any application services. 7 */ 8public function boot(): void 9{10 Gate::policy(Order::class, OrderPolicy::class);11}
编写策略
策略方法
注册策略类后,您可以为其授权的每个操作添加方法。例如,让我们在 PostPolicy
上定义一个 update
方法,该方法确定给定的 App\Models\User
是否可以更新给定的 App\Models\Post
实例。
update
方法将接收一个 User
和一个 Post
实例作为其参数,并且应返回 true
或 false
,指示用户是否有权更新给定的 Post
。因此,在此示例中,我们将验证用户的 id
是否与帖子上的 user_id
匹配
1<?php 2 3namespace App\Policies; 4 5use App\Models\Post; 6use App\Models\User; 7 8class PostPolicy 9{10 /**11 * Determine if the given post can be updated by the user.12 */13 public function update(User $user, Post $post): bool14 {15 return $user->id === $post->user_id;16 }17}
您可以根据需要继续在策略上定义其他方法,以用于其授权的各种操作。例如,您可以定义 view
或 delete
方法来授权各种与 Post
相关的操作,但请记住,您可以随意为策略方法指定任何名称。
如果您在通过 Artisan 控制台生成策略时使用了 --model
选项,则它将已经包含用于 viewAny
、view
、create
、update
、delete
、restore
和 forceDelete
操作的方法。
所有策略都通过 Laravel 服务容器 解析,允许您在策略的构造函数中类型提示任何需要的依赖项,以便自动注入它们。
策略响应
到目前为止,我们只检查了返回简单布尔值的策略方法。但是,有时您可能希望返回更详细的响应,包括错误消息。为此,您可以从您的策略方法返回 Illuminate\Auth\Access\Response
实例
1use App\Models\Post; 2use App\Models\User; 3use Illuminate\Auth\Access\Response; 4 5/** 6 * Determine if the given post can be updated by the user. 7 */ 8public function update(User $user, Post $post): Response 9{10 return $user->id === $post->user_id11 ? Response::allow()12 : Response::deny('You do not own this post.');13}
当从策略返回授权响应时,Gate::allows
方法仍将返回一个简单的布尔值;但是,您可以使用 Gate::inspect
方法来获取 Gate(守卫)返回的完整授权响应
1use Illuminate\Support\Facades\Gate;2 3$response = Gate::inspect('update', $post);4 5if ($response->allowed()) {6 // The action is authorized...7} else {8 echo $response->message();9}
当使用 Gate::authorize
方法时,如果操作未获得授权,该方法将抛出 AuthorizationException
,则授权响应提供的错误消息将传播到 HTTP 响应
1Gate::authorize('update', $post);2 3// The action is authorized...
自定义 HTTP 响应状态
当通过策略方法拒绝操作时,将返回 403
HTTP 响应;但是,有时返回替代 HTTP 状态代码可能很有用。您可以使用 Illuminate\Auth\Access\Response
类上的 denyWithStatus
静态构造函数自定义失败的授权检查返回的 HTTP 状态代码
1use App\Models\Post; 2use App\Models\User; 3use Illuminate\Auth\Access\Response; 4 5/** 6 * Determine if the given post can be updated by the user. 7 */ 8public function update(User $user, Post $post): Response 9{10 return $user->id === $post->user_id11 ? Response::allow()12 : Response::denyWithStatus(404);13}
由于通过 404
响应隐藏资源是 Web 应用程序的一种常见模式,因此为了方便起见,提供了 denyAsNotFound
方法
1use App\Models\Post; 2use App\Models\User; 3use Illuminate\Auth\Access\Response; 4 5/** 6 * Determine if the given post can be updated by the user. 7 */ 8public function update(User $user, Post $post): Response 9{10 return $user->id === $post->user_id11 ? Response::allow()12 : Response::denyAsNotFound();13}
不带模型的方法
某些策略方法仅接收当前经过身份验证的用户的实例。这种情况在授权 create
操作时最为常见。例如,如果您正在创建一个博客,您可能希望确定用户是否有权创建任何帖子。在这些情况下,您的策略方法应仅期望接收用户实例
1/**2 * Determine if the given user can create posts.3 */4public function create(User $user): bool5{6 return $user->role == 'writer';7}
访客用户
默认情况下,如果传入的 HTTP 请求不是由经过身份验证的用户发起的,则所有 Gate(守卫)和策略都会自动返回 false
。但是,您可以通过为用户参数定义声明“可选”类型提示或提供 null
默认值来允许这些授权检查传递到您的 Gate(守卫)和策略
1<?php 2 3namespace App\Policies; 4 5use App\Models\Post; 6use App\Models\User; 7 8class PostPolicy 9{10 /**11 * Determine if the given post can be updated by the user.12 */13 public function update(?User $user, Post $post): bool14 {15 return $user?->id === $post->user_id;16 }17}
策略过滤器
对于某些用户,您可能希望授权给定策略中的所有操作。为此,请在策略上定义 before
方法。before
方法将在策略上的任何其他方法之前执行,让您有机会在实际调用预期策略方法之前授权操作。此功能最常用于授权应用程序管理员执行任何操作
1use App\Models\User; 2 3/** 4 * Perform pre-authorization checks. 5 */ 6public function before(User $user, string $ability): bool|null 7{ 8 if ($user->isAdministrator()) { 9 return true;10 }11 12 return null;13}
如果您想拒绝特定类型用户的所有授权检查,则可以从 before
方法返回 false
。如果返回 null
,则授权检查将传递到策略方法。
如果策略类不包含与要检查的能力名称匹配的方法,则不会调用策略类的 before
方法。
使用策略授权操作
通过用户模型
您的 Laravel 应用程序附带的 App\Models\User
模型包含两个用于授权操作的有用方法:can
和 cannot
。can
和 cannot
方法接收您希望授权的操作名称和相关模型。例如,让我们确定用户是否有权更新给定的 App\Models\Post
模型。通常,这将在控制器方法中完成
1<?php 2 3namespace App\Http\Controllers; 4 5use App\Http\Controllers\Controller; 6use App\Models\Post; 7use Illuminate\Http\RedirectResponse; 8use Illuminate\Http\Request; 9 10class PostController extends Controller11{12 /**13 * Update the given post.14 */15 public function update(Request $request, Post $post): RedirectResponse16 {17 if ($request->user()->cannot('update', $post)) {18 abort(403);19 }20 21 // Update the post...22 23 return redirect('/posts');24 }25}
如果为给定模型注册了策略,则 can
方法将自动调用适当的策略并返回布尔结果。如果未为模型注册策略,则 can
方法将尝试调用与给定操作名称匹配的基于闭包的 Gate(守卫)。
不需要模型的操作
请记住,某些操作可能对应于不需要模型实例的策略方法,例如 create
。在这些情况下,您可以将类名传递给 can
方法。类名将用于确定在授权操作时要使用的策略
1<?php 2 3namespace App\Http\Controllers; 4 5use App\Http\Controllers\Controller; 6use App\Models\Post; 7use Illuminate\Http\RedirectResponse; 8use Illuminate\Http\Request; 9 10class PostController extends Controller11{12 /**13 * Create a post.14 */15 public function store(Request $request): RedirectResponse16 {17 if ($request->user()->cannot('create', Post::class)) {18 abort(403);19 }20 21 // Create the post...22 23 return redirect('/posts');24 }25}
通过 Gate
外观
除了 App\Models\User
模型提供的有用方法外,您始终可以通过 Gate
外观的 authorize
方法授权操作。
与 can
方法类似,此方法接受您希望授权的操作名称和相关模型。如果操作未获得授权,则 authorize
方法将抛出 Illuminate\Auth\Access\AuthorizationException
异常,Laravel 异常处理程序会自动将其转换为状态代码为 403 的 HTTP 响应
1<?php 2 3namespace App\Http\Controllers; 4 5use App\Http\Controllers\Controller; 6use App\Models\Post; 7use Illuminate\Http\RedirectResponse; 8use Illuminate\Http\Request; 9use Illuminate\Support\Facades\Gate;10 11class PostController extends Controller12{13 /**14 * Update the given blog post.15 *16 * @throws \Illuminate\Auth\Access\AuthorizationException17 */18 public function update(Request $request, Post $post): RedirectResponse19 {20 Gate::authorize('update', $post);21 22 // The current user can update the blog post...23 24 return redirect('/posts');25 }26}
不需要模型的操作
如前所述,某些策略方法(如 create
)不需要模型实例。在这些情况下,您应该将类名传递给 authorize
方法。类名将用于确定在授权操作时要使用的策略
1use App\Models\Post; 2use Illuminate\Http\RedirectResponse; 3use Illuminate\Http\Request; 4use Illuminate\Support\Facades\Gate; 5 6/** 7 * Create a new blog post. 8 * 9 * @throws \Illuminate\Auth\Access\AuthorizationException10 */11public function create(Request $request): RedirectResponse12{13 Gate::authorize('create', Post::class);14 15 // The current user can create blog posts...16 17 return redirect('/posts');18}
通过中间件
Laravel 包含一个中间件,可以在传入的请求甚至到达您的路由或控制器之前授权操作。默认情况下,可以使用 can
中间件别名将 Illuminate\Auth\Middleware\Authorize
中间件附加到路由,该别名由 Laravel 自动注册。让我们探讨一个使用 can
中间件授权用户可以更新帖子的示例
1use App\Models\Post;2 3Route::put('/post/{post}', function (Post $post) {4 // The current user may update the post...5})->middleware('can:update,post');
在此示例中,我们向 can
中间件传递了两个参数。第一个是我们希望授权的操作的名称,第二个是我们希望传递给策略方法的路由参数。在本例中,由于我们正在使用隐式模型绑定,因此 App\Models\Post
模型将传递给策略方法。如果用户未被授权执行给定的操作,中间件将返回状态代码为 403 的 HTTP 响应。
为了方便起见,您也可以使用 can
方法将 can
中间件附加到您的路由
1use App\Models\Post;2 3Route::put('/post/{post}', function (Post $post) {4 // The current user may update the post...5})->can('update', 'post');
不需要模型的操作
同样,某些策略方法(如 create
)不需要模型实例。在这些情况下,您可以将类名传递给中间件。类名将用于确定在授权操作时要使用的策略
1Route::post('/post', function () {2 // The current user may create posts...3})->middleware('can:create,App\Models\Post');
在字符串中间件定义中指定整个类名可能会变得很麻烦。因此,您可以选择使用 can
方法将 can
中间件附加到您的路由
1use App\Models\Post;2 3Route::post('/post', function () {4 // The current user may create posts...5})->can('create', Post::class);
通过 Blade 模板
在编写 Blade 模板时,您可能希望仅在用户被授权执行给定操作时才显示页面的某一部分。例如,您可能希望仅在用户可以实际更新帖子时才显示博客帖子的更新表单。在这种情况下,您可以使用 @can
和 @cannot
指令
1@can('update', $post) 2 <!-- The current user can update the post... --> 3@elsecan('create', App\Models\Post::class) 4 <!-- The current user can create new posts... --> 5@else 6 <!-- ... --> 7@endcan 8 9@cannot('update', $post)10 <!-- The current user cannot update the post... -->11@elsecannot('create', App\Models\Post::class)12 <!-- The current user cannot create new posts... -->13@endcannot
这些指令是编写 @if
和 @unless
语句的便捷快捷方式。上面的 @can
和 @cannot
语句等效于以下语句
1@if (Auth::user()->can('update', $post))2 <!-- The current user can update the post... -->3@endif4 5@unless (Auth::user()->can('update', $post))6 <!-- The current user cannot update the post... -->7@endunless
您还可以确定用户是否有权执行给定操作数组中的任何操作。为此,请使用 @canany
指令
1@canany(['update', 'view', 'delete'], $post)2 <!-- The current user can update, view, or delete the post... -->3@elsecanany(['create'], \App\Models\Post::class)4 <!-- The current user can create a post... -->5@endcanany
不需要模型的操作
与大多数其他授权方法一样,如果操作不需要模型实例,您可以将类名传递给 @can
和 @cannot
指令
1@can('create', App\Models\Post::class)2 <!-- The current user can create posts... -->3@endcan4 5@cannot('create', App\Models\Post::class)6 <!-- The current user can't create posts... -->7@endcannot
提供额外的上下文
当使用策略授权操作时,您可以将数组作为第二个参数传递给各种授权函数和助手函数。数组中的第一个元素将用于确定应调用哪个策略,而数组的其余元素将作为参数传递给策略方法,并且可以在进行授权决策时用于额外的上下文。例如,考虑以下包含附加的 $category
参数的 PostPolicy
方法定义
1/**2 * Determine if the given post can be updated by the user.3 */4public function update(User $user, Post $post, int $category): bool5{6 return $user->id === $post->user_id &&7 $user->canUpdateCategory($category);8}
当尝试确定经过身份验证的用户是否可以更新给定的帖子时,我们可以像这样调用此策略方法
1/** 2 * Update the given blog post. 3 * 4 * @throws \Illuminate\Auth\Access\AuthorizationException 5 */ 6public function update(Request $request, Post $post): RedirectResponse 7{ 8 Gate::authorize('update', [$post, $request->category]); 9 10 // The current user can update the blog post...11 12 return redirect('/posts');13}
授权 & Inertia
尽管授权必须始终在服务器上处理,但通常可以方便地为您的前端应用程序提供授权数据,以便正确呈现应用程序的 UI。Laravel 没有为向基于 Inertia 的前端公开授权信息定义必需的约定。
但是,如果您正在使用 Laravel 基于 Inertia 的入门套件之一,则您的应用程序已经包含 HandleInertiaRequests
中间件。在此中间件的 share
方法中,您可以返回共享数据,这些数据将提供给应用程序中的所有 Inertia 页面。此共享数据可以用作定义用户授权信息的便捷位置
1<?php 2 3namespace App\Http\Middleware; 4 5use App\Models\Post; 6use Illuminate\Http\Request; 7use Inertia\Middleware; 8 9class HandleInertiaRequests extends Middleware10{11 // ...12 13 /**14 * Define the props that are shared by default.15 *16 * @return array<string, mixed>17 */18 public function share(Request $request)19 {20 return [21 ...parent::share($request),22 'auth' => [23 'user' => $request->user(),24 'permissions' => [25 'post' => [26 'create' => $request->user()->can('create', Post::class),27 ],28 ],29 ],30 ];31 }32}