Already used as many ORMs as you can imagine. At moment I'm in a love / hate crush with RedBean PHP. Here we go... after a few hours studying it I got a doubt about whats the better way to solve this very basic problem (best way means, in this case, the way that better fits to the RedBean's ease of use philosophy):
It's very common to limit access to some properties of our classes so we can prevent certain kinds of wrong data manipulation. This is usually accomplished with good use of getters and setters. But as far as already know about RedBean, there are no formal setters in the native classes, only some public properties that can be changed and persisted in the database.
What I would like to do is to protect some properties from being changed manually, so I can avoid other programmer to make any kind of weirdness like:
$beam->insertion_date = 'yesterday';
R::store($beam);
That field should never be changed after the row insertion, obviously, but we can't just trust no one will do that. Is there a way to achieve something like turning the insertion_date a protected property or making it inaccessible in some way?
I have a feeling that the best way to do that is using $beam->setMetadata()
and declare that a given property shouldn't be changed, but I don't know how to achieve this in RedBean and still couldn't find enough information in the official manual. Any help is appreciated.
Thanks for reading.
Found a solution!
Before put things back in the database, test if the values that shouldn't have been changed were changed programatically.
This is how the example model should look now:
class Model_book extends RedBean_SimpleModel{
protected $before = false;
public function open(){
if($this->bean->id){
$this->before = clone($this->bean);
}
}
public function update(){
if($this->bean->id){
if($this->before->insertion_date != $this->bean->insertion_date){
throw new exception ("The insertion_date can't be changed.");
}
}
}
}
So now the following code would result in exception:
$beam->insertion_date = 'yesterday';
R::store($beam);
Note: the use of RedBean's meta information API would be a better solution, but it wouldn't make sense to add (only) this validation criteria without add a complete validation layer on top of RedBean. On the other hand, add a custom full validation would become useless in case a built-in validation get added to the ORM in the near future (I bet it will happen), so I'll leave it as it is for now.
Hope that helps.