.NET Core - веб-API - Как сделать загрузку файлов?

Я не могу понять, как написать .NET Core Web API для поддержки загрузки файлов. Обратите внимание: я не использую форму ASP.NET Core MVC для загрузки файлов, а через контейнер Servlet/JSP. Вот как определяется мой project.json,

{
  "dependencies": {
    "Microsoft.NETCore.App": {
      "version": "1.0.1",
      "type": "platform"
    },
    "Microsoft.AspNetCore.Mvc": "1.0.1",
    "Microsoft.AspNetCore.Routing": "1.0.1",
    "Microsoft.AspNetCore.Server.IISIntegration": "1.0.0",
    "Microsoft.AspNetCore.Server.Kestrel": "1.0.1",
    "Microsoft.Extensions.Configuration.EnvironmentVariables": "1.0.0",
    "Microsoft.Extensions.Configuration.FileExtensions": "1.0.0",
    "Microsoft.Extensions.Configuration.Json": "1.0.0",
    "Microsoft.Extensions.Logging": "1.0.0",
    "Microsoft.Extensions.Logging.Console": "1.0.0",
    "Microsoft.Extensions.Logging.Debug": "1.0.0",
    "Microsoft.Extensions.Options.ConfigurationExtensions": "1.0.0",
    "Npgsql": "3.1.9",
    "CoreCompat.Newtonsoft.Json": "9.0.2-beta001",
    "Newtonsoft.Json": "9.0.1"
  },

  "tools": {
    "Microsoft.AspNetCore.Server.IISIntegration.Tools": "1.0.0-preview2-final"
  },

  "frameworks": {
    "netcoreapp1.0": {
      "imports": [
        "dotnet5.6",
        "portable-net45+win8"
      ]
    }
  },

  "buildOptions": {
    "emitEntryPoint": true,
    "preserveCompilationContext": true
  },

  "runtimeOptions": {
    "configProperties": {
      "System.GC.Server": true
    }
  },

  "publishOptions": {
    "include": [
      "wwwroot",
      "**/*.cshtml",
      "appsettings.json",
      "web.config"
    ]
  },

  "scripts": {
    "postpublish": [ "dotnet publish-iis --publish-folder %publish:OutputPath% --framework %publish:FullTargetFramework%" ]
  }
}

Вот как определяется мой Startup,

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;

using QCService.Models;

namespace QCService
{
    public class Startup
    {
        public Startup(IHostingEnvironment env)
        {
            var builder = new ConfigurationBuilder()
                .SetBasePath(env.ContentRootPath)
                .AddJsonFile("appsettings.json", optional: true, reloadOnChange: true)
                .AddJsonFile($"appsettings.{env.EnvironmentName}.json", optional: true)
                .AddEnvironmentVariables();
            Configuration = builder.Build();
        }

        public IConfigurationRoot Configuration { get; }

        // This method gets called by the runtime. Use this method to add services to the container.
        public void ConfigureServices(IServiceCollection services)
        {
            // Add framework services.
            services.AddMvc();

            //Add SurveysRepository for Dependency Injection
            services.AddSingleton<ISurveysRepository, SurveysRepository>();
            //Add FeedbacksRepository for Dependency Injection
            services.AddSingleton<IFeedbacksRepository, FeedbacksRepository>();
            //Add AttachmentsRepository for Dependency Injection
            services.AddSingleton<IAttachmentsRepository, AttachmentsRepository>();
        }

        // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
        public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
        {
            loggerFactory.AddConsole(Configuration.GetSection("Logging"));
            loggerFactory.AddDebug();

            app.UseMvc();
        }
    }
}

Наконец, вот как определяется мой Контроллер,

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Mvc;
//Including model classes for Attachments
using QCService.Models;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Hosting;
using System.IO;
using Microsoft.Net.Http.Headers;

// For more information on enabling Web API for empty projects, visit http://go.microsoft.com/fwlink/?LinkID=397860

namespace QCService.Controllers
{
    [Route("api/[controller]")]
    public class AttachmentsController : Controller
    {
        public IAttachmentsRepository AttachmentItems { get; set; }
        public AttachmentsController(IAttachmentsRepository attachmentItems)
        {
            AttachmentItems = attachmentItems;
        }

        // GET: api/Attachments
        [HttpGet] /*http://localhost:52770/api/Attachments*/
        public IEnumerable<Attachments> Get()
        {
            return AttachmentItems.GetAllAttachments();
        }

        // GET api/Attachments/5
        [HttpGet("{id}")] /*http://localhost:52770/api/Attachments/{AttachmentID}*/
        public Attachments Get(int id)
        {
            return AttachmentItems.GetAttachment(id);
        }

        // GET api/Attachments/5
        [HttpGet("Feedback/{id}")] /*http://localhost:52770/api/Attachments/Feedback/{FeedbackID}*/
        public IEnumerable<Attachments> GetFeedbackAttachments(int id)
        {
            return AttachmentItems.GetFeedbackAttachments(id);
        }

