This file is indexed.

/usr/share/perl5/Tangram/Type/Ref/FromMany.pod is in libtangram-perl 2.10-2.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
=head1 NAME

Tangram::Type::Ref::FromMany - map references to persistent objects

=head1 SYNOPSIS

   use Tangram;

   # or
   use Tangram::Core;
   use Tangram::Type::Ref::FromMany;

   $schema = Tangram::Schema->new(
       classes => { Company => { fields => {
         ref => [ qw( ceo vice_ceo ) ]

   # or

   $schema = Tangram::Schema->new(
       classes => { Company => { fields => {
         ref =>
         {
            ceo =>
            {
               col => 'ceo',
               null => 0
            },

            vice_ceo =>
            {
               ...
            }
      

=head1 DESCRIPTION

This class is responsible for mapping fields that contain a reference
to a Perl object. The persistent fields are grouped in a hash under the C<ref>
key in the field hash.

The target object must belong to a persistent class.

Tangram uses a column on the source object to store the id of the
target object.

The persistent fields may be specified either as a hash or as an array
of field names.

In the hash form, each entry consists in a field name and an
associated option hash. The option hash may contain the following
fields:

=over 4

=item * aggreg

=item * col

=item * type_col

=item * class

=item * null

=item * deep_update

=back

Optional field C<aggreg> specifies that the referenced element (if
any) must be removed (erased) from persistent storage along with the
source object. The default is not to aggregate.

C<col> sets the name of the column that contains the target object's
id. This field is optional, it default to the persistent field
name. You need to specify a column name if the field name is not an
allowed SQL column identifier.

C<type_col> sets the name of the second column, that contains the
target object's type id. This field is optional, it default to the
value of C<col> with "C<_type>" appended.  Set to the empty string
(C<"">) to disable this column, which requires that you also specify
a class.

C<class> specifies the base class of the reference, which
automatically disables C<type_col> (currently unimplemented).

C<null>, if present and set to true, directs deploy() to generate SQL
code that allows NULL values for that column.

Optional field C<deep_update> specificies that the target object has
to be updated automatically when C<update> is called on the source
object. Automatic update ensures consisitency between the Perl
representation and the DBMS state, but degrades update performance so
use it with caution. The default is not to do automatic updates.

B<warning>: there is currently something of a bug with all deep_update
collections.  If you (for instance) insert an object, and a property
of that object that is deep_update, then sometimes things end up
getting inserted twice - especially in mapping tables.  This is
currently under investigation.

You may also pass the names of the persistent fields in an array, in
which case Tangram uses the field names as column names and allows
NULL values.

=head1 IMPLEMENTATION NOTES

=head2 INHERITANCE

Ref directly inherits from Tangram::Type::Scalar, and indirectly from Type.

=head2 STORING REFERENCES

When Tangram stores a reference to another object, it stores the
target object's OID in the source object's table, just like what
happens with other scalar types like String and Number.

At least that's what I<basically> happens, but there can be
complications.

Complication #1: the target object is not persistent yet. Thus it
doesn't have an OID yet. Tangram will attempt to store the target
object; if this succeeds, the target object gets an OID too and
Tangram can proceed.

Complication #2: Tangram may detect that the target object is already
being saved; this happens in presence of cycles. Let's take an
example:

        $homer = Person->new();
        $marge = Person->new();
        $homer->{partner} = $marge;
        $marge->{partner} = $homer;

        $storage->insert( $homer );

What happens here? Tangram detects that $homer refers to $marge, and
that $marge is not yet persistent. So it ignores $homer for a while,
and proceeds to storing $marge.

In the process, Tangram sees that $marge refers to $homer - which is
not persistent yet! its insert() is suspended. Fortunately Tangram
realizes that, and doesn't attempt to store $homer again. Instead it
stores a NULL in the 'partner' column of Marge's row.

Tangram also schedules an UPDATE statement that will be executed just
before returning from the call to insert(). That statement will patch
Marge's 'partner' column with the proper OID.

=head2 LOADING REFERENCES

Tangram never loads the target object in the process of retrieving the
source object. Doing so could have disastrous consequences. For
example, consider a family tree, where each Person has a reference to
Mom and Dad. Pulling any single Person would eventually retrieve
everybody up to Adam and Eve! If Tangram did work that way, that is...

Instead, Tangram ties all the fields that are declared as outgoing
references to a package (Tangram::Lazy::Ref). The underlying object
keeps track of the source object's OID, the Storage object, and the
name of the persistent field.

When - if - a reference field is accessed, the target object is pulled
from the database (if it's not already present in memory), and the
field is untied and its value is replaced with a reference to the
target object. As a result, the target object is loaded on demand, but
in an almost transparent fashion (why almost? because you can always
use tied() on the field and detect that strange things are taking place).