2016-09-08 87 views
6

我们正在构建一个包含Angular2前端,ASP.NET Core web api公共后端和ASP.NET Core web api私人后台。将ASP.NET Core web API中的文件发布到另一个ASP.NET Core web api

将文件从Angular2上传到公共后台工作。但是我们宁愿将它们发布到私有后端。

当前工作的代码

[HttpPost] 
public StatusCodeResult Post(IFormFile file) 
{ 
    ... 
} 

从那里我可以使用file.CopyTo(FILESTREAM)将文件保存到磁盘;

但是,我想重新发送该文件,或这些文件,或者,理想情况下,整个请求到我的第二个web api核心。

我不知道如何用HttpClient类的asp.net核心实现这一点。

我已经试过各种东西,如

StreamContent ss = new StreamContent(HttpContext.Request.Body); 
var result = client.PostAsync("api/Values", ss).Result; 

但我的第二后端得到一个空IFormFile。

我有一种感觉是可以将文件作为流发送并在另一侧重建它们,但无法使其工作。

该解决方案必须使用两个web api核心。

回答

7

解决方案

在DMZ公共后端

[HttpPost] 
public StatusCodeResult Post(IFormFile file) 
{ 
    try 
    { 
     if (file != null && file.Length > 0) 
     { 
      using (var client = new HttpClient()) 
      { 
       try 
       { 
        client.BaseAddress = new Uri(currentPrivateBackendAddress); 

        byte[] data; 
        using (var br = new BinaryReader(file.OpenReadStream())) 
         data = br.ReadBytes((int)file.OpenReadStream().Length); 

        ByteArrayContent bytes = new ByteArrayContent(data); 


        MultipartFormDataContent multiContent = new MultipartFormDataContent(); 

        multiContent.Add(bytes, "file", file.FileName); 

        var result = client.PostAsync("api/Values", multiContent).Result; 


        return StatusCode((int)result.StatusCode); //201 Created the request has been fulfilled, resulting in the creation of a new resource. 

       } 
       catch (Exception) 
       { 
        return StatusCode(500); // 500 is generic server error 
       } 
      } 
     } 

     return StatusCode(400); // 400 is bad request 

    } 
    catch (Exception) 
    { 
     return StatusCode(500); // 500 is generic server error 
    } 
} 

私人后端

[HttpPost] 
public void Post() 
{ 
    //Stream bodyStream = HttpContext.Request.Body; 

    if (Request.HasFormContentType) 
    { 
     var form = Request.Form; 
     foreach (var formFile in form.Files) 
     { 
      var targetDirectory = Path.Combine(_appEnvironment.WebRootPath, "uploads"); 

      var fileName = GetFileName(formFile); 

      var savePath = Path.Combine(targetDirectory, fileName); 

      using (var fileStream = new FileStream(savePath, FileMode.Create)) 
      { 
       formFile.CopyTo(fileStream); 
      }     
     } 
    }   
} 
0

当您调用私有后端API时忽略HttpClient,您是否可以从公共Core API项目引用私有Core API项目并直接从Core API项目调用控制器?看到请求仍然为空/空。如果请求带有值,那么问题是使用HttpClient。

理想情况下,您希望为要分发给使用客户端的私有Core API创建一个包库(一种SDK)。这就像一个包装/代理。通过这种方式,您可以隔离私有后端系统,并且可以单独对其进行故障排除。所以你公开的Core API项目(这是私人后端客户端)可以引用它作为nuget包。

1

嗨,我有同样的问题,这是对我工作:

我的设置是netCore MVC netCoreApi。

我的MVC控制器看起来像:

[HttpPost("UploadFiles")] 
public async Task<IActionResult> Post(List<IFormFile> files) 
{ 
    Sp4RestClient dataPovider = new Sp4RestClient("http://localhost:60077/"); 

    long size = files.Sum(f => f.Length); 

    foreach (var file in files) 
    { 
     await dataPovider.ImportFile(file); 
    } 

    return Ok(); 
} 

DataProvider的方法:

public async Task ImportFile(IFormFile file) 
    { 
     RestClient restClient = new RestClient(_queryBulder.BuildImportFileRequest()); 

     using (var content = new MultipartFormDataContent()) 
     { 
      content.Add(new StreamContent(file.OpenReadStream()) 
      { 
       Headers = 
       { 
        ContentLength = file.Length, 
        ContentType = new MediaTypeHeaderValue(file.ContentType) 
       } 
      }, "File", "FileImport"); 

      var response = await restClient.Post<IFormFile>(content); 
     } 
    } 

而且至少我的WebAPI控制器:

[HttpPost] 
[Route("ImportData")] 
public IActionResult Import(IFormFile file) 
{   
    return Ok(); 
} 

在这里看到完整的代码是我RESTClient实现邮政方法:

public async Task<RestResult<T>> Post<T>(HttpContent content) 
    { 
     using (HttpClient httpClient = new HttpClient()) 
     { 
      HttpResponseMessage response = await httpClient.PostAsync(Endpoint, content); 
      if (response.StatusCode == HttpStatusCode.Created) 
      { 
       T result = JsonConvert.DeserializeObject<T>(await response.Content.ReadAsStringAsync()); 
       return new RestResult<T> { Result = result, ResultCode = HttpStatusCode.OK }; 
      } 
      RestResult<T> nonOkResult = 
       new RestResult<T> { Result = default(T), ResultCode = response.StatusCode, Message = await response.Content.ReadAsStringAsync() }; 
      return nonOkResult; 
     } 
    } 

//啊,我知道我不是让HttpStatusCode.Created回;)

编码快乐;)

相关问题