Class: Metanorma::Ogc::Converter

Inherits:
Standoc::Converter
  • Object
show all
Defined in:
lib/metanorma/ogc/converter.rb,
lib/metanorma/ogc/front.rb,
lib/metanorma/ogc/cleanup.rb,
lib/metanorma/ogc/validate.rb

Overview

A Converter implementation that generates RSD output, and a document schema encapsulation of the document for validation

Constant Summary collapse

PUBLISHER =
"./contributor[role/@type = 'publisher']/organization".freeze
STANDARDTYPE =
%w{standard standard-with-suite abstract-specification
community-standard profile}.freeze
SEQ =

spec of permissible section sequence we skip normative references, it goes to end of list

[
  {
    msg: "Prefatory material must be followed by (clause) Scope",
    val: ["./self::clause[@type = 'scope']"],
  },
  {
    msg: "Scope must be followed by Conformance",
    val: ["./self::clause[@type = 'conformance']"],
  },
  {
    msg: "Normative References must be followed by " \
         "Terms and Definitions",
    val: ["./self::terms | .//terms"],
  },
].freeze
OGC_COLORS =
{
  "text": "rgb(88, 89, 91)",
  "secondary-shade-1": "rgb(237, 193, 35)",
  "secondary-shade-2": "rgb(246, 223, 140)",
  "background-definition-term": "rgb(215, 243, 255)",
  "background-definition-description": "rgb(242, 251, 255)",
  "text-title": "rgb(33, 55, 92)",
  "background-page": "rgb(33, 55, 92)",
  "background-text-label-legacy": "rgb(33, 60, 107)",
  "background-term-preferred-label": "rgb(249, 235, 187)",
  "background-term-deprecated-label": "rgb(237, 237, 238)",
  "background-term-admitted-label": "rgb(223, 236, 249)",
  "background-table-header": "rgb(33, 55, 92)",
  "background-table-row-even": "rgb(252, 246, 222)",
  "background-table-row-odd": "rgb(254, 252, 245)",
  "admonition-note": "rgb(79, 129, 189)",
  "admonition-tip": "rgb(79, 129, 189)",
  "admonition-editor": "rgb(79, 129, 189)",
  "admonition-important": "rgb(79, 129, 189)",
  "admonition-warning": "rgb(79, 129, 189)",
  "admonition-caution": "rgb(79, 129, 189)",
  "admonition-todo": "rgb(79, 129, 189)",
  "admonition-safety-precaution": "rgb(79, 129, 189)",
}.freeze

Instance Method Summary collapse

Instance Method Details

#add_idObject



21
22
23
# File 'lib/metanorma/ogc/cleanup.rb', line 21

def add_id
  %(id="_#{UUIDTools::UUID.random_create}")
end

#adoc2xml(text, flavour) ⇒ Object

preempt subdoctype warning



250
251
252
253
254
255
256
257
258
# File 'lib/metanorma/ogc/converter.rb', line 250

def adoc2xml(text, flavour)
  Nokogiri::XML(text).root and return text
  c = Asciidoctor
    .convert("= X\nA\n:semantic-metadata-headless: true\n" \
             ":novalid:\n:docsubtype: implementation\n" \
             ":doctype: standard\n\n#{text}\n",
             backend: flavour, header_footer: true)
  Nokogiri::XML(c).at("//xmlns:sections")
end

#author_title_key(pubclass, title, bib) ⇒ Object



191
192
193
194
195
196
197
198
199
# File 'lib/metanorma/ogc/cleanup.rb', line 191

def author_title_key(pubclass, title, bib)
  case pubclass
  when 1, 2 then title
  when 3
    cite = ::Relaton::Render::General.new
      .render_all("<references>#{bib.to_xml}</references>")
    cite[:author]
  end
end

#bibdata_cleanup(xmldoc) ⇒ Object



68
69
70
71
72
73
74
75
76
77
# File 'lib/metanorma/ogc/cleanup.rb', line 68

def bibdata_cleanup(xmldoc)
  super
  a = xmldoc.at("//bibdata/status/stage")
  a.text == "published" and a.children = "approved"
  if @doctype == "technical-paper"
    doctype = xmldoc.at("//bibdata/ext/doctype")
    doctype.children = "white-paper"
    @doctype = "white-paper"
  end
end

#bibdata_validate(doc) ⇒ Object



17
18
19
20
# File 'lib/metanorma/ogc/validate.rb', line 17

def bibdata_validate(doc)
  stage_validate(doc)
  version_validate(doc)
end

#boilerplate_file(_xmldoc) ⇒ Object



4
5
6
# File 'lib/metanorma/ogc/cleanup.rb', line 4

def boilerplate_file(_xmldoc)
  File.join(@libdir, "boilerplate.adoc")
end

#clause_parse(attrs, xml, node) ⇒ Object



108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
# File 'lib/metanorma/ogc/converter.rb', line 108

