metanorma-standoc 1.11.1 → 2.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (146) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/rake.yml +3 -31
  3. data/.gitignore +23 -0
  4. data/Gemfile +2 -0
  5. data/lib/asciidoctor/standoc/base.rb +2 -241
  6. data/lib/asciidoctor/standoc/blocks.rb +2 -238
  7. data/lib/asciidoctor/standoc/blocks_notes.rb +2 -100
  8. data/lib/asciidoctor/standoc/cleanup.rb +2 -207
  9. data/lib/asciidoctor/standoc/cleanup_amend.rb +2 -53
  10. data/lib/asciidoctor/standoc/cleanup_block.rb +2 -173
  11. data/lib/asciidoctor/standoc/cleanup_boilerplate.rb +2 -204
  12. data/lib/asciidoctor/standoc/cleanup_footnotes.rb +2 -108
  13. data/lib/asciidoctor/standoc/cleanup_image.rb +2 -69
  14. data/lib/asciidoctor/standoc/cleanup_inline.rb +2 -179
  15. data/lib/asciidoctor/standoc/cleanup_maths.rb +2 -221
  16. data/lib/asciidoctor/standoc/cleanup_ref.rb +2 -169
  17. data/lib/asciidoctor/standoc/cleanup_ref_dl.rb +2 -93
  18. data/lib/asciidoctor/standoc/cleanup_reqt.rb +2 -110
  19. data/lib/asciidoctor/standoc/cleanup_section.rb +2 -184
  20. data/lib/asciidoctor/standoc/cleanup_section_names.rb +2 -91
  21. data/lib/asciidoctor/standoc/cleanup_symbols.rb +2 -47
  22. data/lib/asciidoctor/standoc/cleanup_table.rb +2 -67
  23. data/lib/asciidoctor/standoc/cleanup_terms.rb +2 -113
  24. data/lib/asciidoctor/standoc/cleanup_terms_designations.rb +2 -161
  25. data/lib/asciidoctor/standoc/cleanup_text.rb +2 -95
  26. data/lib/asciidoctor/standoc/cleanup_toc.rb +3 -0
  27. data/lib/asciidoctor/standoc/cleanup_xref.rb +2 -106
  28. data/lib/asciidoctor/standoc/converter.rb +2 -123
  29. data/lib/asciidoctor/standoc/datamodel/attributes_table_preprocessor.rb +2 -56
  30. data/lib/asciidoctor/standoc/datamodel/diagram_preprocessor.rb +2 -102
  31. data/lib/asciidoctor/standoc/datamodel/plantuml_renderer.rb +3 -404
  32. data/lib/asciidoctor/standoc/deprecated.rb +5 -0
  33. data/lib/asciidoctor/standoc/front.rb +2 -219
  34. data/lib/asciidoctor/standoc/front_contributor.rb +2 -191
  35. data/lib/asciidoctor/standoc/inline.rb +2 -229
  36. data/lib/asciidoctor/standoc/lists.rb +2 -119
  37. data/lib/asciidoctor/standoc/macros.rb +2 -203
  38. data/lib/asciidoctor/standoc/macros_form.rb +2 -62
  39. data/lib/asciidoctor/standoc/macros_note.rb +2 -44
  40. data/lib/asciidoctor/standoc/macros_plantuml.rb +2 -112
  41. data/lib/asciidoctor/standoc/macros_terms.rb +2 -180
  42. data/lib/asciidoctor/standoc/ref.rb +2 -225
  43. data/lib/asciidoctor/standoc/ref_sect.rb +2 -143
  44. data/lib/asciidoctor/standoc/ref_utility.rb +2 -0
  45. data/lib/asciidoctor/standoc/render.rb +3 -0
  46. data/lib/asciidoctor/standoc/reqt.rb +2 -89
  47. data/lib/asciidoctor/standoc/section.rb +2 -190
  48. data/lib/asciidoctor/standoc/table.rb +2 -84
  49. data/lib/asciidoctor/standoc/term_lookup_cleanup.rb +2 -178
  50. data/lib/asciidoctor/standoc/terms.rb +2 -153
  51. data/lib/asciidoctor/standoc/utils.rb +2 -116
  52. data/lib/asciidoctor/standoc/validate.rb +2 -157
  53. data/lib/asciidoctor/standoc/validate_section.rb +2 -54
  54. data/lib/isodoc/html/htmlstyle.css +20 -11
  55. data/lib/isodoc/html/htmlstyle.scss +11 -11
  56. data/lib/metanorma/standoc/base.rb +149 -0
  57. data/lib/{asciidoctor → metanorma}/standoc/basicdoc.rng +0 -0
  58. data/lib/{asciidoctor → metanorma}/standoc/biblio.rng +0 -0
  59. data/lib/metanorma/standoc/blocks.rb +239 -0
  60. data/lib/metanorma/standoc/blocks_notes.rb +101 -0
  61. data/lib/metanorma/standoc/cleanup.rb +146 -0
  62. data/lib/metanorma/standoc/cleanup_amend.rb +54 -0
  63. data/lib/metanorma/standoc/cleanup_block.rb +173 -0
  64. data/lib/metanorma/standoc/cleanup_boilerplate.rb +213 -0
  65. data/lib/metanorma/standoc/cleanup_footnotes.rb +109 -0
  66. data/lib/metanorma/standoc/cleanup_image.rb +70 -0
  67. data/lib/metanorma/standoc/cleanup_inline.rb +190 -0
  68. data/lib/metanorma/standoc/cleanup_maths.rb +222 -0
  69. data/lib/metanorma/standoc/cleanup_ref.rb +170 -0
  70. data/lib/metanorma/standoc/cleanup_ref_dl.rb +104 -0
  71. data/lib/metanorma/standoc/cleanup_reqt.rb +111 -0
  72. data/lib/metanorma/standoc/cleanup_section.rb +212 -0
  73. data/lib/metanorma/standoc/cleanup_section_names.rb +92 -0
  74. data/lib/metanorma/standoc/cleanup_symbols.rb +48 -0
  75. data/lib/metanorma/standoc/cleanup_table.rb +68 -0
  76. data/lib/metanorma/standoc/cleanup_terms.rb +140 -0
  77. data/lib/metanorma/standoc/cleanup_terms_designations.rb +199 -0
  78. data/lib/metanorma/standoc/cleanup_text.rb +96 -0
  79. data/lib/metanorma/standoc/cleanup_toc.rb +98 -0
  80. data/lib/metanorma/standoc/cleanup_xref.rb +107 -0
  81. data/lib/metanorma/standoc/converter.rb +124 -0
  82. data/lib/metanorma/standoc/datamodel/attributes_table_preprocessor.rb +57 -0
  83. data/lib/metanorma/standoc/datamodel/diagram_preprocessor.rb +103 -0
  84. data/lib/metanorma/standoc/datamodel/plantuml_renderer.rb +409 -0
  85. data/lib/metanorma/standoc/front.rb +224 -0
  86. data/lib/metanorma/standoc/front_contributor.rb +192 -0
  87. data/lib/metanorma/standoc/inline.rb +232 -0
  88. data/lib/{asciidoctor → metanorma}/standoc/isodoc.rng +90 -18
  89. data/lib/metanorma/standoc/lists.rb +120 -0
  90. data/lib/metanorma/standoc/macros.rb +204 -0
  91. data/lib/metanorma/standoc/macros_form.rb +63 -0
  92. data/lib/metanorma/standoc/macros_note.rb +45 -0
  93. data/lib/metanorma/standoc/macros_plantuml.rb +113 -0
  94. data/lib/metanorma/standoc/macros_terms.rb +181 -0
  95. data/lib/metanorma/standoc/ref.rb +243 -0
  96. data/lib/metanorma/standoc/ref_sect.rb +153 -0
  97. data/lib/metanorma/standoc/ref_utility.rb +129 -0
  98. data/lib/metanorma/standoc/render.rb +115 -0
  99. data/lib/metanorma/standoc/reqt.rb +90 -0
  100. data/lib/{asciidoctor → metanorma}/standoc/reqt.rng +0 -0
  101. data/lib/metanorma/standoc/section.rb +209 -0
  102. data/lib/metanorma/standoc/table.rb +85 -0
  103. data/lib/metanorma/standoc/term_lookup_cleanup.rb +179 -0
  104. data/lib/metanorma/standoc/terms.rb +160 -0
  105. data/lib/metanorma/standoc/utils.rb +101 -0
  106. data/lib/metanorma/standoc/validate.rb +158 -0
  107. data/lib/metanorma/standoc/validate_section.rb +55 -0
  108. data/lib/metanorma/standoc/version.rb +1 -1
  109. data/lib/{asciidoctor → metanorma}/standoc/views/datamodel/model_representation.adoc.erb +0 -0
  110. data/lib/{asciidoctor → metanorma}/standoc/views/datamodel/plantuml_representation.adoc.erb +0 -0
  111. data/lib/metanorma-standoc.rb +1 -1
  112. data/metanorma-standoc.gemspec +4 -4
  113. data/spec/{asciidoctor → metanorma}/base_spec.rb +73 -8
  114. data/spec/{asciidoctor → metanorma}/blank_spec.rb +1 -1
  115. data/spec/{asciidoctor → metanorma}/blocks_spec.rb +49 -20
  116. data/spec/{asciidoctor → metanorma}/cleanup_blocks_spec.rb +25 -1
  117. data/spec/{asciidoctor → metanorma}/cleanup_sections_spec.rb +2 -2
  118. data/spec/{asciidoctor → metanorma}/cleanup_spec.rb +9 -9
  119. data/spec/{asciidoctor → metanorma}/cleanup_terms_spec.rb +528 -91
  120. data/spec/{asciidoctor → metanorma}/datamodel/attributes_table_preprocessor_spec.rb +22 -22
  121. data/spec/{asciidoctor → metanorma}/datamodel/diagram_preprocessor_spec.rb +17 -17
  122. data/spec/{asciidoctor → metanorma}/inline_spec.rb +175 -6
  123. data/spec/{asciidoctor → metanorma}/isobib_cache_spec.rb +5 -9
  124. data/spec/{asciidoctor → metanorma}/lists_spec.rb +1 -1
  125. data/spec/{asciidoctor → metanorma}/macros_json2text_spec.rb +0 -0
  126. data/spec/{asciidoctor → metanorma}/macros_plantuml_spec.rb +3 -3
  127. data/spec/{asciidoctor → metanorma}/macros_spec.rb +8 -8
  128. data/spec/{asciidoctor → metanorma}/macros_yaml2text_spec.rb +0 -0
  129. data/spec/metanorma/refs_dl_spec.rb +863 -0
  130. data/spec/{asciidoctor → metanorma}/refs_spec.rb +1277 -687
  131. data/spec/{asciidoctor → metanorma}/section_spec.rb +90 -3
  132. data/spec/{asciidoctor → metanorma}/table_spec.rb +1 -1
  133. data/spec/{asciidoctor → metanorma}/validate_spec.rb +2 -2
  134. data/spec/spec_helper.rb +0 -1
  135. data/spec/vcr_cassettes/dated_iso_ref_joint_iso_iec.yml +179 -179
  136. data/spec/vcr_cassettes/dated_iso_ref_joint_iso_iec1.yml +12 -12
  137. data/spec/vcr_cassettes/isobib_get_123.yml +13 -13
  138. data/spec/vcr_cassettes/isobib_get_123_1.yml +98 -98
  139. data/spec/vcr_cassettes/isobib_get_123_1_fr.yml +111 -111
  140. data/spec/vcr_cassettes/isobib_get_123_2001.yml +13 -13
  141. data/spec/vcr_cassettes/isobib_get_124.yml +14 -14
  142. data/spec/vcr_cassettes/rfcbib_get_rfc8341.yml +14 -14
  143. data/spec/vcr_cassettes/separates_iev_citations_by_top_level_clause.yml +46 -46
  144. metadata +89 -38
  145. data/lib/asciidoctor/standoc/ref_date_id.rb +0 -62
  146. data/spec/asciidoctor/refs_dl_spec.rb +0 -864
