/*
* SPDX-FileCopyrightText: 2021-2023 The Refinery Authors
*
* SPDX-License-Identifier: EPL-2.0
*/
package tools.refinery.language.parser.antlr;
import com.google.inject.Inject;
import com.google.inject.Singleton;
import org.eclipse.xtext.*;
import org.eclipse.xtext.parser.antlr.ITokenDefProvider;
import tools.refinery.language.services.ProblemGrammarAccess;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Set;
@Singleton
public class IdentifierTokenProvider {
private final int[] identifierTokensArray;
@Inject
private IdentifierTokenProvider(Initializer initializer) {
this.identifierTokensArray = initializer.getIdentifierTokesArray();
}
public boolean isIdentifierToken(int tokenId) {
for (int identifierTokenId : identifierTokensArray) {
if (identifierTokenId == tokenId) {
return true;
}
}
return false;
}
private static class Initializer {
@Inject
private ITokenDefProvider tokenDefProvider;
@Inject
private ProblemGrammarAccess problemGrammarAccess;
private HashMap valueToTokenIdMap;
private Set identifierTokens;
public int[] getIdentifierTokesArray() {
createValueToTokenIdMap();
identifierTokens = new HashSet<>();
collectIdentifierTokensFromRule(problemGrammarAccess.getIdentifierRule());
var identifierTokensArray = new int[identifierTokens.size()];
int i = 0;
for (var tokenId : identifierTokens) {
identifierTokensArray[i] = tokenId;
i++;
}
return identifierTokensArray;
}
private void createValueToTokenIdMap() {
var tokenIdToValueMap = tokenDefProvider.getTokenDefMap();
valueToTokenIdMap = new HashMap<>(tokenIdToValueMap.size());
for (var entry : tokenIdToValueMap.entrySet()) {
valueToTokenIdMap.put(entry.getValue(), entry.getKey());
}
}
private void collectIdentifierTokensFromRule(AbstractRule rule) {
if (rule instanceof TerminalRule) {
collectToken("RULE_" + rule.getName());
return;
}
collectIdentifierTokensFromElement(rule.getAlternatives());
}
private void collectIdentifierTokensFromElement(AbstractElement element) {
if (element instanceof Alternatives alternatives) {
for (var alternative : alternatives.getElements()) {
collectIdentifierTokensFromElement(alternative);
}
} else if (element instanceof RuleCall ruleCall) {
collectIdentifierTokensFromRule(ruleCall.getRule());
} else if (element instanceof Keyword keyword) {
collectToken("'" + keyword.getValue() + "'");
} else {
throw new IllegalArgumentException("Unknown Xtext grammar element: " + element);
}
}
private void collectToken(String value) {
identifierTokens.add(valueToTokenIdMap.get(value));
}
}
}