def clause_parse(attrs, xml, node)
  %w(overview future_outlook value_proposition
     contributors aims objectives topics outlook security)
    .include?(node.attr("type")) and
    attrs = attrs.merge(type: node.attr("type"))
  case node.attr("heading")&.downcase || node.title.downcase
  when "submitters"
    return submitters_parse(attrs.merge(type: "submitters"), xml, node)
  when "contributors"
    return submitters_parse(attrs.merge(type: "contributors"), xml, node)
  when "conformance" then attrs = attrs.merge(type: "conformance")
  when "security considerations"
    attrs = attrs.merge(type: "security")
  end
  super
end

#content_validate(doc) ⇒ Object



12
13
14
15
# File 'lib/metanorma/ogc/validate.rb', line 12

def content_validate(doc)
  super
  bibdata_validate(doc.root)
end

#corporate_author(node, xml) ⇒ Object



19
20
21
22
23
24
25
26
27
28
29
30
# File 'lib/metanorma/ogc/front.rb', line 19

def corporate_author(node, xml)
  node.attr("submitting-organizations") or return
  csv_split(@c.decode(node.attr("submitting-organizations")),
            ";")&.each do |org|
    xml.contributor do |c|
      c.role type: "author"
      c.organization do |a|
        safe_xml_string(a, "name", org)
      end
    end
  end
end

#create_security_clause(xml) ⇒ Object



42
43
44
45
46
47
48
49
50
51
52
# File 'lib/metanorma/ogc/cleanup.rb', line 42

def create_security_clause(xml)
  doctype = xml.at("//bibdata/ext/doctype")&.text
  description = "document"
  description = "standard" if %w(standard community-standard)
    .include?(doctype)
  "    <clause type='security' \#{add_id}>\n      <title>Security considerations</title>\n      <p>\#{@i18n.security_empty.sub('%', description)}</p></clause>\n  CLAUSE\nend\n"

#default_publisherObject



86
87
88
# File 'lib/metanorma/ogc/front.rb', line 86

def default_publisher
  "OGC"
end

#default_requirement_modelObject



26
27
28
# File 'lib/metanorma/ogc/converter.rb', line 26

def default_requirement_model
  "ogc"
end

#doc_converter(node) ⇒ Object



245
246
247
# File 'lib/metanorma/ogc/converter.rb', line 245

def doc_converter(node)
  IsoDoc::Ogc::WordConvert.new(doc_extract_attributes(node))
end

#doctype(node) ⇒ Object



39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
# File 'lib/metanorma/ogc/converter.rb', line 39

def doctype(node)
  d = super
  d1 = ::IsoDoc::Ogc::DOCTYPE_ABBR.invert[d] and d = d1
  unless %w{abstract-specification-topic best-practice other policy
            change-request-supporting-document community-practice
            community-standard discussion-paper engineering-report
            reference-model release-notes standard user-guide white-paper
            technical-paper test-suite draft-standard}.include? d
    @warned_doctype or
      @log.add("Document Attributes", nil,
               "'#{d}' is not a legal document type: reverting to 'standard'")
    @warned_doctype = true
    d = @default_doctype
  end
  d
end

#execsummary_validate(xmldoc) ⇒ Object



58
59
60
61
62
63
64
65
66
# File 'lib/metanorma/ogc/validate.rb', line 58

def execsummary_validate(xmldoc)
  sect = xmldoc.at("//executivesummary")
  @doctype == "engineering-report" && sect.nil? and
    @log.add("Style", nil,
             "Executive Summary required for Engineering Reports!")
  @doctype != "engineering-report" && !sect.nil? and
    @log.add("Style", nil,
             "Executive Summary only allowed for Engineering Reports!")
end

#externalid(node) ⇒ Object



113
114
115
116
117
118
119
120
121
# File 'lib/metanorma/ogc/front.rb', line 113

def externalid(node)
  i = node.attr("external-id") and return i
  d = doctype(node)
  a = node.attr("abbrev")
  d && a or return
  url = "http://www.opengis.net/doc/#{IsoDoc::Ogc::DOCTYPE_ABBR[d]}/#{a}"
  v = node.attr("edition") || node.attr("version") and url += "/#{v}"
  url
end

#externalurl(node) ⇒ Object



132
133
134
135
136
137
138
# File 'lib/metanorma/ogc/front.rb', line 132

def externalurl(node)
  if doctype(node) == "engineering-report"
    "http://www.opengis.net/doc/PER/t14-#{node.attr('referenceurlid')}"
  else
    node.attr("referenceurlid")
  end
end

#highlight_parsex(text, xml) ⇒ Object

KILL



167
168
169
# File 'lib/metanorma/ogc/converter.rb', line 167

def highlight_parsex(text, xml)
  xml.hi { |s| s << text }
end

#html_converter(node) ⇒ Object



235
236
237
# File 'lib/metanorma/ogc/converter.rb', line 235

def html_converter(node)
  IsoDoc::Ogc::HtmlConvert.new(html_extract_attributes(node))
end

#init_toc(node) ⇒ Object



16
17
18
19
20
21
# File 'lib/metanorma/ogc/converter.rb', line 16

def init_toc(node)
  super
  @tocfigures = true
  @toctables = true
  @tocrecommendations = true
end

#insert_security(xml, sect) ⇒ Object



25
26
27
28
29
30
31
32
33
34
35
# File 'lib/metanorma/ogc/cleanup.rb', line 25

def insert_security(xml, sect)
  "document"
  "standard" if %w(standard community-standard)
    .include?(@doctype)
  @doctype == "engineering-report" and return remove_security(xml)
  preface = sect.at("//preface") ||
    sect.add_previous_sibling("<preface/>").first
  sect = xml.at("//clause[@type = 'security']")&.remove ||
    create_security_clause(xml)
  preface.add_child sect
end

#insert_submitters(xml, sect) ⇒ Object



54
55
56
57
58
59
60
61
62
63
64
# File 'lib/metanorma/ogc/cleanup.rb', line 54

def insert_submitters(xml, sect)
  if xml.at("//clause[@type = 'submitters' or @type = 'contributors']")
    p = sect.at("//preface") ||
      sect.add_previous_sibling("<preface/>").first
    xml.xpath("//clause[@type = 'submitters' or @type = 'contributors']")
      .each do |s|
      s.xpath(".//table").each { |t| t["unnumbered"] = true }
      p.add_child s.remove
    end
  end
end

#make_preface(xml, sect) ⇒ Object



15
16
17
18
19
# File 'lib/metanorma/ogc/cleanup.rb', line 15

def make_preface(xml, sect)
  super
  insert_security(xml, sect)
  insert_submitters(xml, sect)
end

#makexml(node) ⇒ Object



34
35
36
37
# File 'lib/metanorma/ogc/converter.rb', line 34

def makexml(node)
  @draft = node.attributes.has_key?("draft")
  super
end

#metadata_attrs(node) ⇒ Object



206
207
208
209
210
211
212
213
214
# File 'lib/metanorma/ogc/converter.rb', line 206

def (node)
  c = update_colors(node)
  ret = super
  c.keys.sort.each do |k|
    ret += "<presentation-metadata><name>color-#{k}</name>" \
      "<value>#{c[k]}</value></presentation-metadata>"
  end
  ret
end

#metadata_author(node, xml) ⇒ Object



8
9
10
11
# File 'lib/metanorma/ogc/front.rb', line 8

def (node, xml)
  corporate_author(node, xml)
  personal_author(node, xml)
end

#metadata_committee(node, xml) ⇒ Object



94
95
96
97
98
99
# File 'lib/metanorma/ogc/front.rb', line 94

def (node, xml)
  node.attr("committee") or return
  xml.editorialgroup do |a|
    (node, a)
  end
end

#metadata_committee1(node, xml) ⇒ Object



101
102
103
104
105
106
107
108
109
110
111
# File 'lib/metanorma/ogc/front.rb', line 101

def (node, xml)
  xml.committee(node.attr("committee") || "technical")
  node.attr("subcommittee") and
    xml.subcommittee(node.attr("subcommittee"),
                     **attr_code(type: node.attr("subcommittee-type"),
                                 number: node.attr("subcommittee-number")))
  (node.attr("workgroup") || node.attr("workinggroup")) and
    xml.workgroup(node.attr("workgroup") || node.attr("workinggroup"),
                  **attr_code(type: node.attr("workgroup-type"),
                              number: node.attr("workgroup-number")))
end


146
147
148
149
150
# File 'lib/metanorma/ogc/front.rb', line 146

def (node, xml)
  node.attr("copyrightyear") and
    node.set_attr("copyright-year", node.attr("copyrightyear"))
  super
end

#metadata_date(node, xml) ⇒ Object



152
153
154
155
156
157
# File 'lib/metanorma/ogc/front.rb', line 152

def (node, xml)
  super
  ogc_date(node, xml, "submissiondate", "received")
  ogc_date(node, xml, "publicationdate", "published")
  ogc_date(node, xml, "approvaldate", "issued")
end

#metadata_id(node, xml) ⇒ Object



123
124
125
126
127
128
129
130
# File 'lib/metanorma/ogc/front.rb', line 123

def (node, xml)
  e = externalid(node) and xml.docidentifier e, type: "ogc-external"
  node.attr("referenceurlid") and
    xml.docidentifier externalurl(node), type: "ogc-external"
  docnumber = node.attr("docnumber") || node.attr("docreference")
  id = node.attr("docidentifier") || docnumber
  xml.docidentifier id, type: "ogc-internal", primary: "true"
end

#metadata_source(node, xml) ⇒ Object



140
141
142
143
144
# File 'lib/metanorma/ogc/front.rb', line 140

def (node, xml)
  super
  node.attr("previous-uri") && xml.uri(node.attr("previous-uri"),
                                       type: "previous")
end

#metadata_subdoctype(node, xml) ⇒ Object



173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
# File 'lib/metanorma/ogc/front.rb', line 173

