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

com.tencent.trpc.codegen.protoc.Protoc Maven / Gradle / Ivy

The newest version!
/*
 * Tencent is pleased to support the open source community by making tRPC available.
 *
 * Copyright (C) 2023 THL A29 Limited, a Tencent company. 
 * All rights reserved.
 *
 * If you have downloaded a copy of the tRPC source code from Tencent,
 * please note that tRPC source code is licensed under the Apache 2.0 License,
 * A copy of the Apache 2.0 License can be found in the LICENSE file.
 */

package com.tencent.trpc.codegen.protoc;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import org.apache.commons.io.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

/**
 * Wrapper of a protoc binary executable file.
 */
public class Protoc {
    private static final Logger log = LoggerFactory.getLogger(Protoc.class);
    private static final String PROTOC = "protoc";
    private final String protocExecutable;
    private final Map plugins;

    public Protoc() {
        this(PROTOC);
    }

    public Protoc(String protocExecutable) {
        this(protocExecutable, Collections.emptyMap());
    }

    public Protoc(String protocExecutable, Map plugins) {
        this.protocExecutable = protocExecutable;
        this.plugins = plugins;
    }

    /**
     * Read .proto files and generate descriptor set file.
     *
     * @param instruction protoc instruction
     * @return {@link ProtocExecutionResult}
     */
    public ProtocExecutionResult generateDescriptorSet(ProtocInstruction instruction) {
        return runProtoc(assembleProtocProcess(instruction.getSourceDirectory(),
                instruction.getSourceFiles(), instruction.getImportPaths(), instruction.getPluginInstructions(),
                "--descriptor_set_out=" + instruction.getOutput(), "--include_imports"));
    }

    /**
     * Read .proto files and generate protoc stub codes.
     *
     * @param instruction protoc instruction
     * @return {@link ProtocExecutionResult}
     */
    public ProtocExecutionResult generateStub(ProtocInstruction instruction) {
        return runProtoc(assembleProtocProcess(instruction.getSourceDirectory(), instruction.getSourceFiles(),
                instruction.getImportPaths(), instruction.getPluginInstructions(),
                String.format("--%s_out=%s",
                        instruction.getLanguage().toString().toLowerCase(), instruction.getOutput())));
    }

    private ProcessBuilder assembleProtocProcess(Path sourceDirectory,
                                                 Collection sourceFiles,
                                                 List importPaths,
                                                 List pluginInstructions,
                                                 String... args) {
        List cmd = new ArrayList<>();
        cmd.add(protocExecutable);
        cmd.add("--proto_path=.");
        if (importPaths != null) {
            cmd.addAll(importPaths.stream()
                    .map(p -> "--proto_path=" + p)
                    .collect(Collectors.toList()));
        }
        pluginInstructions.forEach(instruction -> {
            ProtocPlugin plugin = plugins.get(instruction.getPluginName());
            if (plugin == null) {
                log.warn("cannot find plugin {}, skipped", instruction.getPluginName());
                return;
            }
            cmd.addAll(plugin.buildProtocArguments(instruction.getPluginArguments(),
                    instruction.getPluginOutput().toString()));
        });
        cmd.addAll(Arrays.asList(args));
        cmd.addAll(sourceFiles);
        return new ProcessBuilder().command(cmd).directory(sourceDirectory.toFile());
    }

    private ProtocExecutionResult runProtoc(ProcessBuilder processBuilder) {
        log.debug("executing protoc cmd: {}", processBuilder.command());
        Process p;
        try {
            p = processBuilder.start();
        } catch (IOException e) {
            return ProtocExecutionResult.fail("execute protoc failed", null, e);
        }
        return fetchProcessResult(p);
    }

    private ProtocExecutionResult fetchProcessResult(Process p) {
        StringBuilder outputBuilder = new StringBuilder();
        StringBuilder errorBuilder = new StringBuilder();

        // Using threads to asynchronously read output and error streams.
        Thread outputThread = new Thread(() -> {
            try (BufferedReader reader = new BufferedReader(
                    new InputStreamReader(p.getInputStream(), StandardCharsets.UTF_8))) {
                String line;
                while ((line = reader.readLine()) != null) {
                    outputBuilder.append(line).append(System.lineSeparator());
                }
            } catch (IOException e) {
                log.error("failed to read output stream", e);
            }
        });

        Thread errorThread = new Thread(() -> {
            try (BufferedReader reader = new BufferedReader(
                    new InputStreamReader(p.getErrorStream(), StandardCharsets.UTF_8))) {
                String line;
                while ((line = reader.readLine()) != null) {
                    errorBuilder.append(line).append(System.lineSeparator());
                }
            } catch (IOException e) {
                log.error("failed to read error stream", e);
            }
        });

        outputThread.start();
        errorThread.start();

        try {
            int code = p.waitFor();
            outputThread.join();
            errorThread.join();

            if (code == 0) {
                return ProtocExecutionResult.success(outputBuilder.toString());
            } else {
                return ProtocExecutionResult.fail("run protoc failed with exit-code " + code,
                        errorBuilder.toString(), null);
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            return ProtocExecutionResult.fail("execute protoc failed", null, e);
        }
    }

    @Override
    public String toString() {
        return "Protoc{protocExecutable='" + protocExecutable + "'}";
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy