c# - Dotnet 核心 2.2 libgdipplus 异常

标签 c# docker .net-core openshift alpine-linux

我在 openshift 上运行 dotnet core 2.2 web api,我有一个返回 FileResult 的 api 方法。该文件是使用 epplus 生成的。但是,在容器化应用程序之后,端点返回以下错误。我相信如果使用 system.drawing 库会发生此异常,但我没有使用任何 system.drawing 功能。

System.TypeInitializationException: The type initializer for 'Gdip' threw an exception. ---> System.DllNotFoundException: Unable to load DLL 'libgdiplus': The specified module could not be found.

public async Task<FileResult> GetPSACostCaseAndFeedData([FromQuery] int caseId, [FromQuery] int proposalId, [FromQuery] int equipmentId)
        {
            try
            {
                var queryParams = new Dictionary<string, string> { { "caseId", caseId.ToString() }, { "proposalId", proposalId.ToString() }, { "equipmentId", equipmentId.ToString() } };
                var restObj = _restFactory.createRestRequest(Method.GET, "ProposalService/PSACostData/GetPSACostCaseAndFeedData", queryParams);
                Console.WriteLine("Base URL - "  + restObj.Item2.BaseUrl);
                var response = await restObj.Item2.ExecuteTaskAsync(restObj.Item1);
                if (response.StatusCode == HttpStatusCode.OK)
                {
                    var result = JsonConvert.DeserializeObject<PSACostCaseDetailsResponse>(response.Content);
                    Console.WriteLine("result -" + response.Content.ToString());
                    Console.WriteLine("start writing file");
                    var excelPackage = await _iPSACostExcelHelper.PopulatePsaCostFile(result, Path.Combine(_hostingEnvironment.ContentRootPath, "App_data", "PSACOST_V9.2.2.xlsm"));
                    Console.WriteLine("finished writing file");
                    return await Task.FromResult(File(_excelHelper.ConvertWorkBookToByteArray(excelPackage), ContentTypeEnums.excel, result.customerInformationModel.ProposalNum + "_" + result.caseDataModel.CaseNm + ".xlsm"));
                }
                throw new Exception("There is no PSA cost data for this case");
            }

“finished writing file”行确实打印到标准输出,之后的行失败了。有人可以帮我了解这里出了什么问题吗?为什么返回 FileResult 类型的方法需要 system.drawing?

下面还有我的 dockerfile。

FROM mcr.microsoft.com/dotnet/core/aspnet:2.2.7-alpine3.10
LABEL pipelineName="PSACOST.API" \
      pipelineKey="UMAOKJOH" \
      offeringKey="KKWEECBH"
RUN apk upgrade -U
EXPOSE 5000
ENV ASPNETCORE_URLS=http://*:5000
WORKDIR /app
COPY . /app
USER guest
ENTRYPOINT ["dotnet", "PSACOST.API.Web.dll"] 

最佳答案

我在使用生成二维码的库中的 System.Drawing 时遇到了类似的问题。

问题似乎是 DotNet Core docker 图像、sdk 或运行时没有 libgdiplus。 这对 Linux 来说不是问题,因为您可以在您的容器中执行类似 apt-get install -y libgdiplus 的操作,或者使用 DotNetCore 运行时镜像加上此 libgdiplus 构建您的应用程序 docker 镜像>.

这是一个恼人的问题。我不确定是否有任何解决方案不需要弄乱官方 docker 镜像,但到目前为止运气不好。

目前我可以看到一些解决方案选项。

选项 A: 如果您使用的是 Linux 的 docker DotNetCore 镜像,看起来,请使用 apt 安装缺少的运行时依赖项,例如 libgdiplus,因为它们不是 docker 镜像的一部分。

选项 B: 找到一个好的 docker DotNetCore 图像(sdk 或运行时),所有这些依赖项就位

选项 C: 找到一个不同于 epplus 的库,它不依赖于 System.Drawinglibgdiplus。但我什至不知道这是否可能


更新 1 对于选项 B,您可以尝试使用以下包含 aspnet core runtime + 所需依赖项的 docker 镜像: lonwern/aspnetcore-libgdiplus:2.1

有关如何在选择选项 A 后生成图像的更多信息,请在此处查看图像的创建方式: https://hub.docker.com/r/lonwern/aspnetcore-libgdiplus/dockerfile


更新 2 我创建了包含 DotNet Core SDK 和 AspNet Core 以及 libgdiplus 的公共(public) Docker 镜像,以便在运行使用 System.Drawing 的应用程序时可以使用它们SDK DotNet Core CLI(例如:dotnet vstest、dotnet test 等)或带有运行时。

https://hub.docker.com/repository/docker/sunnyatticsoftware/dotnet-core-sdk-libgdiplus

https://hub.docker.com/repository/docker/sunnyatticsoftware/dotnet-core-aspnet-libgdiplus

有关 repo 协议(protocol)本身的更多信息。

我在我的 CI/CD 管道和网络应用的最终 Docker 镜像中使用它们来避免这些运行时异常。

关于c# - Dotnet 核心 2.2 libgdipplus 异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60509086/

相关文章:

c# - 如何在内部访问修饰符上实现Setter依赖注入(inject)

c# - Windows-7 64 位中的 Oracle 错误

hadoop - 如何让我的 hdfs docker 客户端运行?

php - Docker - NGINX 容器转发到 PHP-FPM 容器

c# - 如何在Azure函数中绑定(bind)到MessageSender?

c# - .NET Web Api 2.1 中 Ninject 绑定(bind)的无参数构造函数错误

c# - 从 C# 保存对 Breeze Web api 的更改

linux - 为什么主机总是响应 `RST` 虽然服务器正在监听端口?

c# - 如何在asp.net core中编写自定义actionResult

c# - ASP.NET Core - 注册应用程序关闭不起作用