def (node, xml)
  s = node.attr("docsubtype")
  s1 = ::IsoDoc::Ogc::DOCSUBTYPE_ABBR.invert[s] and s = s1
  case doctype(node)
  when "standard"
    unless %w{conceptual-model conceptual-model-and-encoding
              conceptual-model-and-implementation encoding extension
              implementation profile profile-with-extension}.include? s
      @log.add("Document Attributes", nil,
               "'#{s}' is not a permitted subtype of Standard: " \
               "reverting to 'implementation'")
      s = "implementation"
    end
  when "best-practice"
    unless %w{general encoding extension profile
              profile-with-extension}.include? s
      @log.add("Document Attributes", nil,
               "'#{s}' is not a permitted subtype of best-practice: " \
               "reverting to 'general'")
      s = "general"
    end
  end
  s && !s.empty? and xml.subdoctype s
end

#metadata_version(node, xml) ⇒ Object



167
168
169
170
171
# File 'lib/metanorma/ogc/front.rb', line 167

def (node, xml)
  node.attr("version") and
    node.set_attr("edition", node.attr("version"), false)
  super
end

#norm_ref_validate(doc) ⇒ Object



143
144
145
146
147
148
149
# File 'lib/metanorma/ogc/validate.rb', line 143

def norm_ref_validate(doc)
  @doctype == "engineering-report" or return super
  doc.xpath("//references[@normative = 'true']").each do |b|
    @log.add("Bibliography", b,
             "Engineering report should not contain normative references")
  end
end

#normref_cleanup(xmldoc) ⇒ Object



116
117
118
119
120
121
122
123
124
# File 'lib/metanorma/ogc/cleanup.rb', line 116

def normref_cleanup(xmldoc)
  r1 = xmldoc.at("//references[title[translate(text(), 'R', 'r') = " \
                 "'Normative references']]")
  r2 = xmldoc.at("//references[title[text() = 'References']]")
  if r1 && r2
    r2["normative"] = false
  end
  super
end

#obligations_cleanup_inherit(xml) ⇒ Object



126
127
128
129
130
131
132
133
134
135
136
# File 'lib/metanorma/ogc/cleanup.rb', line 126

def obligations_cleanup_inherit(xml)
  xml.xpath("//annex").each do |r|
    r["obligation"] = "informative" unless r["obligation"]
  end
  xml.xpath("//clause[not(ancestor::boilerplate)]").each do |r|
    r["obligation"] = "normative" unless r["obligation"]
  end
  xml.xpath(::Metanorma::Standoc::Utils::SUBCLAUSE_XPATH).each do |r|
    o = r.at("./ancestor::*/@obligation")&.text and r["obligation"] = o
  end
end

#ogc_date(node, xml, ogcname, metanormaname) ⇒ Object



159
160
161
162
163
164
165
# File 'lib/metanorma/ogc/front.rb', line 159

def ogc_date(node, xml, ogcname, metanormaname)
  if node.attr(ogcname)
    xml.date type: metanormaname do |d|
      d.on node.attr(ogcname)
    end
  end
end

#ogc_editor(node, xml) ⇒ Object



54
55
56
57
58
59
60
61
62
63
64
# File 'lib/metanorma/ogc/front.rb', line 54

def ogc_editor(node, xml)
  node.attr("editor") or return
  xml.contributor do |c|
    c.role type: "editor"
    c.person do |p|
      p.name do |n|
        n.completename node.attr("editor")
      end
    end
  end
end

#org_abbrevObject



90
91
92
# File 'lib/metanorma/ogc/front.rb', line 90

def org_abbrev
  { "Open Geospatial Consortium" => "OGC" }
end

#outputs(node, ret) ⇒ Object



97
98
99
100
101
102
103
104
105
106
# File 'lib/metanorma/ogc/converter.rb', line 97

def outputs(node, ret)
  File.open("#{@filename}.xml", "w:UTF-8") { |f| f.write(ret) }
  presentation_xml_converter(node).convert("#{@filename}.xml")
  html_converter(node).convert("#{@filename}.presentation.xml", nil,
                               false, "#{@filename}.html")
  doc_converter(node).convert("#{@filename}.presentation.xml", nil,
                              false, "#{@filename}.doc")
  pdf_converter(node)&.convert("#{@filename}.presentation.xml", nil,
                               false, "#{@filename}.pdf")
end

#override_style(node) ⇒ Object



69
70
71
72
73
74
75
76
77
78
# File 'lib/metanorma/ogc/converter.rb', line 69

def override_style(node)
  s = node.attr("style")
  if %w(overview future_outlook value_proposition contributors).include?(s)
    node.set_attr("style", "preface")
    node.set_attr("type", s)
  end
  if %w(aims objectives topics outlook security).include?(s)
    node.set_attr("type", s)
  end
end

#pdf_converter(node) ⇒ Object



239
240
241
242
243
# File 'lib/metanorma/ogc/converter.rb', line 239

