OK, znalazłem rozwiązanie:
/**
* User
*
* @ORM\Table("users")
* @ORM\Entity
*/
class User extends UserEntity
{
...
/**
* @ORM\OneToOne
* (
* targetEntity="UserPreferences",
* cascade={"persist", "remove"},
* inversedBy="user"
* )
*/
protected $userPreferences;
}
/**
* @ORM\Table("user_preferences")
* @ORM\Entity
*/
class UserPreferences extends UserPreferencesEntity
{
/**
* @ORM\Column(name="id", type="integer")
* @ORM\Id
* @ORM\GeneratedValue
*/
protected $id;
/**
* @var int
*
* @ORM\OneToOne(targetEntity="User", mappedBy="id", cascade={"persist", "remove"})
*/
protected $user;
...
}
Przede wszystkim musiałem określić mappedBy i inversedBy (co próbowałem już wcześniej, ale w złym kierunku - mappedBy po stronie właściciela, inversedBy po stronie odwróconej). Pomyślałem też, że strona odwrócona nie musi mieć oddzielnego identyfikatora i próbowałem użyć identyfikatora strony będącej właścicielem (User#id) jako klucza podstawowego również dla tego.
- http://docs.doctrine-project. org/en/latest/reference/unitofwork-associations.html
- http://docs.doctrine-project. org/en/latest/reference/association-mapping.html