Merge remote-tracking branch 'upstream/pull/2890'
[rails.git] / app / models / old_relation.rb
1 # == Schema Information
2 #
3 # Table name: relations
4 #
5 #  relation_id  :bigint(8)        default(0), not null, primary key
6 #  changeset_id :bigint(8)        not null
7 #  timestamp    :datetime         not null
8 #  version      :bigint(8)        not null, primary key
9 #  visible      :boolean          default(TRUE), not null
10 #  redaction_id :integer
11 #
12 # Indexes
13 #
14 #  relations_changeset_id_idx  (changeset_id)
15 #  relations_timestamp_idx     (timestamp)
16 #
17 # Foreign Keys
18 #
19 #  relations_changeset_id_fkey  (changeset_id => changesets.id)
20 #  relations_redaction_id_fkey  (redaction_id => redactions.id)
21 #
22
23 class OldRelation < ApplicationRecord
24   include ConsistencyValidations
25   include ObjectMetadata
26
27   self.table_name = "relations"
28   self.primary_keys = "relation_id", "version"
29
30   # note this needs to be included after the table name changes, or
31   # the queries generated by Redactable will use the wrong table name.
32   include Redactable
33
34   belongs_to :changeset
35   belongs_to :redaction
36   belongs_to :current_relation, :class_name => "Relation", :foreign_key => "relation_id"
37
38   has_many :old_members, -> { order(:sequence_id) }, :class_name => "OldRelationMember", :foreign_key => [:relation_id, :version]
39   has_many :old_tags, :class_name => "OldRelationTag", :foreign_key => [:relation_id, :version]
40
41   validates :changeset, :presence => true, :associated => true
42   validates :timestamp, :presence => true
43   validates :visible, :inclusion => [true, false]
44
45   def self.from_relation(relation)
46     old_relation = OldRelation.new
47     old_relation.visible = relation.visible
48     old_relation.changeset_id = relation.changeset_id
49     old_relation.timestamp = relation.timestamp
50     old_relation.relation_id = relation.id
51     old_relation.version = relation.version
52     old_relation.members = relation.members
53     old_relation.tags = relation.tags
54     old_relation
55   end
56
57   def save_with_dependencies!
58     save!
59
60     tags.each do |k, v|
61       tag = OldRelationTag.new
62       tag.k = k
63       tag.v = v
64       tag.relation_id = relation_id
65       tag.version = version
66       tag.save!
67     end
68
69     members.each_with_index do |m, i|
70       member = OldRelationMember.new
71       member.id = [relation_id, version, i]
72       member.member_type = m[0].classify
73       member.member_id = m[1]
74       member.member_role = m[2]
75       member.save!
76     end
77   end
78
79   def members
80     @members ||= old_members.collect do |member|
81       [member.member_type, member.member_id, member.member_role]
82     end
83   end
84
85   def tags
86     @tags ||= Hash[old_tags.collect { |t| [t.k, t.v] }]
87   end
88
89   attr_writer :members, :tags
90
91   def to_xml
92     doc = OSM::API.new.get_xml_doc
93     doc.root << to_xml_node
94     doc
95   end
96
97   def to_xml_node(changeset_cache = {}, user_display_name_cache = {})
98     el = XML::Node.new "relation"
99     el["id"] = relation_id.to_s
100
101     add_metadata_to_xml_node(el, self, changeset_cache, user_display_name_cache)
102
103     old_members.each do |member|
104       member_el = XML::Node.new "member"
105       member_el["type"] = member.member_type.to_s.downcase
106       member_el["ref"] = member.member_id.to_s # "id" is considered uncool here as it should be unique in XML
107       member_el["role"] = member.member_role.to_s
108       el << member_el
109     end
110
111     add_tags_to_xml_node(el, old_tags)
112
113     el
114   end
115
116   # Temporary method to match interface to relations
117   def relation_members
118     old_members
119   end
120
121   # Pretend we're not in any relations
122   def containing_relation_members
123     []
124   end
125
126   # check whether this element is the latest version - that is,
127   # has the same version as its "current" counterpart.
128   def is_latest_version?
129     current_relation.version == version
130   end
131 end