def pdf_converter(node)
  return nil if node.attr("no-pdf")

  IsoDoc::Ogc::PdfConvert.new(pdf_extract_attributes(node))
end

#personal_author(node, xml) ⇒ Object



32
33
34
35
36
37
38
39
40
41
42
# File 'lib/metanorma/ogc/front.rb', line 32

def personal_author(node, xml)
  ogc_editor(node, xml)
  if node.attr("fullname") || node.attr("surname")
    personal_author1(node, xml, "")
  end
  i = 2
  while node.attr("fullname_#{i}") || node.attr("surname_#{i}")
    personal_author1(node, xml, "_#{i}")
    i += 1
  end
end

#personal_author1(node, xml, suffix) ⇒ Object



66
67
68
69
70
71
72
73
74
75
# File 'lib/metanorma/ogc/front.rb', line 66

def personal_author1(node, xml, suffix)
  xml.contributor do |c|
    personal_role(node, c, suffix)
    c.person do |p|
      p.name do |n|
        personal_author_name(node, n, suffix)
      end
    end
  end
end

#personal_author_name(node, xml, suffix) ⇒ Object



77
78
79
80
81
82
83
84
# File 'lib/metanorma/ogc/front.rb', line 77

def personal_author_name(node, xml, suffix)
  if node.attr("fullname#{suffix}")
    safe_xml_string(xml, "completename", node.attr("fullname#{suffix}"))
  else
    safe_xml_string(xml, "forename", node.attr("givenname#{suffix}"))
    safe_xml_string(xml, "surname", node.attr("surname#{suffix}"))
  end
end

#personal_role(node, contrib, suffix) ⇒ Object



44
45
46
47
48
49
50
51
52
# File 'lib/metanorma/ogc/front.rb', line 44

def personal_role(node, contrib, suffix)
  type = node.attr("role#{suffix}")&.downcase || "editor"
  if type == "contributor"
    contrib.role type: "author" do |r|
      safe_xml_string(r, "description", type)
    end
  else contrib.role type: type
  end
end

#preface_sequence_validate(root) ⇒ Object



128
129
130
131
132
133
134
135
136
137
138
139
140
141
# File 'lib/metanorma/ogc/validate.rb', line 128

def preface_sequence_validate(root)
  @doctype == "engineering-report" and return
  root.at("//preface/abstract") or @log.add("Style", nil,
                                            "Abstract is missing!")
  root.at("//bibdata/keyword | //bibdata/ext/keyword") or
    @log.add("Style", nil, "Keywords are missing!")
  root.at("//foreword") or @log.add("Style", nil,
                                    "Preface is missing!")
  root.at("//bibdata/contributor[role/@type = 'author']/organization/" \
          "name") or
    @log.add("Style", nil, "Submitting Organizations is missing!")
  root.at("//clause[@type = 'submitters' or @type = 'contributors']") or
    @log.add("Style", nil, "Submitters is missing!")
end

#presentation_xml_converter(node) ⇒ Object



229
230
231
232
233
# File 'lib/metanorma/ogc/converter.rb', line 229

def presentation_xml_converter(node)
  IsoDoc::Ogc::PresentationXMLConvert
    .new(html_extract_attributes(node)
    .merge(output_formats: ::Metanorma::Ogc::Processor.new.output_formats))
end

#pub_class(bib) ⇒ Object



161
162
163
164
165
166
167
168
169
# File 'lib/metanorma/ogc/cleanup.rb', line 161

def pub_class(bib)
  return 1 if bib.at("#{PUBLISHER}[abbreviation = 'OGC']")
  return 1 if bib.at("#{PUBLISHER}[name = 'Open Geospatial " \
                     "Consortium']")
  return 2 if bib.at("./docidentifier[@type][not(#{skip_docid} or " \
                     "@type = 'metanorma')]")

  3
end

#remove_security(xml) ⇒ Object



37
38
39
40
# File 'lib/metanorma/ogc/cleanup.rb', line 37

def remove_security(xml)
  a = xml.at("//clause[@type = 'security']") and
    a.delete("type")
end

#safe_xml_string(node, key, value) ⇒ Object



13
14
15
16
17
# File 'lib/metanorma/ogc/front.rb', line 13

def safe_xml_string(node, key, value)
  node.send key do |n|
    n << value
  end
end

#schema_fileObject



4
5
6
# File 'lib/metanorma/ogc/validate.rb', line 4

def schema_file
  "ogc.rng"
end

#section(node) ⇒ Object



64
65
66
67
# File 'lib/metanorma/ogc/converter.rb', line 64

def section(node)
  override_style(node)
  super
end

#section_names_terms_cleanup(xml) ⇒ Object



79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
# File 'lib/metanorma/ogc/cleanup.rb', line 79

def section_names_terms_cleanup(xml)
  @i18n or return
  replace_title(xml, "//definitions[@type = 'symbols']", @i18n.symbols)
  replace_title(xml, "//definitions[@type = 'abbreviated_terms']",
                @i18n.abbrev)
  replace_title(xml, "//definitions[not(@type)]", @i18n.symbolsabbrev)
  replace_title(xml, "//sections//terms#{SYMnoABBR} | " \
                     "//sections//clause[.//terms]#{SYMnoABBR}",
                @i18n.termsdefsymbols, true)
  replace_title(xml, "//sections//terms#{ABBRnoSYM} | " \
                     "//sections//clause[.//terms]#{ABBRnoSYM}",
                @i18n.termsdefabbrev, true)
  replace_title(xml, "//sections//terms#{SYMABBR} | " \
                     "//sections//clause[.//terms]#{SYMABBR}",
                @i18n.termsdefsymbolsabbrev, true)
  replace_title(xml, "//sections//terms#{NO_SYMABBR} | " \
                     "//sections//clause[.//terms]#{NO_SYMABBR}",
                @i18n.termsdefsymbolsabbrev, true)
  replace_title(xml, "//sections//terms[not(.//definitions)] | " \
                     "//sections//clause[.//terms][not(.//definitions)]",
                @i18n.termsdef, true)
end

#section_validate(doc) ⇒ Object



68
69
70
71
72
73
# File 'lib/metanorma/ogc/validate.rb', line 68

def section_validate(doc)
  preface_sequence_validate(doc.root)
  execsummary_validate(doc.root)
  sections_sequence_validate(doc.root)
  super
end

#sections_cleanup(xml) ⇒ Object



8
9
10
11
12
13
# File 'lib/metanorma/ogc/cleanup.rb', line 8

def sections_cleanup(xml)
  super
  xml.xpath("//*[@inline-header]").each do |h|
    h.delete("inline-header")
  end
end

#sections_order_cleanup(xml) ⇒ Object



138
139
140
141
# File 'lib/metanorma/ogc/cleanup.rb', line 138

def sections_order_cleanup(xml)
  super
  sort_annexes(xml)
end

#sections_sequence_validate(root) ⇒ Object



108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
# File 'lib/metanorma/ogc/validate.rb', line 108

def sections_sequence_validate(root)
  STANDARDTYPE.include?(@doctype) or return
  names = root.xpath("//sections/* | //bibliography/*")
  names = seqcheck(names, SEQ[0][:msg], SEQ[0][:val])
  names = seqcheck(names, SEQ[1][:msg], SEQ[1][:val])
  names = seqcheck(names, SEQ[2][:msg], SEQ[2][:val])
  n = names.shift
  if n&.at("./self::definitions")
    n = names.shift
  end
  if n.nil? || n.name != "clause"
    @log.add("Style", nil,
             "Document must contain at least one clause")
    return
  end
  root.at("//references | //clause[descendant::references]" \
          "[not(parent::clause)]") or
    @log.add("Style", nil, "Normative References are mandatory")
end

#sectiontype(node, level = true) ⇒ Object



56
57
58
59
60
61
62
# File 'lib/metanorma/ogc/converter.rb', line 56

def sectiontype(node, level = true)
  ret = sectiontype_streamline(sectiontype1(node))
  return ret if ret == "terms and definitions" &&
    node.attr("style") == "appendix" && node.level == 1

  super
end

#sectiontype1(node) ⇒ Object

legacy encoding



92
93
94
95
# File 'lib/metanorma/ogc/converter.rb', line 92

def sectiontype1(node)
  role_style(node, "executive_summary") and return "executivesummary"
  super
end

#sectiontype_streamline(ret) ⇒ Object



80
81
82
83
84
85
86
87
88
89
# File 'lib/metanorma/ogc/converter.rb', line 80

def sectiontype_streamline(ret)
  case ret
  when "preface" then "foreword"
  when "foreword", "introduction" then "donotrecognise-foreword"
  when "references" then "normative references"
  when "glossary" then "terms and definitions"
  else
    super
  end
end

#seqcheck(names, msg, accepted) ⇒ Object



97
98
99
100
101
102
103
104
105
106
# File 'lib/metanorma/ogc/validate.rb', line 97

def seqcheck(names, msg, accepted)
  n = names.shift
  return [] if n.nil?

  test = accepted.map { |a| n.at(a) }
  if test.all?(&:nil?)
    @log.add("Style", nil, msg)
  end
  names
end

#set_obligation(attrs, node) ⇒ Object



171
172
173
174
175
176
177
178
179
# File 'lib/metanorma/ogc/converter.rb', line 171

def set_obligation(attrs, node)
  if node.attr("style") == "appendix" && node.level == 1
    attrs[:obligation] = if node.attributes.has_key?("obligation")
                           node.attr("obligation")
                         else "informative"
                         end
  else super
  end
end

#sort_annexes(xml) ⇒ Object



143
144
145
146
147
148
149
150
151
# File 'lib/metanorma/ogc/cleanup.rb', line 143