@@ -0,0 +1,113 @@
1
+ module Metanorma
2
+ module Standoc
3
+ class PlantUMLBlockMacroBackend
4
+ # https://stackoverflow.com/questions/2108727/which-in-ruby-checking-if-program-exists-in-path-from-ruby
5
+ def self.plantuml_installed?
6
+ cmd = "plantuml"
7
+ exts = ENV["PATHEXT"] ? ENV["PATHEXT"].split(";") : [""]
8
+ ENV["PATH"].split(File::PATH_SEPARATOR).each do |path|
9
+ exts.each do |ext|
10
+ exe = File.join(path, "#{cmd}#{ext}")
11
+ return exe if File.executable?(exe) && !File.directory?(exe)
12
+ end
13
+ end
14
+ raise "PlantUML not installed"
15
+ end
16
+
17
+ def self.run(umlfile, outfile)
18
+ system "plantuml #{umlfile.path}" or (warn $? and return false)
19
+ i = 0
20
+ until !Gem.win_platform? || File.exist?(outfile) || i == 15
21
+ sleep(1)
22
+ i += 1
23
+ end
24
+ File.exist?(outfile)
25
+ end
26
+
27
+ # if no :imagesdir: leave image file in plantuml
28
+ # sleep need for windows because dot works in separate process and
29
+ # plantuml process may finish earlier then dot, as result png file
30
+ # maybe not created yet after plantuml finish
31
+ #
32
+ # # Warning: metanorma/metanorma-standoc#187
33
+ # Windows Ruby 2.4 will crash if a Tempfile is "mv"ed.
34
+ # This is why we need to copy and then unlink.
35
+ def self.generate_file(parent, reader)
36
+ localdir = Metanorma::Utils::localdir(parent.document)
37
+ imagesdir = parent.document.attr("imagesdir")
38
+ umlfile, outfile = save_plantuml parent, reader, localdir
39
+ run(umlfile, outfile) or
40
+ raise "No image output from PlantUML (#{umlfile}, #{outfile})!"
41
+ umlfile.unlink
42
+
43
+ path = path_prep(localdir, imagesdir)
44
+ filename = File.basename(outfile.to_s)
45
+ FileUtils.cp(outfile, path) and outfile.unlink
46
+
47
+ imagesdir ? filename : File.join(path, filename)
48
+ end
49
+
50
+ def self.path_prep(localdir, imagesdir)
51
+ path = Pathname.new(localdir) + (imagesdir || "plantuml")
52
+ File.writable?(localdir) or
53
+ raise "Destination path #{path} not writable for PlantUML!"
54
+ path.mkpath
55
+ File.writable?(path) or
56
+ raise "Destination path #{path} not writable for PlantUML!"
57
+ # File.exist?(path) or raise "Destination path #{path} already exists for PlantUML!"
58
+ path
59
+ end
60
+
61
+ def self.save_plantuml(_parent, reader, _localdir)
62
+ src = prep_source(reader)
63
+ /^@startuml (?<fn>[^\n]+)\n/ =~ src
64
+ Tempfile.open(["plantuml", ".pml"], encoding: "utf-8") do |f|
65
+ f.write(src)
66
+ [f, File.join(File.dirname(f.path),
67
+ "#{fn || File.basename(f.path, '.pml')}.png")]
68
+ end
69
+ end
70
+
71
+ def self.prep_source(reader)
72
+ src = reader.source
73
+ reader.lines.first.sub(/\s+$/, "").match /^@startuml($| )/ or
74
+ src = "@startuml\n#{src}\n@enduml\n"
75
+ %r{@enduml\s*$}m.match?(src) or
76
+ raise "@startuml without matching @enduml in PlantUML!"
77
+ src
78
+ end
79
+
80
+ def self.generate_attrs(attrs)
81
+ %w(id align float title role width height alt)
82
+ .inject({}) do |memo, key|
83
+ memo[key] = attrs[key] if attrs.has_key? key
84
+ memo
85
+ end
86
+ end
87
+ end
88
+
89
+ class PlantUMLBlockMacro < Asciidoctor::Extensions::BlockProcessor
90
+ use_dsl
91
+ named :plantuml
92
+ on_context :literal
93
+ parse_content_as :raw
94
+
95
+ def abort(parent, reader, attrs, msg)
96
+ warn msg
97
+ attrs["language"] = "plantuml"
98
+ create_listing_block parent, reader.source,
99
+ (attrs.reject { |k, _v| k == 1 })
100
+ end
101
+
102
+ def process(parent, reader, attrs)
103
+ PlantUMLBlockMacroBackend.plantuml_installed?
104
+ filename = PlantUMLBlockMacroBackend.generate_file(parent, reader)
105
+ through_attrs = PlantUMLBlockMacroBackend.generate_attrs attrs
106
+ through_attrs["target"] = filename
107
+ create_image_block parent, through_attrs
108
+ rescue StandardError => e
109
+ abort(parent, reader, attrs, e.message)
110
+ end
111
+ end
112
+ end
113
+ end
@@ -0,0 +1,181 @@
1
+ require "csv"
2
+
3
+ module Metanorma
4
+ module Standoc
5
+ class PreferredTermInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
6
+ use_dsl
7
+ named :preferred
8
+ parse_content_as :text
9
+ using_format :short
10
+
11
+ def process(parent, _target, attrs)
12
+ out = Asciidoctor::Inline.new(parent, :quoted, attrs["text"]).convert
13
+ %{<preferred><expression><name>#{out}</name></expression></preferred>}
14
+ end
15
+ end
16
+
17
+ class AltTermInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
18
+ use_dsl
19
+ named :alt
20
+ parse_content_as :text
21
+ using_format :short
22
+
23
+ def process(parent, _target, attrs)
24
+ out = Asciidoctor::Inline.new(parent, :quoted,
25
+ attrs["text"]).convert
26
+ %{<admitted><expression><name>#{out}</name></expression></admitted>}
27
+ end
28
+ end
29
+
30
+ class DeprecatedTermInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
31
+ use_dsl
32
+ named :deprecated
33
+ parse_content_as :text
34
+ using_format :short
35
+
36
+ def process(parent, _target, attrs)
37
+ out = Asciidoctor::Inline.new(parent, :quoted,
38
+ attrs["text"]).convert
39
+ %{<deprecates><expression><name>#{out}</name></expression></deprecates>}
40
+ end
41
+ end
42
+
43
+ class DomainTermInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
44
+ use_dsl
45
+ named :domain
46
+ parse_content_as :text
47
+ using_format :short
48
+
49
+ def process(parent, _target, attrs)
50
+ out = Asciidoctor::Inline.new(parent, :quoted,
51
+ attrs["text"]).convert
52
+ %{<domain>#{out}</domain>}
53
+ end
54
+ end
55
+
56
+ class TermRefInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
57
+ use_dsl
58
+ named :term
59
+ name_positional_attributes "name", "termxref"
60
+ using_format :short
61
+
62
+ def process(_parent, _target, attrs)
63
+ termref = attrs["termxref"] || attrs["name"]
64
+ "<concept type='term'><termxref>#{attrs['name']}</termxref>"\
65
+ "<renderterm>#{termref}</renderterm><xrefrender/></concept>"
66
+ end
67
+ end
68
+
69
+ class SymbolRefInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
70
+ use_dsl
71
+ named :symbol
72
+ name_positional_attributes "name", "termxref"
73
+ using_format :short
74
+
75
+ def process(_parent, _target, attrs)
76
+ termref = attrs["termxref"] || attrs["name"]
77
+ "<concept type='symbol'><termxref>#{attrs['name']}</termxref>"\
78
+ "<renderterm>#{termref}</renderterm><xrefrender/></concept>"
79
+ end
80
+ end
81
+
82
+ # Possibilities:
83
+ # {{<<id>>, term}}
84
+ # {{<<id>>, term, text}}
85
+ # {{<<termbase:id>>, term}}
86
+ # {{<<termbase:id>>, term, text}}
87
+ # {{term}} equivalent to term:[term]
88
+ # {{term, text}} equivalent to term:[term, text]
89
+ # text may optionally be followed by crossreference-rendering, options=""
90
+ class ConceptInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
91
+ use_dsl
92
+ named :concept
93
+ match /\{\{(?<content>|.*?[^\\])\}\}/
94
+ using_format :short
95
+
96
+ def preprocess_attrs(target)
97
+ m = /^(?<id>&lt;&lt;.+?&gt;&gt;)?(?<rest>.*)$/.match(target)
98
+ ret = { id: m[:id]&.sub(/^&lt;&lt;/, "")&.sub(/&gt;&gt;$/, "") }
99
+ if m2 = /^(?<rest>.*?)(?<opt>,opt(ion)?s=.+)$/
100
+ .match(m[:rest].sub(/^,/, ""))
101
+ ret[:opt] = CSV.parse_line(m2[:opt].sub(/^,opt(ion)?s=/, "")
102
+ .sub(/^"(.+)"$/, "\\1").sub(/^'(.+)'$/, "\\1"))
103
+ begin
104
+ attrs = CSV.parse_line(m2[:rest]) || []
105
+ rescue StandardError
106
+ raise "error processing #{m2[:rest]} as CSV"
107
+ end
108
+ else
109
+ begin
110
+ attrs = CSV.parse_line(m[:rest].sub(/^,/, "")) || []
111
+ rescue StandardError
112
+ raise "error processing #{m[:rest]} as CSV"
113
+ end
114
+ end
115
+ ret.merge(term: attrs[0], word: attrs[1] || attrs[0],
116
+ render: attrs[2])
117
+ end
118
+
119
+ def generate_attrs(opts)
120
+ ret = ""
121
+ opts.include?("noital") and ret += " ital='false'"
122
+ opts.include?("noref") and ret += " ref='false'"
123
+ opts.include?("ital") and ret += " ital='true'"
124
+ opts.include?("ref") and ret += " ref='true'"
125
+ opts.include?("nolinkmention") and ret += " linkmention='false'"
126
+ opts.include?("linkmention") and ret += " linkmention='true'"
127
+ opts.include?("nolinkref") and ret += " linkref='false'"
128
+ opts.include?("linkref") and ret += " linkref='true'"
129
+ ret
130
+ end
131
+
132
+ def process(parent, target, _attrs)
133
+ attrs = preprocess_attrs(target)
134
+ term = Asciidoctor::Inline.new(parent, :quoted,
135
+ attrs[:term]).convert
136
+ word = Asciidoctor::Inline.new(parent, :quoted,
137
+ attrs[:word]).convert
138
+ xref = Asciidoctor::Inline.new(parent, :quoted,
139
+ attrs[:render]).convert
140
+ opt = generate_attrs(attrs[:opt] || [])
141
+ if attrs[:id] then "<concept#{opt} key='#{attrs[:id]}'><refterm>"\
142
+ "#{term}</refterm><renderterm>#{word}</renderterm>"\
143
+ "<xrefrender>#{xref}</xrefrender></concept>"
144
+ else "<concept#{opt}><termxref>#{term}</termxref><renderterm>"\
145
+ "#{word}</renderterm><xrefrender>#{xref}</xrefrender></concept>"
146
+ end
147
+ rescue StandardError => e
148
+ raise("processing {{#{target}}}: #{e.message}")
149
+ end
150
+ end
151
+
152
+ # Possibilities:
153
+ # related:relation[<<id>>, term]
154
+ # related:relation[<<termbase:id>>, term]
155
+ # related:relation[term] equivalent to a crossreference to term:[term]
156
+ class RelatedTermInlineMacro < Asciidoctor::Extensions::InlineMacroProcessor
157
+ use_dsl
158
+ named :related
159
+ parse_content_as :text
160
+
161
+ def preprocess_attrs(target)
162
+ m = /^(?<id>&lt;&lt;.+?&gt;&gt;, ?)?(?<rest>.*)$/.match(target)
163
+ { id: m[:id]&.sub(/^&lt;&lt;/, "")&.sub(/&gt;&gt;, ?$/, ""),
164
+ term: m[:rest] }
165
+ end
166
+
167
+ def process(parent, target, attrs)
168
+ out = preprocess_attrs(attrs["text"])
169
+ term = Asciidoctor::Inline.new(parent, :quoted,
170
+ out[:term]).convert
171
+ if out[:id] then "<related type='#{target}' key='#{out[:id]}'>"\
172
+ "<refterm>#{term}</refterm></related>"
173
+ else "<related type='#{target}'><termxref>#{term}</termxref>"\
174
+ "<xrefrender>#{term}</xrefrender></related>"
175
+ end
176
+ rescue StandardError => e
177
+ raise("processing related:#{target}[#{attrs['text']}]: #{e.message}")
178
+ end
179
+ end
180
+ end
181
+ end
@@ -0,0 +1,243 @@
1
+ require_relative "ref_utility"
2
+
3
+ module Metanorma
4
+ module Standoc
5
+ module Refs
6
+ def iso_publisher(bib, code)
7
+ code.sub(/ .*$/, "").split("/").each do |abbrev|
8
+ bib.contributor do |c|
9
+ c.role **{ type: "publisher" }
10
+ c.organization do |org|
11
+ organization(org, abbrev, true)
12
+ end
13
+ end
14
+ end
15
+ end
16
+
17
+ def isorefrender1(bib, match, year, allp = "")
18
+ bib.title(**plaintxt) { |i| i << ref_normalise(match[:text]) }
19
+ docid(bib, match[:usrlbl]) if match[:usrlbl]
20
+ docid(bib, id_and_year(match[:code], year) + allp)
21
+ docnumber(bib, match[:code])
22
+ end
23
+
24
+ def isorefmatchescode(match)
25
+ yr = norm_year(match[:year])
26
+ { code: match[:code], year: yr, match: match,
27
+ title: match[:text], usrlbl: match[:usrlbl],
28
+ lang: (@lang || :all) }
29
+ end
30
+
31
+ def isorefmatchesout(item, xml)
32
+ if item[:doc] then use_retrieved_relaton(item, xml)
33
+ else
34
+ xml.bibitem **attr_code(ref_attributes(item[:ref][:match])) do |t|
35
+ isorefrender1(t, item[:ref][:match], item[:ref][:year])
36
+ item[:ref][:year] and t.date **{ type: "published" } do |d|
37
+ set_date_range(d, item[:ref][:year])
38
+ end
39
+ iso_publisher(t, item[:ref][:match][:code])
40
+ end
41
+ end
42
+ end
43
+
44
+ def isorefmatches2code(match)
45
+ { code: match[:code], no_year: true,
46
+ note: match[:fn], year: nil, match: match,
47
+ title: match[:text], usrlbl: match[:usrlbl],
48
+ lang: (@lang || :all) }
49
+ end
50
+
51
+ def isorefmatches2out(item, xml)
52
+ if item[:doc] then use_retrieved_relaton(item, xml)
53
+ else isorefmatches2_1(xml, item[:ref][:match])
54
+ end
55
+ end
56
+
57
+ def isorefmatches2_1(xml, match)
58
+ xml.bibitem **attr_code(ref_attributes(match)) do |t|
59
+ isorefrender1(t, match, "--")
60
+ t.date **{ type: "published" } do |d|
61
+ d.on "--"
62
+ end
63
+ iso_publisher(t, match[:code])
64
+ unless match[:fn].nil?
65
+ t.note(**plaintxt.merge(type: "Unpublished-Status")) do |p|
66
+ p << (match[:fn]).to_s
67
+ end
68
+ end
69
+ end
70
+ end
71
+
72
+ def isorefmatches3code(match)
73
+ yr = norm_year(match[:year])
74
+ hasyr = !yr.nil? && yr != "--"
75
+ { code: match[:code], match: match, yr: yr, hasyr: hasyr,
76
+ year: hasyr ? yr : nil,
77
+ all_parts: true, no_year: yr == "--",
78
+ text: match[:text], usrlbl: match[:usrlbl],
79
+ lang: (@lang || :all) }
80
+ end
81
+
82
+ def isorefmatches3out(item, xml)
83
+ if item[:doc] then use_retrieved_relaton(item, xml)
84
+ else
85
+ isorefmatches3_1(xml, item[:ref][:match], item[:ref][:yr],
86
+ item[:ref][:hasyr], item[:doc])
87
+ end
88
+ end
89
+
90
+ def isorefmatches3_1(xml, match, yr, _hasyr, _ref)
91
+ xml.bibitem(**attr_code(ref_attributes(match))) do |t|
92
+ isorefrender1(t, match, yr, " (all parts)")
93
+ conditional_date(t, match, yr == "--")
94
+ iso_publisher(t, match[:code])
95
+ if match.names.include?("fn") && match[:fn]
96
+ t.note(**plaintxt.merge(type: "Unpublished-Status")) do |p|
97
+ p << (match[:fn]).to_s
98
+ end
99
+ end
100
+ t.extent **{ type: "part" } do |e|
101
+ e.referenceFrom "all"
102
+ end
103
+ end
104
+ end
105
+
106
+ def refitem_render1(match, code, bib)
107
+ if code[:type] == "path"
108
+ bib.uri code[:key].sub(/\.[a-zA-Z0-9]+$/, ""), **{ type: "URI" }
109
+ bib.uri code[:key].sub(/\.[a-zA-Z0-9]+$/, ""), **{ type: "citation" }
110
+ end
111
+ code[:id].sub!(/[:-](19|20)[0-9][0-9]$/, "")
112
+ docid(bib, match[:usrlbl]) if match[:usrlbl]
113
+ docid(bib, /^\d+$/.match?(code[:id]) ? "[#{code[:id]}]" : code[:id])
114
+ code[:type] == "repo" and
115
+ bib.docidentifier code[:key], **{ type: "repository" }
116
+ end
117
+
118
+ def refitem_render(xml, match, code)
119
+ xml.bibitem **attr_code(id: match[:anchor],
120
+ hidden: code[:hidden]) do |t|
121
+ t.formattedref **{ format: "application/x-isodoc+xml" } do |i|
122
+ i << ref_normalise_no_format(match[:text])
123
+ end
124
+ yr_match = /[:-](?<year>(19|20)[0-9][0-9])\b/.match(code[:id])
125
+ refitem_render1(match, code, t)
126
+ docnumber(t, code[:id]) unless /^\d+$|^\(.+\)$/.match?(code[:id])
127
+ conditional_date(t, yr_match || match, false)
128
+ end
129
+ end
130
+
131
+ # TODO: alternative where only title is available
132
+ def refitemcode(item, node)
133
+ m = NON_ISO_REF.match(item) and return refitem1code(item, m).compact
134
+ @log.add("AsciiDoc Input", node, "#{MALFORMED_REF}: #{item}")
135
+ {}
136
+ end
137
+
138
+ def refitem1code(_item, match)
139
+ code = analyse_ref_code(match[:code])
140
+ ((code[:id] && code[:numeric]) || code[:nofetch]) and
141
+ return { code: nil, match: match, analyse_code: code,
142
+ hidden: code[:hidden] }
143
+ year = refitem1yr(code[:id])
144
+ { code: code[:id], analyse_code: code,
145
+ year: year,
146
+ title: match[:text], match: match, hidden: code[:hidden],
147
+ usrlbl: match[:usrlbl], lang: (@lang || :all) }
148
+ end
149
+
150
+ def refitem1yr(code)
151
+ yr_match = /[:-](?<year>(19|20)[0-9][0-9])\b/.match(code)
152
+ yr_match ? yr_match[:year] : nil
153
+ end
154
+
155
+ def refitemout(item, xml)
156
+ return nil if item[:ref][:match].nil?
157
+
158
+ item[:doc] or return refitem_render(xml, item[:ref][:match],
159
+ item[:ref][:analyse_code])
160
+ use_retrieved_relaton(item, xml)
161
+ end
162
+
163
+ ISO_REF =
164
+ %r{^<ref\sid="(?<anchor>[^"]+)">
165
+ \[(?<usrlbl>\([^)]+\))?(?<code>(ISO|IEC)[^0-9]*\s[0-9-]+|IEV)
166
+ (:(?<year>[0-9][0-9-]+))?\]</ref>,?\s*(?<text>.*)$}xm.freeze
167
+
168
+ ISO_REF_NO_YEAR =
169
+ %r{^<ref\sid="(?<anchor>[^"]+)">
170
+ \[(?<usrlbl>\([^)]+\))?(?<code>(ISO|IEC)[^0-9]*\s[0-9-]+):
171
+ (--|&\#821[12];)\]</ref>,?\s*
172
+ (<fn[^>]*>\s*<p>(?<fn>[^\]]+)</p>\s*</fn>)?,?\s?(?<text>.*)$}xm
173
+ .freeze
174
+
175
+ ISO_REF_ALL_PARTS =
176
+ %r{^<ref\sid="(?<anchor>[^"]+)">
177
+ \[(?<usrlbl>\([^)]+\))?(?<code>(ISO|IEC)[^0-9]*\s[0-9]+)
178
+ (:(?<year>--|&\#821[12];|[0-9][0-9-]+))?\s
179
+ \(all\sparts\)\]</ref>,?\s*
180
+ (<fn[^>]*>\s*<p>(?<fn>[^\]]+)</p>\s*</fn>,?\s?)?(?<text>.*)$}xm.freeze
181
+
182
+ NON_ISO_REF = %r{^<ref\sid="(?<anchor>[^"]+)">
183
+ \[(?<usrlbl>\([^)]+\))?(?<code>[^\]]+?)\]</ref>,?\s*(?<text>.*)$}xm
184
+ .freeze
185
+
186
+ def reference1_matches(item)
187
+ matched = ISO_REF.match item
188
+ matched2 = ISO_REF_NO_YEAR.match item
189
+ matched3 = ISO_REF_ALL_PARTS.match item
190
+ [matched, matched2, matched3]
191
+ end
192
+
193
+ def reference1code(item, node)
194
+ matched, matched2, matched3 = reference1_matches(item)
195
+ if matched3.nil? && matched2.nil? && matched.nil?
196
+ refitemcode(item, node).merge(process: 0)
197
+ elsif !matched.nil? then isorefmatchescode(matched).merge(process: 1)
198
+ elsif !matched2.nil? then isorefmatches2code(matched2).merge(process: 2)
199
+ elsif !matched3.nil? then isorefmatches3code(matched3).merge(process: 3)
200
+ end
201
+ end
202
+
203
+ def reference1out(item, xml)
204
+ case item[:ref][:process]
205
+ when 0 then refitemout(item, xml)
206
+ when 1 then isorefmatchesout(item, xml)
207
+ when 2 then isorefmatches2out(item, xml)
208
+ when 3 then isorefmatches3out(item, xml)
209
+ end
210
+ end
211
+
212
+ def reference_preproc(node)
213
+ refs = node.items.each_with_object([]) do |b, m|
214
+ m << reference1code(b.text, node)
215
+ end
216
+ results = refs.each_with_index.with_object(Queue.new) do |(ref, i), res|
217
+ fetch_ref_async(ref.merge(ord: i), i, res)
218
+ end
219
+ [refs, results]
220
+ end
221
+
222
+ def reference(node)
223
+ refs, results = reference_preproc(node)
224
+ ret = reference_queue(refs, results)
225
+ noko do |xml|
226
+ ret.each { |b| reference1out(b, xml) }
227
+ end.join
228
+ end
229
+
230
+ def reference_queue(refs, results)
231
+ refs.each.with_object([]) do |_, m|
232
+ ref, i, doc = results.pop
233
+ m[i.to_i] = { ref: ref }
234
+ if doc.is_a?(RelatonBib::RequestError)
235
+ @log.add("Bibliography", nil, "Could not retrieve #{ref[:code]}: "\
236
+ "no access to online site")
237
+ else m[i.to_i][:doc] = doc
238
+ end
239
+ end
240
+ end
241
+ end
242
+ end
243
+ end
@@ -0,0 +1,153 @@
1
+ module Metanorma
2
+ module Standoc
3
+ module Section
4
+ def in_biblio?
5
+ @biblio
6
+ end
7
+
8
+ def in_norm_ref?
9
+ @norm_ref
10
+ end
11
+
12
+ def bibliography_parse(attrs, xml, node)
13
+ x = biblio_prep(attrs, xml, node) and return x
14
+ @biblio = true
15
+ attrs = attrs.merge(normative: node.attr("normative") || false)
16
+ xml.references **attr_code(attrs) do |xml_section|
17
+ xml_section.title { |t| t << node.title }
18
+ xml_section << node.content
19
+ end
20
+ @biblio = false
21
+ end
22
+
23
+ def bibitem_parse(attrs, xml, node)
24
+ norm_ref = @norm_ref
25
+ biblio = @biblio
26
+ @biblio = false
27
+ @norm_ref = false
28
+ ret = clause_parse(attrs, xml, node)
29
+ @biblio = biblio
30
+ @norm_ref = norm_ref
31
+ ret
32
+ end
33
+
34
+ def norm_ref_parse(attrs, xml, node)
35
+ x = biblio_prep(attrs, xml, node) and return x
36
+ @norm_ref = true
37
+ attrs = attrs.merge(normative: node.attr("normative") || true)
38
+ xml.references **attr_code(attrs) do |xml_section|
39
+ xml_section.title { |t| t << node.title }
40
+ xml_section << node.content
41
+ end
42
+ @norm_ref = false
43
+ end
44
+
45
+ def biblio_prep(attrs, xml, node)
46
+ if node.option? "bibitem"
47
+ bibitem_parse(attrs, xml, node)
48
+ else
49
+ node.attr("style") == "bibliography" or
50
+ @log.add("AsciiDoc Input", node,
51
+ "Section not marked up as [bibliography]!")
52
+ nil
53
+ end
54
+ end
55
+
56
+ def global_ievcache_name
57
+ "#{Dir.home}/.iev/cache"
58
+ end
59
+
60
+ def local_ievcache_name(cachename)
61
+ return nil if cachename.nil?
62
+
63
+ cachename += "_iev" unless cachename.empty?
64
+ cachename = "iev" if cachename.empty?
65
+ "#{cachename}/cache"
66
+ end
67
+
68
+ def fetch_ref(xml, code, year, **opts)
69
+ return nil if opts[:no_year]
70
+
71
+ code = code.sub(/^\([^)]+\)/, "")
72
+ hit = @bibdb&.fetch(code, year, opts)
73
+ return nil if hit.nil?
74
+
75
+ xml.parent.add_child(smart_render_xml(hit, code, opts))
76
+ xml
77
+ rescue RelatonBib::RequestError
78
+ @log.add("Bibliography", nil, "Could not retrieve #{code}: "\
79
+ "no access to online site")
80
+ nil
81
+ end
82
+
83
+ def fetch_ref_async(ref, idx, res)
84
+ if ref[:code].nil? || ref[:no_year] || @bibdb.nil?
85
+ res << [ref, idx, nil]
86
+ else
87
+ @bibdb.fetch_async(ref[:code], ref[:year], ref) do |doc|
88
+ res << [ref, idx, doc]
89
+ end
90
+ end
91
+ end
92
+
93
+ def emend_biblio(xml, code, title, usrlbl)
94
+ unless xml.at("/bibitem/docidentifier[not(@type = 'DOI')][text()]")
95
+ @log.add("Bibliography", nil,
96
+ "ERROR: No document identifier retrieved for #{code}")
97
+ xml.root << "<docidentifier>#{code}</docidentifier>"
98
+ end
99
+ unless xml.at("/bibitem/title[text()]")
100
+ @log.add("Bibliography", nil,
101
+ "ERROR: No title retrieved for #{code}")
102
+ xml.root << "<title>#{title || '(MISSING TITLE)'}</title>"
103
+ end
104
+ usrlbl and xml.at("/bibitem/docidentifier").next =
105
+ "<docidentifier type='metanorma'>#{mn_code(usrlbl)}</docidentifier>"
106
+ end
107
+
108
+ def smart_render_xml(xml, code, opts)
109
+ xml.respond_to? :to_xml or return nil
110
+ xml = Nokogiri::XML(xml.to_xml(lang: opts[:lang]))
111
+ emend_biblio(xml, code, opts[:title], opts[:usrlbl])
112
+ xml.xpath("//date").each { |d| Metanorma::Utils::endash_date(d) }
113
+ xml.traverse do |n|
114
+ n.text? and n.replace(Metanorma::Utils::smartformat(n.text))
115
+ end
116
+ xml.to_xml.sub(/<\?[^>]+>/, "")
117
+ end
118
+
119
+ def use_retrieved_relaton(item, xml)
120
+ xml.parent.add_child(smart_render_xml(item[:doc], item[:ref][:code],
121
+ item[:ref]))
122
+ use_my_anchor(xml, item[:ref][:match][:anchor], item.dig(:ref, :analyse_code, :hidden))
123
+ end
124
+
125
+ def init_bib_caches(node)
126
+ return if @no_isobib
127
+
128
+ global = !@no_isobib_cache && !node.attr("local-cache-only")
129
+ local = node.attr("local-cache") || node.attr("local-cache-only")
130
+ local = nil if @no_isobib_cache
131
+ @bibdb = Relaton::DbCache.init_bib_caches(
132
+ local_cache: local,
133
+ flush_caches: node.attr("flush-caches"),
134
+ global_cache: global,
135
+ )
136
+ end
137
+
138
+ def init_iev_caches(node)
139
+ unless @no_isobib_cache || @no_isobib
140
+ node.attr("local-cache-only") or
141
+ @iev_globalname = global_ievcache_name
142
+ @iev_localname = local_ievcache_name(node.attr("local-cache") ||
143
+ node.attr("local-cache-only"))
144
+ if node.attr("flush-caches")
145
+ FileUtils.rm_f @iev_globalname unless @iev_globalname.nil?
146
+ FileUtils.rm_f @iev_localname unless @iev_localname.nil?
147
+ end
148
+ end
149
+ # @iev = Iev::Db.new(globalname, localname) unless @no_isobib
150
+ end
151
+ end
152
+ end
153
+ end