-
-
Notifications
You must be signed in to change notification settings - Fork 893
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Laravel] Define read-only model attributes #6875
Comments
just create the property on your model (it is essentially what laravel does, just using the magic property way), and apply it there. It should work as expected then. For example if you add this to your user model (
The E-Mail field will be available inside your read requests, but not inside, PUT,POST,PATCH
|
But it would be nice to use laravels "fillable" array here, it is used in some way right now i can see in the code but not for this purpose afaik. I'll happily try and add a PR for this tho. EDIT: Using the fillable array, would also get rid of setting createdAt, updatedAt, etc.., which is great, because those usually should never be filled, they are automatically populated. |
Indeed, This should be done at: core/src/Laravel/Eloquent/Metadata/Factory/Property/EloquentPropertyMetadataFactory.php Lines 89 to 92 in 57f15cf
In the meantime using |
Ok, it seems this works on custom attributes, but with "built-ins" such as id, created_at, updated at... there is not working well. Below is what it looks like in my case. I have an abstract parent class for attributes that available on every model and I need to use the #[
ApiResource(
operations: [
new GetCollection(name: 'tenants'),
new Post(name: 'tenants.create'),
],
rules: [
'companyName' => 'required|simple_text|max:128',
'firstname' => 'required|simple_text|max:32',
'lastname' => 'required|simple_text|max:32',
'username' => 'required|simple_text|max:32|unique:tenants',
'email' => 'required|email|max:64',
],
),
]
class Tenant extends AbstractModel
{
/** @use HasFactory<TenantFactory> */
use HasFactory;
#[ApiProperty(writable: false)]
public function slug(): Attribute
{
return Attribute::make(get: fn(?string $value) => $value);
}
/**
* The attributes that are mass assignable.
*
* @var array<int, string>
*/
protected $fillable = [
'company_name',
'firstname',
'lastname',
'username',
'email',
];
} abstract class AbstractModel extends Model
{
use HasUuids;
#[ApiProperty(writable: false)]
public function id(): Attribute
{
return Attribute::make(get: fn(string $value) => $value);
}
#[ApiProperty(writable: false)]
public function createdAt(): Attribute
{
return Attribute::make(get: fn(string $value) => $value);
}
#[ApiProperty(writable: false)]
public function updatedAt(): Attribute
{
return Attribute::make(get: fn(string $value) => $value);
}
} The "slug" attribute disappears at POST endpoint, but the others... not. |
You can specify
Because Eloquent properties are snake-cased I think you need the snake_case property to make this work. Can't dig too much into this right now but I'm sure that we'll come up with a solution! |
Ok, it's works: # ...
#[ApiProperty(writable: false, property: 'created_at')]
class Tenant extends AbstractModel
{
// ...
} But it's not: #[ApiProperty(writable: false, property: 'created_at')]
abstract class AbstractModel extends Model
{
// ...
} I don't want to apply it to each model separately, as it is the same for all of them. It would be nice if there was a solution for this. |
Ohh, I tried it this way and it seems good, I'm still testing it. abstract class AbstractModel extends Model
{
#[ApiProperty(writable: false)]
public \DateTime $created_at;
} |
The problem with this is that if I define the property, Laravel will no longer set its default value, so I lose the built-in automation that handles these fields. |
Closes: api-platform#6875 Signed-off-by: Tobias Oitzinger <[email protected]>
Closes: api-platform#6875 Signed-off-by: Tobias Oitzinger <[email protected]>
Closes: api-platform#6875 Signed-off-by: Tobias Oitzinger <[email protected]>
Closes: api-platform#6875 Signed-off-by: Tobias Oitzinger <[email protected]>
Closes: api-platform#6875 Signed-off-by: Tobias Oitzinger <[email protected]>
API Platform version(s) affected: 4.0.10
Description
I can't find a way to define read-only model attributes for an API, which are only accessible through GET endpoints (e.g., an ID field). According to the documentation here, it's possible to make properties read-only using the
ApiProperty
metadata class, where thereadable
andwritable
options can be configured. However, I don't understand where or how to apply this in Laravel. Laravel doesn't have explicit properties for table columns—there's only an 'attributes' array and accessors/mutators for accessing them. So where should I use theApiProperty
configuration?The text was updated successfully, but these errors were encountered: