That way, we can set it to "" for old lexers, and check that it's present on new lexers. (In the future, we might also use it for better presentation in the documentation.)
153 lines
6 KiB
Python
153 lines
6 KiB
Python
"""
|
|
pygments.lexers.qvt
|
|
~~~~~~~~~~~~~~~~~~~
|
|
|
|
Lexer for QVT Operational language.
|
|
|
|
:copyright: Copyright 2006-2023 by the Pygments team, see AUTHORS.
|
|
:license: BSD, see LICENSE for details.
|
|
"""
|
|
|
|
from pygments.lexer import RegexLexer, bygroups, include, combined, default, \
|
|
words
|
|
from pygments.token import Text, Comment, Operator, Keyword, Punctuation, \
|
|
Name, String, Number
|
|
|
|
__all__ = ['QVToLexer']
|
|
|
|
|
|
class QVToLexer(RegexLexer):
|
|
"""
|
|
For the QVT Operational Mapping language.
|
|
|
|
Reference for implementing this: «Meta Object Facility (MOF) 2.0
|
|
Query/View/Transformation Specification», Version 1.1 - January 2011
|
|
(https://www.omg.org/spec/QVT/1.1/), see §8.4, «Concrete Syntax» in
|
|
particular.
|
|
|
|
Notable tokens assignments:
|
|
|
|
- Name.Class is assigned to the identifier following any of the following
|
|
keywords: metamodel, class, exception, primitive, enum, transformation
|
|
or library
|
|
|
|
- Name.Function is assigned to the names of mappings and queries
|
|
|
|
- Name.Builtin.Pseudo is assigned to the pre-defined variables 'this',
|
|
'self' and 'result'.
|
|
"""
|
|
# With obvious borrowings & inspiration from the Java, Python and C lexers
|
|
|
|
name = 'QVTO'
|
|
aliases = ['qvto', 'qvt']
|
|
filenames = ['*.qvto']
|
|
url = 'https://www.omg.org/spec/QVT/1.1'
|
|
version_added = ''
|
|
|
|
tokens = {
|
|
'root': [
|
|
(r'\n', Text),
|
|
(r'[^\S\n]+', Text),
|
|
(r'(--|//)(\s*)(directive:)?(.*)$',
|
|
bygroups(Comment, Comment, Comment.Preproc, Comment)),
|
|
# Uncomment the following if you want to distinguish between
|
|
# '/*' and '/**', à la javadoc
|
|
# (r'/[*]{2}(.|\n)*?[*]/', Comment.Multiline),
|
|
(r'/[*](.|\n)*?[*]/', Comment.Multiline),
|
|
(r'\\\n', Text),
|
|
(r'(and|not|or|xor|##?)\b', Operator.Word),
|
|
(r'(:{1,2}=|[-+]=)\b', Operator.Word),
|
|
(r'(@|<<|>>)\b', Keyword), # stereotypes
|
|
(r'!=|<>|==|=|!->|->|>=|<=|[.]{3}|[+/*%=<>&|.~]', Operator),
|
|
(r'[]{}:(),;[]', Punctuation),
|
|
(r'(true|false|unlimited|null)\b', Keyword.Constant),
|
|
(r'(this|self|result)\b', Name.Builtin.Pseudo),
|
|
(r'(var)\b', Keyword.Declaration),
|
|
(r'(from|import)\b', Keyword.Namespace, 'fromimport'),
|
|
(r'(metamodel|class|exception|primitive|enum|transformation|'
|
|
r'library)(\s+)(\w+)',
|
|
bygroups(Keyword.Word, Text, Name.Class)),
|
|
(r'(exception)(\s+)(\w+)',
|
|
bygroups(Keyword.Word, Text, Name.Exception)),
|
|
(r'(main)\b', Name.Function),
|
|
(r'(mapping|helper|query)(\s+)',
|
|
bygroups(Keyword.Declaration, Text), 'operation'),
|
|
(r'(assert)(\s+)\b', bygroups(Keyword, Text), 'assert'),
|
|
(r'(Bag|Collection|Dict|OrderedSet|Sequence|Set|Tuple|List)\b',
|
|
Keyword.Type),
|
|
include('keywords'),
|
|
('"', String, combined('stringescape', 'dqs')),
|
|
("'", String, combined('stringescape', 'sqs')),
|
|
include('name'),
|
|
include('numbers'),
|
|
# (r'([a-zA-Z_]\w*)(::)([a-zA-Z_]\w*)',
|
|
# bygroups(Text, Text, Text)),
|
|
],
|
|
|
|
'fromimport': [
|
|
(r'(?:[ \t]|\\\n)+', Text),
|
|
(r'[a-zA-Z_][\w.]*', Name.Namespace),
|
|
default('#pop'),
|
|
],
|
|
|
|
'operation': [
|
|
(r'::', Text),
|
|
(r'(.*::)([a-zA-Z_]\w*)([ \t]*)(\()',
|
|
bygroups(Text, Name.Function, Text, Punctuation), '#pop')
|
|
],
|
|
|
|
'assert': [
|
|
(r'(warning|error|fatal)\b', Keyword, '#pop'),
|
|
default('#pop'), # all else: go back
|
|
],
|
|
|
|
'keywords': [
|
|
(words((
|
|
'abstract', 'access', 'any', 'assert', 'blackbox', 'break',
|
|
'case', 'collect', 'collectNested', 'collectOne', 'collectselect',
|
|
'collectselectOne', 'composes', 'compute', 'configuration',
|
|
'constructor', 'continue', 'datatype', 'default', 'derived',
|
|
'disjuncts', 'do', 'elif', 'else', 'end', 'endif', 'except',
|
|
'exists', 'extends', 'forAll', 'forEach', 'forOne', 'from', 'if',
|
|
'implies', 'in', 'inherits', 'init', 'inout', 'intermediate',
|
|
'invresolve', 'invresolveIn', 'invresolveone', 'invresolveoneIn',
|
|
'isUnique', 'iterate', 'late', 'let', 'literal', 'log', 'map',
|
|
'merges', 'modeltype', 'new', 'object', 'one', 'ordered', 'out',
|
|
'package', 'population', 'property', 'raise', 'readonly',
|
|
'references', 'refines', 'reject', 'resolve', 'resolveIn',
|
|
'resolveone', 'resolveoneIn', 'return', 'select', 'selectOne',
|
|
'sortedBy', 'static', 'switch', 'tag', 'then', 'try', 'typedef',
|
|
'unlimited', 'uses', 'when', 'where', 'while', 'with', 'xcollect',
|
|
'xmap', 'xselect'), suffix=r'\b'), Keyword),
|
|
],
|
|
|
|
# There is no need to distinguish between String.Single and
|
|
# String.Double: 'strings' is factorised for 'dqs' and 'sqs'
|
|
'strings': [
|
|
(r'[^\\\'"\n]+', String),
|
|
# quotes, percents and backslashes must be parsed one at a time
|
|
(r'[\'"\\]', String),
|
|
],
|
|
'stringescape': [
|
|
(r'\\([\\btnfr"\']|u[0-3][0-7]{2}|u[0-7]{1,2})', String.Escape)
|
|
],
|
|
'dqs': [ # double-quoted string
|
|
(r'"', String, '#pop'),
|
|
(r'\\\\|\\"', String.Escape),
|
|
include('strings')
|
|
],
|
|
'sqs': [ # single-quoted string
|
|
(r"'", String, '#pop'),
|
|
(r"\\\\|\\'", String.Escape),
|
|
include('strings')
|
|
],
|
|
'name': [
|
|
(r'[a-zA-Z_]\w*', Name),
|
|
],
|
|
# numbers: excerpt taken from the python lexer
|
|
'numbers': [
|
|
(r'(\d+\.\d*|\d*\.\d+)([eE][+-]?[0-9]+)?', Number.Float),
|
|
(r'\d+[eE][+-]?[0-9]+', Number.Float),
|
|
(r'\d+', Number.Integer)
|
|
],
|
|
}
|