def sort_annexes(xml)
  last = xml.at("//annex[last()]") or return
  last.next = "<sentinel/>" and last = last.next_element
  gl = xml.at("//annex[.//term]") and last.previous = gl.remove
  rev = xml.at("//annex[title[normalize-space(.) = 'Revision history']]") ||
    xml.at("//annex[title[normalize-space(.) = 'Revision History']]") and
    last.previous = rev.remove
  last.remove
end

#sort_biblio(bib) ⇒ Object



153
154
155
156
157
# File 'lib/metanorma/ogc/cleanup.rb', line 153

def sort_biblio(bib)
  bib.sort do |a, b|
    sort_biblio_key(a) <=> sort_biblio_key(b)
  end
end

#sort_biblio_ids_key(bib) ⇒ Object



207
208
209
210
211
212
213
214
215
216
217
# File 'lib/metanorma/ogc/cleanup.rb', line 207

def sort_biblio_ids_key(bib)
  id = bib.at("./docidentifier[@primary]") ||
    bib.at("./docidentifier[not(#{skip_docid} or @type = 'metanorma')]")
  metaid = bib.at("./docidentifier[@type = 'metanorma']")&.text
  /\d-(?<partid>\d+)/ =~ id&.text
  { id: id&.text,
    num: bib.at("./docnumber")&.text,
    abbrid: /^\[\d+\]$/.match?(metaid) ? metaid : nil,
    partid: partid&.to_i || 0,
    type: id ? id["type"] : nil }
end

#sort_biblio_key(bib) ⇒ Object

