lib/rouge/lexers/erlang.rb



# -*- coding: utf-8 -*- #

module Rouge
  module Lexers
    class Erlang < RegexLexer
      title "Erlang"
      desc "The Erlang programming language (erlang.org)"
      tag 'erlang'
      aliases 'erl'
      filenames '*.erl', '*.hrl'

      mimetypes 'text/x-erlang', 'application/x-erlang'

      def self.analyze_text(text)
        return 0.3 if text =~ /^-module[(]\w+[)][.]/
      end

      keywords = %w(
        after begin case catch cond end fun if
        let of query receive try when
      )

      builtins = %w(
        abs append_element apply atom_to_list binary_to_list
        bitstring_to_list binary_to_term bit_size bump_reductions
        byte_size cancel_timer check_process_code delete_module
        demonitor disconnect_node display element erase exit
        float float_to_list fun_info fun_to_list
        function_exported garbage_collect get get_keys
        group_leader hash hd integer_to_list iolist_to_binary
        iolist_size is_atom is_binary is_bitstring is_boolean
        is_builtin is_float is_function is_integer is_list
        is_number is_pid is_port is_process_alive is_record
        is_reference is_tuple length link list_to_atom
        list_to_binary list_to_bitstring list_to_existing_atom
        list_to_float list_to_integer list_to_pid list_to_tuple
        load_module localtime_to_universaltime make_tuple md5
        md5_final md5_update memory module_loaded monitor
        monitor_node node nodes open_port phash phash2
        pid_to_list port_close port_command port_connect
        port_control port_call port_info port_to_list
        process_display process_flag process_info purge_module
        put read_timer ref_to_list register resume_process
        round send send_after send_nosuspend set_cookie
        setelement size spawn spawn_link spawn_monitor
        spawn_opt split_binary start_timer statistics
        suspend_process system_flag system_info system_monitor
        system_profile term_to_binary tl trace trace_delivered
        trace_info trace_pattern trunc tuple_size tuple_to_list
        universaltime_to_localtime unlink unregister whereis
      )

      operators = %r{(\+\+?|--?|\*|/|<|>|/=|=:=|=/=|=<|>=|==?|<-|!|\?)}
      word_operators = %w(
        and andalso band bnot bor bsl bsr bxor
        div not or orelse rem xor
      )

      atom_re = %r{(?:[a-z][a-zA-Z0-9_]*|'[^\n']*[^\\]')}

      variable_re = %r{(?:[A-Z_][a-zA-Z0-9_]*)}

      escape_re = %r{(?:\\(?:[bdefnrstv\'"\\/]|[0-7][0-7]?[0-7]?|\^[a-zA-Z]))}

      macro_re = %r{(?:#{variable_re}|#{atom_re})}

      base_re = %r{(?:[2-9]|[12][0-9]|3[0-6])}

      state :root do
        rule(/\s+/, Text)
        rule(/%.*\n/, Comment)
        rule(%r{(#{keywords.join('|')})\b}, Keyword)
        rule(%r{(#{builtins.join('|')})\b}, Name::Builtin)
        rule(%r{(#{word_operators.join('|')})\b}, Operator::Word)
        rule(/^-/, Punctuation, :directive)
        rule(operators, Operator)
        rule(/"/, Str, :string)
        rule(/<</, Name::Label)
        rule(/>>/, Name::Label)
        rule %r{(#{atom_re})(:)} do
          groups Name::Namespace, Punctuation
        end
        rule %r{(?:^|(?<=:))(#{atom_re})(\s*)(\()} do
          groups Name::Function, Text, Punctuation
        end
        rule(%r{[+-]?#{base_re}#[0-9a-zA-Z]+}, Num::Integer)
        rule(/[+-]?\d+/, Num::Integer)
        rule(/[+-]?\d+.\d+/, Num::Float)
        rule(%r{[\]\[:_@\".{}()|;,]}, Punctuation)
        rule(variable_re, Name::Variable)
        rule(atom_re, Name)
        rule(%r{\?#{macro_re}}, Name::Constant)
        rule(%r{\$(?:#{escape_re}|\\[ %]|[^\\])}, Str::Char)
        rule(%r{##{atom_re}(:?\.#{atom_re})?}, Name::Label)
      end

      state :string do
        rule(escape_re, Str::Escape)
        rule(/"/, Str, :pop!)
        rule(%r{~[0-9.*]*[~#+bBcdefginpPswWxX]}, Str::Interpol)
        rule(%r{[^"\\~]+}, Str)
        rule(/~/, Str)
      end

      state :directive do
        rule %r{(define)(\s*)(\()(#{macro_re})} do
          groups Name::Entity, Text, Punctuation, Name::Constant
          pop!
        end
        rule %r{(record)(\s*)(\()(#{macro_re})} do
          groups Name::Entity, Text, Punctuation, Name::Label
          pop!
        end
        rule(atom_re, Name::Entity, :pop!)
      end
    end
  end
end