A one-to-one relationship involves a single parent entity which has a property pointing to a single child entity. The child entity can have a property pointing back to the parent (ie. is "bidirectional"), but it doesn't have to.
If the child points back to the parent, you can effectively think of the child entity as being the parent of that one-to-one relationship - so both the parent and the child have a one-to-one relationship with the other entity. Thus, you don't really have to worry about whether a relationship is unidirectional or bidirectional, as they are treated the same way.
What you do have to worry about though, is which side owns the relationship. This is easy to figure out, because the owning entity is stored in the database table which has the foreign key column. For example, suppose we have an entity called Customer and another entity called MarketingPreferences. The code for these entities looks like this:
classCustomer{publicint $id;publicstring $name;publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file.classMarketingPreferences{publicint $id;publicCustomer $customer;publicbool $allowEmail;publicbool $allowTextMessage;}
We store the values of the properties for these entities in two database tables, named customer and marketing_preferences:
customer
id
name
marketing_preferences
id
customer_id<-- Foreign key
allow_email
allow_text_message
The marketing_preferences table contains a foreign key column named customer_id which holds the primary key value of the customer table. Note that it is not necessary (nor would it be desirable) to store the id of the marketing_preferences table in the customer table as well (although we could do that instead of having customer_id in the marketing_preferences table).
So, in the above example, because the marketing_preferences table holds the foreign key that defines the one-to-one relationship, the MarketingPreferences entity owns the relationship. This means that the mapping between the database column and the property will need to go on the MarketingPreferences entity, not the Customer entity. Even so, on the Customer entity, we still need to tell Objectiphy where to find the mapping information, hence we use the mappedBy attribute in the mapping definition, as shown below.
Examples
Here are two example of how to set up the mapping for a one-to-one relationship between two entities that point to each other, using attributes or annotations (these examples are all equivalent; you only need to use one type of mapping provider, it is up to you which one).
Note that the sourceJoinColumn attribute (or @ORM\JoinColumn annotation if using Doctrine annotations) exists on the owning side (in this case the 'child' object - MarketingPreferences), and the other side (in this case, the 'parent', Customer) has a mappedBy attribute, telling Objectiphy where to look for the join column. The value of mappedBy must be a property name, not a column name!
Example 1: Minimal Mapping
The following example shows the bare minimum mapping information necessary, on the assumption that the foreign key in the owning table is the primary key of the owned table and that a left join is sufficient. If you are using non-standard columns, or require an inner join, you will need to define them in full as shown in example 2 (note, if using Doctrine annotations, you cannot specify the join type with an annotation).
useObjectiphy\Objectiphy\Table;useObjectiphy\Objectiphy\Column;useObjectiphy\Objectiphy\Relationship;#[Table(name:'customer')]classCustomer{ #[Column(isPrimaryKey: true)]publicint $id; #[Column]publicstring $name; #[Relationship( relationshipType:'one_to_one', childClassName:MarketingPreferences::class, mappedBy:'customer')]publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file.#[Table(name:'marketing_preferences')]classMarketingPreferences{ #[Column(isPrimaryKey: true)]publicint $id; #[Relationship( relationshipType:'one_to_one', childClassName:Customer::class)] publicCustomer $customer; #[Column]publicbool $allowEmail; #[Column]publicbool $allowTextMessage;}
useObjectiphy\Objectiphy\Table;useObjectiphy\Objectiphy\Column;useObjectiphy\Objectiphy\Relationship;/** * @Table(name="customer") */classCustomer{/** * @Column(isPrimaryKey=true) */publicint $id;/** * @Column */publicstring $name;/** * @Relationship( * relationshipType="one_to_one", * childClassName="MarketingPreferences", * mappedBy="customer" * ) */publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file./** * @Table(name="marketing_preferences") */classMarketingPreferences{/** * @Column(isPrimaryKey=true) */publicint $id;/** * @Relationship( * relationshipType="one_to_one", * childClassName="Customer" * ) */publicCustomer $customer;/** * @Column */publicbool $allowEmail;/** * @Column */publicbool $allowTextMessage;}
useDoctrine\ORM\Mappingas ORM;/** * @ORM\Table(name="customer") */classCustomer{/** * @ORM\Id * @ORM\Column(type="integer") * @ORM\GeneratedValue(strategy="AUTO") */publicint $id;/** * @ORM\Column() */publicstring $name;/** * @ORM\OneToOne( * targetEntity="MarketingPreferences", * mappedBy="customer" * ) */publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file./** * @ORM\Table(name="marketing_preferences") */classMarketingPreferences{/** * @ORM\Id * @ORM\Column(type="integer") * @ORM\GeneratedValue(strategy="AUTO") */publicint $id;/** * @ORM\OneToOne(targetEntity="Customer") */publicCustomer $customer;/** * @ORM\Column */publicbool $allowEmail;/** * @ORM\Column */publicbool $allowTextMessage;}
Example 2: Full Mapping
The following example shows all of the mapping information defined, even though some items could be omitted as Objectiphy can guess some of the mapping if it follows standard conventions. You only need to specify all of the mapping information if you are not joining in a standard way from foreign key to primary key, if your foreign key column name is not suffixed with 'id', or the primary key of the target table, or if you need an inner join (note, if using Doctrine annotations, you cannot specify the join type with an annotation). Example 1, above, shows the exact same mapping as this in an abbreviated format.
useObjectiphy\Objectiphy\Table;useObjectiphy\Objectiphy\Column;useObjectiphy\Objectiphy\Relationship;#[Table(name:'customer')]classCustomer{ #[Column(isPrimaryKey: true)]publicint $id; #[Column] publicstring $name; #[Relationship( relationshipType:'one_to_one', childClassName:MarketingPreferences::class, mappedBy:'customer')]publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file.#[Table(name:'marketing_preferences')]classMarketingPreferences{ #[Column(isPrimaryKey: true)] publicint $id; #[Relationship( relationshipType:'one_to_one', childClassName:Customer::class, sourceJoinColumn:'customer_id', targetJoinColumn:'id', joinType:'LEFT')]publicCustomer $customer; #[Column]publicbool $allowEmail; #[Column]publicbool $allowTextMessage;}
useObjectiphy\Objectiphy\Table;useObjectiphy\Objectiphy\Column;useObjectiphy\Objectiphy\Relationship;/** * @Table(name="customer") */classCustomer{/** * @Column(isPrimaryKey=true) */publicint $id;/** * @Column */publicstring $name;/** * @Relationship( * relationshipType="one_to_one", * childClassName="MarketingPreferences", * mappedBy="customer" * ) */publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file./** * @Table(name="marketing_preferences") */classMarketingPreferences{/** * @Column(isPrimaryKey=true) */publicint $id;/** * @Relationship( * relationshipType="one_to_one", * childClassName="Customer", * sourceJoinColumn="customer_id", * targetJoinColumn="id", * joinType="LEFT" * ) */publicCustomer $customer;/** * @Column */publicbool $allowEmail;/** * @Column */publicbool $allowTextMessage;}
useDoctrine\ORM\Mappingas ORM;/** * @ORM\Table(name="customer") */classCustomer{/** * @ORM\Id * @ORM\Column(type="integer") * @ORM\GeneratedValue(strategy="AUTO") */publicint $id;/** * @ORM\Column() */publicstring $name;/** * @ORM\OneToOne( * targetEntity="MarketingPreferences", * mappedBy="customer" * ) */publicMarketingPreferences $preferences;}// Note: Each class would of course normally be// in a separate file./** * @ORM\Table(name="marketing_preferences") */classMarketingPreferences{/** * @ORM\Id * @ORM\Column(type="integer") * @ORM\GeneratedValue(strategy="AUTO") */publicint $id;/** * @ORM\OneToOne(targetEntity="Customer") * @ORM\JoinColumn( * name="customer_id", * referencedColumnName="id" * ) */publicCustomer $customer;/** * @ORM\Column */publicbool $allowEmail;/** * @ORM\Column */publicbool $allowTextMessage;}