|
""" |
|
pygments.lexers.nix |
|
~~~~~~~~~~~~~~~~~~~ |
|
|
|
Lexers for the NixOS Nix language. |
|
|
|
:copyright: Copyright 2006-2025 by the Pygments team, see AUTHORS. |
|
:license: BSD, see LICENSE for details. |
|
""" |
|
|
|
import re |
|
|
|
from pygments.lexer import RegexLexer, include |
|
from pygments.token import Text, Comment, Operator, Keyword, Name, String, \ |
|
Number, Punctuation, Literal |
|
|
|
__all__ = ['NixLexer'] |
|
|
|
|
|
class NixLexer(RegexLexer): |
|
""" |
|
For the Nix language. |
|
""" |
|
|
|
name = 'Nix' |
|
url = 'http://nixos.org/nix/' |
|
aliases = ['nixos', 'nix'] |
|
filenames = ['*.nix'] |
|
mimetypes = ['text/x-nix'] |
|
version_added = '2.0' |
|
|
|
keywords = ['rec', 'with', 'let', 'in', 'inherit', 'assert', 'if', |
|
'else', 'then', '...'] |
|
builtins = ['import', 'abort', 'baseNameOf', 'dirOf', 'isNull', 'builtins', |
|
'map', 'removeAttrs', 'throw', 'toString', 'derivation'] |
|
operators = ['++', '+', '?', '.', '!', '//', '==', '/', |
|
'!=', '&&', '||', '->', '=', '<', '>', '*', '-'] |
|
|
|
punctuations = ["(", ")", "[", "]", ";", "{", "}", ":", ",", "@"] |
|
|
|
tokens = { |
|
'root': [ |
|
|
|
(r'#.*$', Comment.Single), |
|
|
|
|
|
(r'/\*', Comment.Multiline, 'comment'), |
|
|
|
|
|
(r'\s+', Text), |
|
|
|
|
|
('({})'.format('|'.join(re.escape(entry) + '\\b' for entry in keywords)), Keyword), |
|
|
|
|
|
('({})'.format('|'.join(re.escape(entry) + '\\b' for entry in builtins)), |
|
Name.Builtin), |
|
|
|
(r'\b(true|false|null)\b', Name.Constant), |
|
|
|
|
|
(r'-?(\d+\.\d*|\.\d+)([eE][-+]?\d+)?', Number.Float), |
|
|
|
|
|
(r'-?[0-9]+', Number.Integer), |
|
|
|
|
|
(r'[\w.+-]*(\/[\w.+-]+)+', Literal), |
|
(r'~(\/[\w.+-]+)+', Literal), |
|
(r'\<[\w.+-]+(\/[\w.+-]+)*\>', Literal), |
|
|
|
|
|
('({})'.format('|'.join(re.escape(entry) for entry in operators)), |
|
Operator), |
|
|
|
|
|
(r'\b(or|and)\b', Operator.Word), |
|
|
|
(r'\{', Punctuation, 'block'), |
|
|
|
|
|
('({})'.format('|'.join(re.escape(entry) for entry in punctuations)), Punctuation), |
|
|
|
|
|
(r'"', String.Double, 'doublequote'), |
|
(r"''", String.Multiline, 'multiline'), |
|
|
|
|
|
(r'[a-zA-Z][a-zA-Z0-9\+\-\.]*\:[\w%/?:@&=+$,\\.!~*\'-]+', Literal), |
|
|
|
|
|
(r'[\w-]+(?=\s*=)', String.Symbol), |
|
(r'[a-zA-Z_][\w\'-]*', Text), |
|
|
|
(r"\$\{", String.Interpol, 'antiquote'), |
|
], |
|
'comment': [ |
|
(r'[^/*]+', Comment.Multiline), |
|
(r'/\*', Comment.Multiline, '#push'), |
|
(r'\*/', Comment.Multiline, '#pop'), |
|
(r'[*/]', Comment.Multiline), |
|
], |
|
'multiline': [ |
|
(r"''(\$|'|\\n|\\r|\\t|\\)", String.Escape), |
|
(r"''", String.Multiline, '#pop'), |
|
(r'\$\{', String.Interpol, 'antiquote'), |
|
(r"[^'\$]+", String.Multiline), |
|
(r"\$[^\{']", String.Multiline), |
|
(r"'[^']", String.Multiline), |
|
(r"\$(?=')", String.Multiline), |
|
], |
|
'doublequote': [ |
|
(r'\\(\\|"|\$|n)', String.Escape), |
|
(r'"', String.Double, '#pop'), |
|
(r'\$\{', String.Interpol, 'antiquote'), |
|
(r'[^"\\\$]+', String.Double), |
|
(r'\$[^\{"]', String.Double), |
|
(r'\$(?=")', String.Double), |
|
(r'\\', String.Double), |
|
], |
|
'antiquote': [ |
|
(r"\}", String.Interpol, '#pop'), |
|
|
|
(r"\$\{", String.Interpol, '#push'), |
|
include('root'), |
|
], |
|
'block': [ |
|
(r"\}", Punctuation, '#pop'), |
|
include('root'), |
|
], |
|
} |
|
|
|
def analyse_text(text): |
|
rv = 0.0 |
|
|
|
if re.search(r'import.+?<[^>]+>', text): |
|
rv += 0.4 |
|
if re.search(r'mkDerivation\s+(\(|\{|rec)', text): |
|
rv += 0.4 |
|
if re.search(r'=\s+mkIf\s+', text): |
|
rv += 0.4 |
|
if re.search(r'\{[a-zA-Z,\s]+\}:', text): |
|
rv += 0.1 |
|
return rv |
|
|