%YAML 1.2
---
# See http://www.sublimetext.com/docs/3/syntax.html
file_extensions:
  - log
scope: text.log
variables:
  ipv4_part: (?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)
  hours_minutes_seconds: (?:[01]\d|2[0-3]):(?:[0-5]\d):(?:[0-5]\d)
  error: \b(?i:fail(?:ure|ed)?|error|exception|fatal|critical)\b
  warning: \b(?i:warn(?:ing)?)\b
  info: \b(?i:info)\b
  debug: \b(?i:debug)\b
contexts:
  main:
    - include: log_level_lines
    - include: main_without_log_level_line
  main_without_log_level_line:
    - match: (\w+)(=)
      captures:
        1: variable.parameter.log
        2: keyword.operator.assignment.log
    - match: (')([^']*)(')
      scope: string.quoted.single.log
      captures:
        1: punctuation.definition.string.begin.log
        3: punctuation.definition.string.end.log
    - match: \"
      captures:
        1: punctuation.definition.string.begin.log
      push: double_quoted_string
    - include: dates
    - include: ip_addresses
    - include: numbers
    - include: log_levels
    #- include: scope:text.html.markdown#autolink-inet
    - match: \b\w+:/{2,3}
      scope: markup.underline.link.scheme.log
      push: url-host
  log_level_lines:
    - match: ^(?=.*{{error}})
      push:
        - error_line
        - main_pop_at_eol
    - match: ^(?=.*{{warning}})
      push:
        - warning_line
        - main_pop_at_eol
    - match: ^(?=.*{{info}})
      push:
        - info_line
        - main_pop_at_eol
    - match: ^(?=.*{{debug}})
      push:
        - debug_line
        - main_pop_at_eol
  log_levels:
    - match: '{{error}}'
      scope: markup.error.log
    - match: '{{warning}}'
      scope: markup.warning.log
    - match: '{{info}}'
      scope: markup.info.log
    - match: '{{debug}}'
      scope: markup.info.log
  error_line:
    - meta_scope: meta.annotation.error-line.log
    - include: immediately_pop
  warning_line:
    - meta_scope: meta.annotation.warning-line.log
    - include: immediately_pop
  info_line:
    - meta_scope: meta.annotation.info-line.log
    - include: immediately_pop
  debug_line:
    - meta_scope: meta.annotation.debug-line.log
    - include: immediately_pop
  immediately_pop:
    - match: ''
      pop: true
  pop_at_eol:
    - match: $
      pop: true
  main_pop_at_eol:
    - include: main_without_log_level_line
    - include: pop_at_eol
  dates:
    - match: \b\d{4}-\d{2}-\d{2}(?=\b|T)
      scope: meta.date.log meta.number.integer.decimal.log constant.numeric.value.log
      push: maybe_date_time_separator
    - match: \b\d{4}/\d{2}/\d{2}(?=\b|T)
      scope: meta.date.log meta.number.integer.decimal.log constant.numeric.value.log
      push: maybe_date_time_separator
    - match: \b(?={{hours_minutes_seconds}})
      push: time
  time:
    - match: (?:{{hours_minutes_seconds}})(?:(\.)\d{3})?\b
      scope: meta.time.log meta.number.integer.decimal.log constant.numeric.value.log
      captures:
        1: punctuation.separator.decimal.log
    - include: immediately_pop
  maybe_date_time_separator:
    - match: T(?={{hours_minutes_seconds}})
      scope: meta.date.log meta.time.log keyword.other.log
      set: time
    - include: immediately_pop
  ip_addresses:
    - match: \b(?=(?:{{ipv4_part}}\.){3}{{ipv4_part}}\b)
      push:
        - meta_scope: meta.ipaddress.v4.log meta.number.integer.decimal.log
        - match: \d+
          scope: constant.numeric.value.log
        - match: \.
          scope: punctuation.separator.sequence.log
        - include: immediately_pop
    - match: (?=(?:\h{0,4}:){2,6}\h{1,4}\b)
      push:
        - meta_scope: meta.ipaddress.v6.log meta.number.integer.hexadecimal.log
        - match: \h{1,4}
          scope: constant.numeric.value.log
        - match: ':'
          scope: punctuation.separator.sequence.log
        - include: immediately_pop
  numbers:
    - match: \b(0x)(\h+)(?:(\.)(\h+))?\b
      scope: meta.number.float.hexadecimal.log
      captures:
        1: constant.numeric.base.log
        2: constant.numeric.value.log
        3: constant.numeric.value.log punctuation.separator.decimal.log
        4: constant.numeric.value.log
    - match: \b\d+(\.)\d+\b
      scope: meta.number.float.log constant.numeric.value.log
      captures:
        1: punctuation.separator.decimal.log
    - match: \b\d+\b
      scope: meta.number.integer.log constant.numeric.value.log
  url-host:
    - meta_content_scope: markup.underline.link.host.log
    - match: \[
      scope: punctuation.section.brackets.begin.log
      push:
        - match: \]
          scope: punctuation.section.brackets.end.log
          pop: true
        - include: ip_addresses
    - match: '[^:/\s]+'
    - match: ''
      set: url-after-host
  url-after-host:
    - match: (:)(\d+)
      scope: markup.underline.link.port.log
      captures:
        1: punctuation.separator.mapping.port.log
        2: constant.numeric.value.log
    #- include: scope:text.html.markdown#url-escaped-chars
    - match: (%)\h{2}
      scope: markup.underline.link.path.log constant.character.escape.url.log
      captures:
        1: punctuation.definition.escape.log
    # After a valid domain, zero or more non-space non-< characters may follow
    - match: (?=[?!.,:*_~]*[\s<]) # Trailing punctuation (specifically, ?, !, ., ,, :, *, _, and ~) will not be considered part of the autolink, though they may be included in the interior of the link
      pop: true
    - match: \( # When an autolink ends in ), we scan the entire autolink for the total number of parentheses. If there is a greater number of closing parentheses than opening ones, we don’t consider the last character part of the autolink, in order to facilitate including an autolink inside a parenthesis
      push:
        - meta_scope: markup.underline.link.path.log
        - match: (?=[?!.,:*_~]*[\s<])
          pop: true
        - match: \)
          pop: true
    - match: (?=\)[?!.,:*_~]*[\s<])
      pop: true
    - match: '[^?!.,:*_~\s<&()%]+|\S'
      scope: markup.underline.link.path.log
    - include: immediately_pop
  double_quoted_string:
    - meta_scope: string.quoted.double.log
    - match: \\"
      scope: constant.character.escape.log
    - match: \\n
      scope: constant.character.escape.log
    - match: \"
      scope: punctuation.definition.string.end.log
      pop: true