You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			145 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Python
		
	
			
		
		
	
	
			145 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			Python
		
	
"""
 | 
						|
    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': [
 | 
						|
            # comments starting with #
 | 
						|
            (r'#.*$', Comment.Single),
 | 
						|
 | 
						|
            # multiline comments
 | 
						|
            (r'/\*', Comment.Multiline, 'comment'),
 | 
						|
 | 
						|
            # whitespace
 | 
						|
            (r'\s+', Text),
 | 
						|
 | 
						|
            # keywords
 | 
						|
            ('({})'.format('|'.join(re.escape(entry) + '\\b' for entry in keywords)), Keyword),
 | 
						|
 | 
						|
            # highlight the builtins
 | 
						|
            ('({})'.format('|'.join(re.escape(entry) + '\\b' for entry in builtins)),
 | 
						|
             Name.Builtin),
 | 
						|
 | 
						|
            (r'\b(true|false|null)\b', Name.Constant),
 | 
						|
 | 
						|
            # floats
 | 
						|
            (r'-?(\d+\.\d*|\.\d+)([eE][-+]?\d+)?', Number.Float),
 | 
						|
 | 
						|
            # integers
 | 
						|
            (r'-?[0-9]+', Number.Integer),
 | 
						|
 | 
						|
            # paths
 | 
						|
            (r'[\w.+-]*(\/[\w.+-]+)+', Literal),
 | 
						|
            (r'~(\/[\w.+-]+)+', Literal),
 | 
						|
            (r'\<[\w.+-]+(\/[\w.+-]+)*\>', Literal),
 | 
						|
 | 
						|
            # operators
 | 
						|
            ('({})'.format('|'.join(re.escape(entry) for entry in operators)),
 | 
						|
             Operator),
 | 
						|
 | 
						|
            # word operators
 | 
						|
            (r'\b(or|and)\b', Operator.Word),
 | 
						|
 | 
						|
            (r'\{', Punctuation, 'block'),
 | 
						|
 | 
						|
            # punctuations
 | 
						|
            ('({})'.format('|'.join(re.escape(entry) for entry in punctuations)), Punctuation),
 | 
						|
 | 
						|
            # strings
 | 
						|
            (r'"', String.Double, 'doublequote'),
 | 
						|
            (r"''", String.Multiline, 'multiline'),
 | 
						|
 | 
						|
            # urls
 | 
						|
            (r'[a-zA-Z][a-zA-Z0-9\+\-\.]*\:[\w%/?:@&=+$,\\.!~*\'-]+', Literal),
 | 
						|
 | 
						|
            # names of variables
 | 
						|
            (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'),
 | 
						|
            # TODO: we should probably escape also here ''${ \${
 | 
						|
            (r"\$\{", String.Interpol, '#push'),
 | 
						|
            include('root'),
 | 
						|
        ],
 | 
						|
        'block': [
 | 
						|
            (r"\}", Punctuation, '#pop'),
 | 
						|
            include('root'),
 | 
						|
        ],
 | 
						|
    }
 | 
						|
 | 
						|
    def analyse_text(text):
 | 
						|
        rv = 0.0
 | 
						|
        # TODO: let/in
 | 
						|
        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
 |