sort by: doc class (OGC, other standard (not DOI &c), other then standard class (docid class other than DOI &c) then if OGC, doc title else if other, authors then docnumber if present, numeric sort

else alphanumeric metanorma id (abbreviation)

then doc part number if present, numeric sort then doc id (not DOI &c) then title



179
180
181
182
183
184
185
186
187
188
189
# File 'lib/metanorma/ogc/cleanup.rb', line 179

def sort_biblio_key(bib)
  pubclass = pub_class(bib)
  ids = sort_biblio_ids_key(bib)
  title = title_key(bib)
  sortkey3 = author_title_key(pubclass, title, bib)
  num = if ids[:num].nil? then ids[:abbrid]
        else sprintf("%09d", ids[:num].to_i)
        end
  "#{pubclass} :: #{ids[:type]} :: #{sortkey3} :: #{num} :: " \
    "#{sprintf('%09d', ids[:partid])} :: #{ids[:id]} :: #{title}"
end

#stage_type_validate(stage, doctype) ⇒ Object



32
33
34
35
36
37
38
39
40
41
42
43
44
45
# File 'lib/metanorma/ogc/validate.rb', line 32

def stage_type_validate(stage, doctype)
  case doctype
  when "standard", "abstract-specification-topic", "draft-standard"
    %w(draft work-item-draft).include?(stage)
  when "community-standard"
    %w(draft swg-draft).include?(stage)
  when "best-practice", "community-practice"
    %w(draft swg-draft work-item-draft).include?(stage)
  else %w(swg-draft oab-review public-rfc tc-vote work-item-draft
          deprecated rescinded legacy).include?(stage)
  end and
    @log.add("Document Attributes", nil,
             "#{stage} is not an allowed status for #{doctype}")
end

#stage_validate(xmldoc) ⇒ Object



22
23
24
25
26
27
28
29
30
# File 'lib/metanorma/ogc/validate.rb', line 22

def stage_validate(xmldoc)
  @doctype == "engineering-report" and return
  stage = xmldoc&.at("//bibdata/status/stage")&.text
  %w(draft swg-draft oab-review public-rfc tc-vote work-item-draft
     approved deprecated retired rescinded legacy).include? stage or
    @log.add("Document Attributes", nil,
             "#{stage} is not a recognised status")
  stage_type_validate(stage, @doctype)
end

#style(_node, _text) ⇒ Object



136
137
138
# File 'lib/metanorma/ogc/converter.rb', line 136

def style(_node, _text)
  nil
end

#submitters_parse(attrs, xml, node) ⇒ Object



125
126
127
128
129
130
131
132
133
134
# File 'lib/metanorma/ogc/converter.rb', line 125

def submitters_parse(attrs, xml, node)
  title = @i18n.submitters
  doctype(node) == "engineering-report" ||
    attrs[:type] == "contributors" and
    title = @i18n.contributors_clause
  xml.clause **attr_code(attrs) do |xml_section|
    xml_section.title title
    xml_section << node.content
  end
end

#symbol_key(sym) ⇒ Object

Numbers sort before letters; we leave out using thorn to force that sort order. case insensitive



221
222
223
224
225
# File 'lib/metanorma/ogc/cleanup.rb', line 221

def symbol_key(sym)
  @c.decode(asciimath_key(sym).text)
    .gsub(/[\[\]{}<>()]/, "").gsub(/\s/m, "")
    .gsub(/[[:punct:]]|[_^]/, ":\\0").delete("`")
end

#symbols_cleanup(docxml) ⇒ Object



227
228
229
230
231
232
233
234
235
236
237
# File 'lib/metanorma/ogc/cleanup.rb', line 227

def symbols_cleanup(docxml)
  docxml.xpath("//definitions/dl").each do |dl|
    dl_out = extract_symbols_list(dl)
    dl_out.sort! do |a, b|
      a[:key].downcase <=> b[:key].downcase || a[:key] <=> b[:key] ||
        a[:dt] <=> b[:dt]
    end
    dl.children = dl_out.map { |d| d[:dt].to_s + d[:dd].to_s }.join("\n")
  end
  docxml
end

#table_cell(node, xml_tr, tblsec) ⇒ Object



148
149
150
151
# File 'lib/metanorma/ogc/converter.rb', line 148

def table_cell(node, xml_tr, tblsec)
  node.set_attr("valign", "middle")
  super
end

#term_def_parse(attrs, xml, node, _toplevel) ⇒ Object



140
141
142
143
144
145
146
# File 'lib/metanorma/ogc/converter.rb', line 140

def term_def_parse(attrs, xml, node, _toplevel)
  if node.attr("style") == "appendix" && node.level == 1
    terms_annex_parse(attrs, xml, node)
  else
    super
  end
end

#termdef_boilerplate_cleanup(xmldoc) ⇒ Object



66
# File 'lib/metanorma/ogc/cleanup.rb', line 66

def termdef_boilerplate_cleanup(xmldoc); end

#termdef_cleanup(xmldoc) ⇒ Object



102
103
104
105
# File 'lib/metanorma/ogc/cleanup.rb', line 102

def termdef_cleanup(xmldoc)
  super
  termdef_subclause_cleanup(xmldoc)
end

#termdef_subclause_cleanup(xmldoc) ⇒ Object

skip annex/terms/terms, which is empty node



108
109
110
111
112
113
114
# File 'lib/metanorma/ogc/cleanup.rb', line 108

def termdef_subclause_cleanup(xmldoc)
  xmldoc.xpath("//annex//clause[terms]").each do |t|
    t.xpath("./clause | ./terms | ./definitions").size == 1 or next
    t.children.each { |n| n.parent = t.parent }
    t.remove
  end
end

#terms_annex_parse(attrs, xml, node) ⇒ Object



153
154
155
156
157
158
159
160
161
162
163
164
# File 'lib/metanorma/ogc/converter.rb', line 153

def terms_annex_parse(attrs, xml, node)
  attrs1 = attrs.merge(id: "_#{UUIDTools::UUID.random_create}")
  xml.annex **attr_code(attrs1) do |xml_section|
    xml_section.title { |name| name << node.title }
    xml_section.terms **attr_code(attrs) do |terms|
      (s = node.attr("source")) && s.split(",").each do |s1|
        terms.termdocsource(nil, **attr_code(bibitemid: s1))
      end
      terms << node.content
    end
  end
end

#title(node, xml) ⇒ Object



198
199
200
201
202
203
# File 'lib/metanorma/ogc/front.rb', line 198

def title(node, xml)
  super
  at = { format: "text/plain", type: "abbrev" }
  a = node.attr("abbrev") and
    xml.title a, **attr_code(at)
end

#title_key(bib) ⇒ Object



201
202
203
204
205
# File 'lib/metanorma/ogc/cleanup.rb', line 201

def title_key(bib)
  title = bib.at("./title[@type = 'main']") ||
    bib.at("./title") || bib.at("./formattedref")
  title&.text&.sub!(/^(OGC|Open Geospatial Consortium)\b/, "")
end

#title_validate(_root) ⇒ Object



8
9
10
# File 'lib/metanorma/ogc/validate.rb', line 8

def title_validate(_root)
  nil
end

#toc(value) ⇒ Object

ignore, we generate ToC outside of asciidoctor



24
# File 'lib/metanorma/ogc/converter.rb', line 24

def toc(value); end

#update_colors(node) ⇒ Object



216
217
218
219
220
221
222
223
224
225
226
227
# File 'lib/metanorma/ogc/converter.rb', line 216

def update_colors(node)
  c = OGC_COLORS.dup
  if document_scheme(node) == "2022"
    c[:"secondary-shade-1"] = "rgb(0, 177, 255)"
    c[:"secondary-shade-2"] = "rgb(0, 177, 255)"
  end
  if node.attr("doctype") == "white-paper"
    c[:"text-title"] = "rgb(68, 84, 106)"
    c[:"background-page"] = "rgb(68, 84, 106)"
  end
  c
end

#version_validate(xmldoc) ⇒ Object



47
48
49
50
51
52
53
54
55
56
# File 'lib/metanorma/ogc/validate.rb', line 47

def version_validate(xmldoc)
  version = xmldoc.at("//bibdata/edition")&.text
  if %w(engineering-report discussion-paper).include? @doctype
    version.nil? or @log.add("Document Attributes", nil,
                             "Version not permitted for #{@doctype}")
  else
    version.nil? and @log.add("Document Attributes", nil,
                              "Version required for #{@doctype}")
  end
end