All Downloads are FREE. Search and download functionalities are using the official Maven repository.

chapi.ast.protobuf.ProtobufAnalyser.kt Maven / Gradle / Ivy

package chapi.ast.protobuf

import chapi.ast.antlr.Protobuf2Lexer
import chapi.ast.antlr.Protobuf2Parser
import chapi.ast.antlr.Protobuf3Lexer
import chapi.ast.antlr.Protobuf3Parser
import chapi.domain.core.CodeContainer
import chapi.parser.Analyser
import org.antlr.v4.runtime.*
import org.antlr.v4.runtime.tree.ParseTreeWalker

class ProtobufAnalyser : Analyser {
    override fun analysis(code: String, filePath: String): CodeContainer {
        val protobuf3Analyser = Protobuf3Analyser()
        val analysis = protobuf3Analyser.analysis(code, filePath)
        if (protobuf3Analyser.errors.isNotEmpty()) {
            val protobuf2Analyser = Protobuf2Analyser()
            return protobuf2Analyser.analysis(code, filePath)
        }

        return analysis
    }
}


class SyntaxErrorListener : BaseErrorListener() {
    val errorMessages = mutableListOf()

    override fun syntaxError(
        recognizer: Recognizer<*, *>?,
        offendingSymbol: Any?,
        line: Int,
        charPositionInLine: Int,
        msg: String?,
        e: RecognitionException?,
    ) {
        errorMessages.add("line $line:$charPositionInLine - $msg")
    }

    fun hasErrors(): Boolean {
        return errorMessages.isNotEmpty()
    }

    fun getErrors(): List {
        return errorMessages
    }
}

class Protobuf3Analyser : Analyser {
    var errors = listOf()

    override fun analysis(code: String, filePath: String): CodeContainer {
        val (context, errorListener) = this.parse(code)
        val listener = ProtobufFullIdentListener(fileName = filePath)
        ParseTreeWalker().walk(listener, context)

        val nodeInfo = listener.getNodeInfo()

        if (errorListener.hasErrors()) {
            errors = errorListener.getErrors()
        }

        return nodeInfo
    }

    private fun parse(str: String): Pair {
        val fromString = CharStreams.fromString(str)
        val lexer = Protobuf3Lexer(fromString)
        val tokenStream = CommonTokenStream(lexer)
        val parser = Protobuf3Parser(tokenStream)

        // 创建并添加错误监听器
        val errorListener = SyntaxErrorListener()
        parser.removeErrorListeners()  // 移除默认控制台错误输出
        parser.addErrorListener(errorListener)

        return parser.proto() to errorListener
    }
}

class Protobuf2Analyser : Analyser {
    override fun analysis(code: String, filePath: String): CodeContainer {
        val context = this.parse(code).proto()
        val listener = Protobuf2FullIdentListener(fileName = filePath)
        ParseTreeWalker().walk(listener, context)

        return listener.getNodeInfo()
    }

    private fun parse(str: String): Protobuf2Parser {
        val fromString = CharStreams.fromString(str)
        val lexer = Protobuf2Lexer(fromString)
        val tokenStream = CommonTokenStream(lexer)
        val parser = Protobuf2Parser(tokenStream)
        return parser
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy