• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#
2# Copyright (C) 2012 The Android Open Source Project
3#
4# Licensed under the Apache License, Version 2.0 (the "License");
5# you may not use this file except in compliance with the License.
6# You may obtain a copy of the License at
7#
8#      http://www.apache.org/licenses/LICENSE-2.0
9#
10# Unless required by applicable law or agreed to in writing, software
11# distributed under the License is distributed on an "AS IS" BASIS,
12# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13# See the License for the specific language governing permissions and
14# limitations under the License.
15#
16
17"""
18A set of helpers for rendering Mako templates with a Metadata model.
19"""
20
21import metadata_model
22import re
23import markdown
24import textwrap
25import sys
26import bs4
27# Monkey-patch BS4. WBR element must not have an end tag.
28bs4.builder.HTMLTreeBuilder.empty_element_tags.add("wbr")
29
30from collections import OrderedDict
31
32# Relative path from HTML file to the base directory used by <img> tags
33IMAGE_SRC_METADATA="images/camera2/metadata/"
34
35# Prepend this path to each <img src="foo"> in javadocs
36JAVADOC_IMAGE_SRC_METADATA="../../../../" + IMAGE_SRC_METADATA
37
38_context_buf = None
39
40def _is_sec_or_ins(x):
41  return isinstance(x, metadata_model.Section) or    \
42         isinstance(x, metadata_model.InnerNamespace)
43
44##
45## Metadata Helpers
46##
47
48def find_all_sections(root):
49  """
50  Find all descendants that are Section or InnerNamespace instances.
51
52  Args:
53    root: a Metadata instance
54
55  Returns:
56    A list of Section/InnerNamespace instances
57
58  Remarks:
59    These are known as "sections" in the generated C code.
60  """
61  return root.find_all(_is_sec_or_ins)
62
63def find_parent_section(entry):
64  """
65  Find the closest ancestor that is either a Section or InnerNamespace.
66
67  Args:
68    entry: an Entry or Clone node
69
70  Returns:
71    An instance of Section or InnerNamespace
72  """
73  return entry.find_parent_first(_is_sec_or_ins)
74
75# find uniquely named entries (w/o recursing through inner namespaces)
76def find_unique_entries(node):
77  """
78  Find all uniquely named entries, without recursing through inner namespaces.
79
80  Args:
81    node: a Section or InnerNamespace instance
82
83  Yields:
84    A sequence of MergedEntry nodes representing an entry
85
86  Remarks:
87    This collapses multiple entries with the same fully qualified name into
88    one entry (e.g. if there are multiple entries in different kinds).
89  """
90  if not isinstance(node, metadata_model.Section) and    \
91     not isinstance(node, metadata_model.InnerNamespace):
92      raise TypeError("expected node to be a Section or InnerNamespace")
93
94  d = OrderedDict()
95  # remove the 'kinds' from the path between sec and the closest entries
96  # then search the immediate children of the search path
97  search_path = isinstance(node, metadata_model.Section) and node.kinds \
98                or [node]
99  for i in search_path:
100      for entry in i.entries:
101          d[entry.name] = entry
102
103  for k,v in d.iteritems():
104      yield v.merge()
105
106def path_name(node):
107  """
108  Calculate a period-separated string path from the root to this element,
109  by joining the names of each node and excluding the Metadata/Kind nodes
110  from the path.
111
112  Args:
113    node: a Node instance
114
115  Returns:
116    A string path
117  """
118
119  isa = lambda x,y: isinstance(x, y)
120  fltr = lambda x: not isa(x, metadata_model.Metadata) and \
121                   not isa(x, metadata_model.Kind)
122
123  path = node.find_parents(fltr)
124  path = list(path)
125  path.reverse()
126  path.append(node)
127
128  return ".".join((i.name for i in path))
129
130def has_descendants_with_enums(node):
131  """
132  Determine whether or not the current node is or has any descendants with an
133  Enum node.
134
135  Args:
136    node: a Node instance
137
138  Returns:
139    True if it finds an Enum node in the subtree, False otherwise
140  """
141  return bool(node.find_first(lambda x: isinstance(x, metadata_model.Enum)))
142
143def get_children_by_throwing_away_kind(node, member='entries'):
144  """
145  Get the children of this node by compressing the subtree together by removing
146  the kind and then combining any children nodes with the same name together.
147
148  Args:
149    node: An instance of Section, InnerNamespace, or Kind
150
151  Returns:
152    An iterable over the combined children of the subtree of node,
153    as if the Kinds never existed.
154
155  Remarks:
156    Not recursive. Call this function repeatedly on each child.
157  """
158
159  if isinstance(node, metadata_model.Section):
160    # Note that this makes jump from Section to Kind,
161    # skipping the Kind entirely in the tree.
162    node_to_combine = node.combine_kinds_into_single_node()
163  else:
164    node_to_combine = node
165
166  combined_kind = node_to_combine.combine_children_by_name()
167
168  return (i for i in getattr(combined_kind, member))
169
170def get_children_by_filtering_kind(section, kind_name, member='entries'):
171  """
172  Takes a section and yields the children of the merged kind under this section.
173
174  Args:
175    section: An instance of Section
176    kind_name: A name of the kind, i.e. 'dynamic' or 'static' or 'controls'
177
178  Returns:
179    An iterable over the children of the specified merged kind.
180  """
181
182  matched_kind = next((i for i in section.merged_kinds if i.name == kind_name), None)
183
184  if matched_kind:
185    return getattr(matched_kind, member)
186  else:
187    return ()
188
189##
190## Filters
191##
192
193# abcDef.xyz -> ABC_DEF_XYZ
194def csym(name):
195  """
196  Convert an entry name string into an uppercase C symbol.
197
198  Returns:
199    A string
200
201  Example:
202    csym('abcDef.xyz') == 'ABC_DEF_XYZ'
203  """
204  newstr = name
205  newstr = "".join([i.isupper() and ("_" + i) or i for i in newstr]).upper()
206  newstr = newstr.replace(".", "_")
207  return newstr
208
209# abcDef.xyz -> abc_def_xyz
210def csyml(name):
211  """
212  Convert an entry name string into a lowercase C symbol.
213
214  Returns:
215    A string
216
217  Example:
218    csyml('abcDef.xyz') == 'abc_def_xyz'
219  """
220  return csym(name).lower()
221
222# pad with spaces to make string len == size. add new line if too big
223def ljust(size, indent=4):
224  """
225  Creates a function that given a string will pad it with spaces to make
226  the string length == size. Adds a new line if the string was too big.
227
228  Args:
229    size: an integer representing how much spacing should be added
230    indent: an integer representing the initial indendation level
231
232  Returns:
233    A function that takes a string and returns a string.
234
235  Example:
236    ljust(8)("hello") == 'hello   '
237
238  Remarks:
239    Deprecated. Use pad instead since it works for non-first items in a
240    Mako template.
241  """
242  def inner(what):
243    newstr = what.ljust(size)
244    if len(newstr) > size:
245      return what + "\n" + "".ljust(indent + size)
246    else:
247      return newstr
248  return inner
249
250def _find_new_line():
251
252  if _context_buf is None:
253    raise ValueError("Context buffer was not set")
254
255  buf = _context_buf
256  x = -1 # since the first read is always ''
257  cur_pos = buf.tell()
258  while buf.tell() > 0 and buf.read(1) != '\n':
259    buf.seek(cur_pos - x)
260    x = x + 1
261
262  buf.seek(cur_pos)
263
264  return int(x)
265
266# Pad the string until the buffer reaches the desired column.
267# If string is too long, insert a new line with 'col' spaces instead
268def pad(col):
269  """
270  Create a function that given a string will pad it to the specified column col.
271  If the string overflows the column, put the string on a new line and pad it.
272
273  Args:
274    col: an integer specifying the column number
275
276  Returns:
277    A function that given a string will produce a padded string.
278
279  Example:
280    pad(8)("hello") == 'hello   '
281
282  Remarks:
283    This keeps track of the line written by Mako so far, so it will always
284    align to the column number correctly.
285  """
286  def inner(what):
287    wut = int(col)
288    current_col = _find_new_line()
289
290    if len(what) > wut - current_col:
291      return what + "\n".ljust(col)
292    else:
293      return what.ljust(wut - current_col)
294  return inner
295
296# int32 -> TYPE_INT32, byte -> TYPE_BYTE, etc. note that enum -> TYPE_INT32
297def ctype_enum(what):
298  """
299  Generate a camera_metadata_type_t symbol from a type string.
300
301  Args:
302    what: a type string
303
304  Returns:
305    A string representing the camera_metadata_type_t
306
307  Example:
308    ctype_enum('int32') == 'TYPE_INT32'
309    ctype_enum('int64') == 'TYPE_INT64'
310    ctype_enum('float') == 'TYPE_FLOAT'
311
312  Remarks:
313    An enum is coerced to a byte since the rest of the camera_metadata
314    code doesn't support enums directly yet.
315  """
316  return 'TYPE_%s' %(what.upper())
317
318
319# Calculate a java type name from an entry with a Typedef node
320def _jtypedef_type(entry):
321  typedef = entry.typedef
322  additional = ''
323
324  # Hacky way to deal with arrays. Assume that if we have
325  # size 'Constant x N' the Constant is part of the Typedef size.
326  # So something sized just 'Constant', 'Constant1 x Constant2', etc
327  # is not treated as a real java array.
328  if entry.container == 'array':
329    has_variable_size = False
330    for size in entry.container_sizes:
331      try:
332        size_int = int(size)
333      except ValueError:
334        has_variable_size = True
335
336    if has_variable_size:
337      additional = '[]'
338
339  try:
340    name = typedef.languages['java']
341
342    return "%s%s" %(name, additional)
343  except KeyError:
344    return None
345
346# Box if primitive. Otherwise leave unboxed.
347def _jtype_box(type_name):
348  mapping = {
349    'boolean': 'Boolean',
350    'byte': 'Byte',
351    'int': 'Integer',
352    'float': 'Float',
353    'double': 'Double',
354    'long': 'Long'
355  }
356
357  return mapping.get(type_name, type_name)
358
359def jtype_unboxed(entry):
360  """
361  Calculate the Java type from an entry type string, to be used whenever we
362  need the regular type in Java. It's not boxed, so it can't be used as a
363  generic type argument when the entry type happens to resolve to a primitive.
364
365  Remarks:
366    Since Java generics cannot be instantiated with primitives, this version
367    is not applicable in that case. Use jtype_boxed instead for that.
368
369  Returns:
370    The string representing the Java type.
371  """
372  if not isinstance(entry, metadata_model.Entry):
373    raise ValueError("Expected entry to be an instance of Entry")
374
375  metadata_type = entry.type
376
377  java_type = None
378
379  if entry.typedef:
380    typedef_name = _jtypedef_type(entry)
381    if typedef_name:
382      java_type = typedef_name # already takes into account arrays
383
384  if not java_type:
385    if not java_type and entry.enum and metadata_type == 'byte':
386      # Always map byte enums to Java ints, unless there's a typedef override
387      base_type = 'int'
388
389    else:
390      mapping = {
391        'int32': 'int',
392        'int64': 'long',
393        'float': 'float',
394        'double': 'double',
395        'byte': 'byte',
396        'rational': 'Rational'
397      }
398
399      base_type = mapping[metadata_type]
400
401    # Convert to array (enums, basic types)
402    if entry.container == 'array':
403      additional = '[]'
404    else:
405      additional = ''
406
407    java_type = '%s%s' %(base_type, additional)
408
409  # Now box this sucker.
410  return java_type
411
412def jtype_boxed(entry):
413  """
414  Calculate the Java type from an entry type string, to be used as a generic
415  type argument in Java. The type is guaranteed to inherit from Object.
416
417  It will only box when absolutely necessary, i.e. int -> Integer[], but
418  int[] -> int[].
419
420  Remarks:
421    Since Java generics cannot be instantiated with primitives, this version
422    will use boxed types when absolutely required.
423
424  Returns:
425    The string representing the boxed Java type.
426  """
427  unboxed_type = jtype_unboxed(entry)
428  return _jtype_box(unboxed_type)
429
430def _is_jtype_generic(entry):
431  """
432  Determine whether or not the Java type represented by the entry type
433  string and/or typedef is a Java generic.
434
435  For example, "Range<Integer>" would be considered a generic, whereas
436  a "MeteringRectangle" or a plain "Integer" would not be considered a generic.
437
438  Args:
439    entry: An instance of an Entry node
440
441  Returns:
442    True if it's a java generic, False otherwise.
443  """
444  if entry.typedef:
445    local_typedef = _jtypedef_type(entry)
446    if local_typedef:
447      match = re.search(r'<.*>', local_typedef)
448      return bool(match)
449  return False
450
451def _jtype_primitive(what):
452  """
453  Calculate the Java type from an entry type string.
454
455  Remarks:
456    Makes a special exception for Rational, since it's a primitive in terms of
457    the C-library camera_metadata type system.
458
459  Returns:
460    The string representing the primitive type
461  """
462  mapping = {
463    'int32': 'int',
464    'int64': 'long',
465    'float': 'float',
466    'double': 'double',
467    'byte': 'byte',
468    'rational': 'Rational'
469  }
470
471  try:
472    return mapping[what]
473  except KeyError as e:
474    raise ValueError("Can't map '%s' to a primitive, not supported" %what)
475
476def jclass(entry):
477  """
478  Calculate the java Class reference string for an entry.
479
480  Args:
481    entry: an Entry node
482
483  Example:
484    <entry name="some_int" type="int32"/>
485    <entry name="some_int_array" type="int32" container='array'/>
486
487    jclass(some_int) == 'int.class'
488    jclass(some_int_array) == 'int[].class'
489
490  Returns:
491    The ClassName.class string
492  """
493
494  return "%s.class" %jtype_unboxed(entry)
495
496def jkey_type_token(entry):
497  """
498  Calculate the java type token compatible with a Key constructor.
499  This will be the Java Class<T> for non-generic classes, and a
500  TypeReference<T> for generic classes.
501
502  Args:
503    entry: An entry node
504
505  Returns:
506    The ClassName.class string, or 'new TypeReference<ClassName>() {{ }}' string
507  """
508  if _is_jtype_generic(entry):
509    return "new TypeReference<%s>() {{ }}" %(jtype_boxed(entry))
510  else:
511    return jclass(entry)
512
513def jidentifier(what):
514  """
515  Convert the input string into a valid Java identifier.
516
517  Args:
518    what: any identifier string
519
520  Returns:
521    String with added underscores if necessary.
522  """
523  if re.match("\d", what):
524    return "_%s" %what
525  else:
526    return what
527
528def enum_calculate_value_string(enum_value):
529  """
530  Calculate the value of the enum, even if it does not have one explicitly
531  defined.
532
533  This looks back for the first enum value that has a predefined value and then
534  applies addition until we get the right value, using C-enum semantics.
535
536  Args:
537    enum_value: an EnumValue node with a valid Enum parent
538
539  Example:
540    <enum>
541      <value>X</value>
542      <value id="5">Y</value>
543      <value>Z</value>
544    </enum>
545
546    enum_calculate_value_string(X) == '0'
547    enum_calculate_Value_string(Y) == '5'
548    enum_calculate_value_string(Z) == '6'
549
550  Returns:
551    String that represents the enum value as an integer literal.
552  """
553
554  enum_value_siblings = list(enum_value.parent.values)
555  this_index = enum_value_siblings.index(enum_value)
556
557  def is_hex_string(instr):
558    return bool(re.match('0x[a-f0-9]+$', instr, re.IGNORECASE))
559
560  base_value = 0
561  base_offset = 0
562  emit_as_hex = False
563
564  this_id = enum_value_siblings[this_index].id
565  while this_index != 0 and not this_id:
566    this_index -= 1
567    base_offset += 1
568    this_id = enum_value_siblings[this_index].id
569
570  if this_id:
571    base_value = int(this_id, 0)  # guess base
572    emit_as_hex = is_hex_string(this_id)
573
574  if emit_as_hex:
575    return "0x%X" %(base_value + base_offset)
576  else:
577    return "%d" %(base_value + base_offset)
578
579def enumerate_with_last(iterable):
580  """
581  Enumerate a sequence of iterable, while knowing if this element is the last in
582  the sequence or not.
583
584  Args:
585    iterable: an Iterable of some sequence
586
587  Yields:
588    (element, bool) where the bool is True iff the element is last in the seq.
589  """
590  it = (i for i in iterable)
591
592  first = next(it)  # OK: raises exception if it is empty
593
594  second = first  # for when we have only 1 element in iterable
595
596  try:
597    while True:
598      second = next(it)
599      # more elements remaining.
600      yield (first, False)
601      first = second
602  except StopIteration:
603    # last element. no more elements left
604    yield (second, True)
605
606def pascal_case(what):
607  """
608  Convert the first letter of a string to uppercase, to make the identifier
609  conform to PascalCase.
610
611  If there are dots, remove the dots, and capitalize the letter following
612  where the dot was. Letters that weren't following dots are left unchanged,
613  except for the first letter of the string (which is made upper-case).
614
615  Args:
616    what: a string representing some identifier
617
618  Returns:
619    String with first letter capitalized
620
621  Example:
622    pascal_case("helloWorld") == "HelloWorld"
623    pascal_case("foo") == "Foo"
624    pascal_case("hello.world") = "HelloWorld"
625    pascal_case("fooBar.fooBar") = "FooBarFooBar"
626  """
627  return "".join([s[0:1].upper() + s[1:] for s in what.split('.')])
628
629def jkey_identifier(what):
630  """
631  Return a Java identifier from a property name.
632
633  Args:
634    what: a string representing a property name.
635
636  Returns:
637    Java identifier corresponding to the property name. May need to be
638    prepended with the appropriate Java class name by the caller of this
639    function. Note that the outer namespace is stripped from the property
640    name.
641
642  Example:
643    jkey_identifier("android.lens.facing") == "LENS_FACING"
644  """
645  return csym(what[what.find('.') + 1:])
646
647def jenum_value(enum_entry, enum_value):
648  """
649  Calculate the Java name for an integer enum value
650
651  Args:
652    enum: An enum-typed Entry node
653    value: An EnumValue node for the enum
654
655  Returns:
656    String representing the Java symbol
657  """
658
659  cname = csym(enum_entry.name)
660  return cname[cname.find('_') + 1:] + '_' + enum_value.name
661
662def generate_extra_javadoc_detail(entry):
663  """
664  Returns a function to add extra details for an entry into a string for inclusion into
665  javadoc. Adds information about units, the list of enum values for this key, and the valid
666  range.
667  """
668  def inner(text):
669    if entry.units:
670      text += '\n\n<b>Units</b>: %s\n' % (dedent(entry.units))
671    if entry.enum and not (entry.typedef and entry.typedef.languages.get('java')):
672      text += '\n\n<b>Possible values:</b>\n<ul>\n'
673      for value in entry.enum.values:
674        if not value.hidden:
675          text += '  <li>{@link #%s %s}</li>\n' % ( jenum_value(entry, value ), value.name )
676      text += '</ul>\n'
677    if entry.range:
678      if entry.enum and not (entry.typedef and entry.typedef.languages.get('java')):
679        text += '\n\n<b>Available values for this device:</b><br>\n'
680      else:
681        text += '\n\n<b>Range of valid values:</b><br>\n'
682      text += '%s\n' % (dedent(entry.range))
683    if entry.hwlevel != 'legacy': # covers any of (None, 'limited', 'full')
684      text += '\n\n<b>Optional</b> - This value may be {@code null} on some devices.\n'
685    if entry.hwlevel == 'full':
686      text += \
687        '\n<b>Full capability</b> - \n' + \
688        'Present on all camera devices that report being {@link CameraCharacteristics#INFO_SUPPORTED_HARDWARE_LEVEL_FULL HARDWARE_LEVEL_FULL} devices in the\n' + \
689        'android.info.supportedHardwareLevel key\n'
690    if entry.hwlevel == 'limited':
691      text += \
692        '\n<b>Limited capability</b> - \n' + \
693        'Present on all camera devices that report being at least {@link CameraCharacteristics#INFO_SUPPORTED_HARDWARE_LEVEL_LIMITED HARDWARE_LEVEL_LIMITED} devices in the\n' + \
694        'android.info.supportedHardwareLevel key\n'
695    if entry.hwlevel == 'legacy':
696      text += "\nThis key is available on all devices."
697
698    return text
699  return inner
700
701
702def javadoc(metadata, indent = 4):
703  """
704  Returns a function to format a markdown syntax text block as a
705  javadoc comment section, given a set of metadata
706
707  Args:
708    metadata: A Metadata instance, representing the the top-level root
709      of the metadata for cross-referencing
710    indent: baseline level of indentation for javadoc block
711  Returns:
712    A function that transforms a String text block as follows:
713    - Indent and * for insertion into a Javadoc comment block
714    - Trailing whitespace removed
715    - Entire body rendered via markdown to generate HTML
716    - All tag names converted to appropriate Javadoc {@link} with @see
717      for each tag
718
719  Example:
720    "This is a comment for Javadoc\n" +
721    "     with multiple lines, that should be   \n" +
722    "     formatted better\n" +
723    "\n" +
724    "    That covers multiple lines as well\n"
725    "    And references android.control.mode\n"
726
727    transforms to
728    "    * <p>This is a comment for Javadoc\n" +
729    "    * with multiple lines, that should be\n" +
730    "    * formatted better</p>\n" +
731    "    * <p>That covers multiple lines as well</p>\n" +
732    "    * and references {@link CaptureRequest#CONTROL_MODE android.control.mode}\n" +
733    "    *\n" +
734    "    * @see CaptureRequest#CONTROL_MODE\n"
735  """
736  def javadoc_formatter(text):
737    comment_prefix = " " * indent + " * ";
738
739    # render with markdown => HTML
740    javatext = md(text, JAVADOC_IMAGE_SRC_METADATA)
741
742    # Crossref tag names
743    kind_mapping = {
744        'static': 'CameraCharacteristics',
745        'dynamic': 'CaptureResult',
746        'controls': 'CaptureRequest' }
747
748    # Convert metadata entry "android.x.y.z" to form
749    # "{@link CaptureRequest#X_Y_Z android.x.y.z}"
750    def javadoc_crossref_filter(node):
751      if node.applied_visibility == 'public':
752        return '{@link %s#%s %s}' % (kind_mapping[node.kind],
753                                     jkey_identifier(node.name),
754                                     node.name)
755      else:
756        return node.name
757
758    # For each public tag "android.x.y.z" referenced, add a
759    # "@see CaptureRequest#X_Y_Z"
760    def javadoc_see_filter(node_set):
761      node_set = (x for x in node_set if x.applied_visibility == 'public')
762
763      text = '\n'
764      for node in node_set:
765        text = text + '\n@see %s#%s' % (kind_mapping[node.kind],
766                                      jkey_identifier(node.name))
767
768      return text if text != '\n' else ''
769
770    javatext = filter_tags(javatext, metadata, javadoc_crossref_filter, javadoc_see_filter)
771
772    def line_filter(line):
773      # Indent each line
774      # Add ' * ' to it for stylistic reasons
775      # Strip right side of trailing whitespace
776      return (comment_prefix + line).rstrip()
777
778    # Process each line with above filter
779    javatext = "\n".join(line_filter(i) for i in javatext.split("\n")) + "\n"
780
781    return javatext
782
783  return javadoc_formatter
784
785def dedent(text):
786  """
787  Remove all common indentation from every line but the 0th.
788  This will avoid getting <code> blocks when rendering text via markdown.
789  Ignoring the 0th line will also allow the 0th line not to be aligned.
790
791  Args:
792    text: A string of text to dedent.
793
794  Returns:
795    String dedented by above rules.
796
797  For example:
798    assertEquals("bar\nline1\nline2",   dedent("bar\n  line1\n  line2"))
799    assertEquals("bar\nline1\nline2",   dedent(" bar\n  line1\n  line2"))
800    assertEquals("bar\n  line1\nline2", dedent(" bar\n    line1\n  line2"))
801  """
802  text = textwrap.dedent(text)
803  text_lines = text.split('\n')
804  text_not_first = "\n".join(text_lines[1:])
805  text_not_first = textwrap.dedent(text_not_first)
806  text = text_lines[0] + "\n" + text_not_first
807
808  return text
809
810def md(text, img_src_prefix=""):
811    """
812    Run text through markdown to produce HTML.
813
814    This also removes all common indentation from every line but the 0th.
815    This will avoid getting <code> blocks in markdown.
816    Ignoring the 0th line will also allow the 0th line not to be aligned.
817
818    Args:
819      text: A markdown-syntax using block of text to format.
820      img_src_prefix: An optional string to prepend to each <img src="target"/>
821
822    Returns:
823      String rendered by markdown and other rules applied (see above).
824
825    For example, this avoids the following situation:
826
827      <!-- Input -->
828
829      <!--- can't use dedent directly since 'foo' has no indent -->
830      <notes>foo
831          bar
832          bar
833      </notes>
834
835      <!-- Bad Output -- >
836      <!-- if no dedent is done generated code looks like -->
837      <p>foo
838        <code><pre>
839          bar
840          bar</pre></code>
841      </p>
842
843    Instead we get the more natural expected result:
844
845      <!-- Good Output -->
846      <p>foo
847      bar
848      bar</p>
849
850    """
851    text = dedent(text)
852
853    # full list of extensions at http://pythonhosted.org/Markdown/extensions/
854    md_extensions = ['tables'] # make <table> with ASCII |_| tables
855    # render with markdown
856    text = markdown.markdown(text, md_extensions)
857
858    # prepend a prefix to each <img src="foo"> -> <img src="${prefix}foo">
859    text = re.sub(r'src="([^"]*)"', 'src="' + img_src_prefix + r'\1"', text)
860    return text
861
862def filter_tags(text, metadata, filter_function, summary_function = None):
863    """
864    Find all references to tags in the form outer_namespace.xxx.yyy[.zzz] in
865    the provided text, and pass them through filter_function and summary_function.
866
867    Used to linkify entry names in HMTL, javadoc output.
868
869    Args:
870      text: A string representing a block of text destined for output
871      metadata: A Metadata instance, the root of the metadata properties tree
872      filter_function: A Node->string function to apply to each node
873        when found in text; the string returned replaces the tag name in text.
874      summary_function: A Node list->string function that is provided the list of
875        unique tag nodes found in text, and which must return a string that is
876        then appended to the end of the text. The list is sorted alphabetically
877        by node name.
878    """
879
880    tag_set = set()
881    def name_match(name):
882      return lambda node: node.name == name
883
884    # Match outer_namespace.x.y or outer_namespace.x.y.z, making sure
885    # to grab .z and not just outer_namespace.x.y.  (sloppy, but since we
886    # check for validity, a few false positives don't hurt)
887    for outer_namespace in metadata.outer_namespaces:
888
889      tag_match = outer_namespace.name + \
890        r"\.([a-zA-Z0-9\n]+)\.([a-zA-Z0-9\n]+)(\.[a-zA-Z0-9\n]+)?([/]?)"
891
892      def filter_sub(match):
893        whole_match = match.group(0)
894        section1 = match.group(1)
895        section2 = match.group(2)
896        section3 = match.group(3)
897        end_slash = match.group(4)
898
899        # Don't linkify things ending in slash (urls, for example)
900        if end_slash:
901          return whole_match
902
903        candidate = ""
904
905        # First try a two-level match
906        candidate2 = "%s.%s.%s" % (outer_namespace.name, section1, section2)
907        got_two_level = False
908
909        node = metadata.find_first(name_match(candidate2.replace('\n','')))
910        if not node and '\n' in section2:
911          # Linefeeds are ambiguous - was the intent to add a space,
912          # or continue a lengthy name? Try the former now.
913          candidate2b = "%s.%s.%s" % (outer_namespace.name, section1, section2[:section2.find('\n')])
914          node = metadata.find_first(name_match(candidate2b))
915          if node:
916            candidate2 = candidate2b
917
918        if node:
919          # Have two-level match
920          got_two_level = True
921          candidate = candidate2
922        elif section3:
923          # Try three-level match
924          candidate3 = "%s%s" % (candidate2, section3)
925          node = metadata.find_first(name_match(candidate3.replace('\n','')))
926
927          if not node and '\n' in section3:
928            # Linefeeds are ambiguous - was the intent to add a space,
929            # or continue a lengthy name? Try the former now.
930            candidate3b = "%s%s" % (candidate2, section3[:section3.find('\n')])
931            node = metadata.find_first(name_match(candidate3b))
932            if node:
933              candidate3 = candidate3b
934
935          if node:
936            # Have 3-level match
937            candidate = candidate3
938
939        # Replace match with crossref or complain if a likely match couldn't be matched
940
941        if node:
942          tag_set.add(node)
943          return whole_match.replace(candidate,filter_function(node))
944        else:
945          print >> sys.stderr,\
946            "  WARNING: Could not crossref likely reference {%s}" % (match.group(0))
947          return whole_match
948
949      text = re.sub(tag_match, filter_sub, text)
950
951    if summary_function is not None:
952      return text + summary_function(sorted(tag_set, key=lambda x: x.name))
953    else:
954      return text
955
956def any_visible(section, kind_name, visibilities):
957  """
958  Determine if entries in this section have an applied visibility that's in
959  the list of given visibilities.
960
961  Args:
962    section: A section of metadata
963    kind_name: A name of the kind, i.e. 'dynamic' or 'static' or 'controls'
964    visibilities: An iterable of visibilities to match against
965
966  Returns:
967    True if the section has any entries with any of the given visibilities. False otherwise.
968  """
969
970  for inner_namespace in get_children_by_filtering_kind(section, kind_name,
971                                                        'namespaces'):
972    if any(filter_visibility(inner_namespace.merged_entries, visibilities)):
973      return True
974
975  return any(filter_visibility(get_children_by_filtering_kind(section, kind_name,
976                                                              'merged_entries'),
977                               visibilities))
978
979
980def filter_visibility(entries, visibilities):
981  """
982  Remove entries whose applied visibility is not in the supplied visibilities.
983
984  Args:
985    entries: An iterable of Entry nodes
986    visibilities: An iterable of visibilities to filter against
987
988  Yields:
989    An iterable of Entry nodes
990  """
991  return (e for e in entries if e.applied_visibility in visibilities)
992
993def remove_synthetic(entries):
994  """
995  Filter the given entries by removing those that are synthetic.
996
997  Args:
998    entries: An iterable of Entry nodes
999
1000  Yields:
1001    An iterable of Entry nodes
1002  """
1003  return (e for e in entries if not e.synthetic)
1004
1005def wbr(text):
1006  """
1007  Insert word break hints for the browser in the form of <wbr> HTML tags.
1008
1009  Word breaks are inserted inside an HTML node only, so the nodes themselves
1010  will not be changed. Attributes are also left unchanged.
1011
1012  The following rules apply to insert word breaks:
1013  - For characters in [ '.', '/', '_' ]
1014  - For uppercase letters inside a multi-word X.Y.Z (at least 3 parts)
1015
1016  Args:
1017    text: A string of text containing HTML content.
1018
1019  Returns:
1020    A string with <wbr> inserted by the above rules.
1021  """
1022  SPLIT_CHARS_LIST = ['.', '_', '/']
1023  SPLIT_CHARS = r'([.|/|_/,]+)' # split by these characters
1024  CAP_LETTER_MIN = 3 # at least 3 components split by above chars, i.e. x.y.z
1025  def wbr_filter(text):
1026      new_txt = text
1027
1028      # for johnyOrange.appleCider.redGuardian also insert wbr before the caps
1029      # => johny<wbr>Orange.apple<wbr>Cider.red<wbr>Guardian
1030      for words in text.split(" "):
1031        for char in SPLIT_CHARS_LIST:
1032          # match at least x.y.z, don't match x or x.y
1033          if len(words.split(char)) >= CAP_LETTER_MIN:
1034            new_word = re.sub(r"([a-z])([A-Z])", r"\1<wbr>\2", words)
1035            new_txt = new_txt.replace(words, new_word)
1036
1037      # e.g. X/Y/Z -> X/<wbr>Y/<wbr>/Z. also for X.Y.Z, X_Y_Z.
1038      new_txt = re.sub(SPLIT_CHARS, r"\1<wbr>", new_txt)
1039
1040      return new_txt
1041
1042  # Do not mangle HTML when doing the replace by using BeatifulSoup
1043  # - Use the 'html.parser' to avoid inserting <html><body> when decoding
1044  soup = bs4.BeautifulSoup(text, features='html.parser')
1045  wbr_tag = lambda: soup.new_tag('wbr') # must generate new tag every time
1046
1047  for navigable_string in soup.findAll(text=True):
1048      parent = navigable_string.parent
1049
1050      # Insert each '$text<wbr>$foo' before the old '$text$foo'
1051      split_by_wbr_list = wbr_filter(navigable_string).split("<wbr>")
1052      for (split_string, last) in enumerate_with_last(split_by_wbr_list):
1053          navigable_string.insert_before(split_string)
1054
1055          if not last:
1056            # Note that 'insert' will move existing tags to this spot
1057            # so make a new tag instead
1058            navigable_string.insert_before(wbr_tag())
1059
1060      # Remove the old unmodified text
1061      navigable_string.extract()
1062
1063  return soup.decode()
1064