        // POST api/Attachments
        [HttpPost]/*http://localhost:52770/api/Attachments/*/
        public async Task<IActionResult> PostFiles(ICollection<IFormFile> files)
        {
            try
            {
                System.Console.WriteLine("You received the call!");
                WriteLog("PostFiles call received!", true);
                //We would always copy the attachments to the folder specified above but for now dump it wherver....
                long size = files.Sum(f => f.Length);

                // full path to file in temp location
                var filePath = Path.GetTempFileName();
                var fileName = Path.GetTempFileName();

                foreach (var formFile in files)
                {
                    if (formFile.Length > 0)
                    {
                        using (var stream = new FileStream(filePath, FileMode.Create))
                        {
                            await formFile.CopyToAsync(stream);
                            //formFile.CopyToAsync(stream);
                        }
                    }
                }

                // process uploaded files
                // Don't rely on or trust the FileName property without validation.
                //Displaying File Name for verification purposes for now -Rohit

                return Ok(new { count = files.Count, fileName, size, filePath });
            }
            catch (Exception exp)
            {
                System.Console.WriteLine("Exception generated when uploading file - " + exp.Message);
                WriteLog("Exception generated when uploading file - " + exp.Message, true);
                string message = $"file / upload failed!";
                return Json(message);
            }
        }

        /// <summary>
        /// Writes a log entry to the local file system
        /// </summary>
        /// <param name="Message">Message to be written to the log file</param>
        /// <param name="InsertNewLine">Inserts a new line</param>
        public void WriteLog(string Message, bool InsertNewLine)
        {
            LogActivity ologObject = null;
            try
            {
                string MessageString = (InsertNewLine == true ? Environment.NewLine : "") + Message;
                if (ologObject == null)
                    ologObject = LogActivity.GetLogObject();
                ologObject.WriteLog(Message, InsertNewLine);
            }
            catch (Exception ex)
            {
                Console.WriteLine("Unable to write to the log file : " + ex.Message);
                Console.WriteLine("Stack Trace : " + ex.StackTrace);
            }
        }
    }
}

Я прошел через эту ссылку, https://docs.microsoft.com/en-us/aspnet/core/mvc/models/file-uploads но просто не может заставить его работать! Любая помощь приветствуется.

Я использую Google Advanced Rest Client для публикации данных следующим образом: Вот как я отправляю запрос POST

Я продолжаю получать ответ с сообщением... Положение дел: 500 - внутренняя ошибка сервера Время загрузки: 62 мс Заголовки ответов 5 Запросить заголовки 2 Перенаправление 0 Задержки Content-Type: multipart/form-data; граница = ---- WebKitFormBoundary0RKUhduCjSNZOEMN Контент-длина: 9106 Исходное сообщение

POST/api/Вложения HTTP/1.1 HOST: localhost: 52770 content-type: multipart/form-data; граница = ---- WebKitFormBoundary0RKUhduCjSNZOEMN длина контента: 9106

------ WebKitFormBoundary0RKUhduCjSNZOEMN Content-Disposition: form-data; name= "fileUpload1"; имя файла = "1i4ymeoyov_In_the_Wild_Testing.png" Content-Type: image/png

PNG

IHDR, я 3 (tEXtSoftwareAdobe ImageReadyq e < iTXtXML: com.adobe.xmp 8 ^ 2IDATx ] [email protected] H Pe P8 Ȉ b ̌3 p q * 7 " h + Yf O atD (< h|DLXdOH =} } ov8_U..... ------ WebKitFormBoundary0RKUhduCjSNZOEMN -

Ответ 1

Это основная причина: form-data; name= "fileUpload1"

Ваше имя "fileUpload1" должно быть "файлами", которые соответствуют объявлению файла PostFiles (ICollection <IFormFile> )

Ответ 2

В запрос ваш URL-адрес неверен. Проверьте маршрут на контроллере, [Маршрут ( "api/[controller]" )], а имя метода действия PostFiles. Итак, правильный URL-адрес будет http://localhost:52770/api/Attachments/PostFiles

Либо исправьте URL-адрес, либо переименуйте метод действия как Индекс

Ответ 3

Я думаю, что ваша проблема связана с ошибкой в ​​chrome 56.

ARC не может отправлять файлы в новую версию Chrome: 56.0.2924.76

Эта проблема затрагивает как почтальон, так и ChromeRestClient

Простым решением является опция Использовать XHR и удалить заголовок Content-Type: multipart/form-data.

Это работает как для почтальона, так и для хрома

Ответ 4

Здравствуйте! Измените тип содержимого "multipart/form-data" на "application/octet-stream". может перемещаться внутри блока кода без ошибки. попробуйте код, как показано ниже, отправьте файл в двоичном виде

'  byte[] fileToSend = File.ReadAllBytes(filePath);
                HttpWebRequest httpWebRequest = (HttpWebRequest)WebReq
uest.Create(url);
                httpWebRequest.Method = "POST";
                httpWebRequest.ContentType = "application/octet-stream";
                httpWebRequest.ContentLength = fileToSend.Length;              

                using (Stream requestStream = httpWebRequest.GetRequestStream())
                {
                    requestStream.Write(fileToSend, 0, fileToSend.Length);
                    requestStream.Close();
                }' 

Надежда будет работать.. Спасибо