8

自己ホスト型の c# コンソール Web API があります。AngularJS を使用して非同期 http 要求を実行するいくつかの Web アプリケーションにサービスを提供します。CORS サポートと NTLM 認証の両方が必要です。

現在、両方を有効にしていますが、それらを実装した方法により、それらがキャンセルされてしまうようです。これは、サーバーからの最初の 401 応答によるものだと思います。通常、これにより 2 つの間で認証ハンドシェイクが行われますが、Fiddler で確認すると、この最初の 401 応答には Access-Control-Allow-Origin タグがないため、ブラウザがハンドシェイクを放棄します。CORS の必要性を無効にするためにアプリケーションに同じホスト名を付けたところ、ハンドシェイクは完全に機能しました。カスタム NtlmSelfHostConfiguration を元の HttpSelfHostConfiguration に置き換えることで NTLM 認証を無効にしました。Access-Control-Allow-Origin タグは完全に実行され、CORS が許可されます。機能しないのは、両方がアクティブな場合のみです。

更新: NtlmSelfHostConfiguration の使用中に Web サーバーが実際にそれを呼び出すことができることを確認するためだけに、それを必要としないリクエストで CorsHandle にブレークポイントを配置し、正常にヒットしました。NtlmSelfHostConfiguration に、Corshandle の呼び出しを物理的に妨げるものがないことを確認したかっただけです。認証に失敗したリクエストでも CorsHandle を呼び出す方法を見つける必要があるようです。

これが私の含意です:

プログラム.cs:

using System.Web.Http;
using System.Web.Http.Validation.Providers;
using System.Web.Http.SelfHost;
namespace DCMAPI
{
    class Program
    {
        static void Main(string[] args)
        {
            string BaseAddress = "http://localhost:8080/";

            //NtlmSelfHostConfiguration is defined in HttpSelfHostConfiguration.cs  
            //it inherits from HttpSelfHostConfiguration 
            //and enables Ntlm Authentication.

            var config = new NtlmSelfHostConfiguration(BaseAddress);

            config.Routes.MapHttpRoute(
                "API Default",
                "api/{controller}/{id}", 
                new { id = RouteParameter.Optional });

            //CorsHandler is also defined in CorsHandler.cs.  It is what enables CORS

            config.MessageHandlers.Add(new CorsHandler());

            var appXmlType =
            config.Formatters.XmlFormatter.SupportedMediaTypes.FirstOrDefault
                (t => t.MediaType == "application/xml");
            config.Formatters.XmlFormatter.SupportedMediaTypes.Remove(appXmlType);                   

            config.Services.RemoveAll
                (typeof(System.Web.Http.Validation.ModelValidatorProvider),
                v => v is InvalidModelValidatorProvider);

            using (HttpSelfHostServer server = new HttpSelfHostServer(config))
            {
                server.OpenAsync().Wait();
                Console.WriteLine("running");
                Console.ReadLine();
            }

        }
    }
}

NtlmSelfHostConfiguration.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Web.Http;
using System.Web.Http.SelfHost;
using System.Web.Http.SelfHost.Channels;
using System.ServiceModel;
using System.ServiceModel.Channels;

namespace DCMAPI
{
    public class NtlmSelfHostConfiguration : HttpSelfHostConfiguration
    {
        public NtlmSelfHostConfiguration(string baseAddress)
            : base(baseAddress)
        { }

        public NtlmSelfHostConfiguration(Uri baseAddress)
            : base(baseAddress)
        { }

        protected override BindingParameterCollection OnConfigureBinding
            (HttpBinding httpBinding)
        {
            httpBinding.Security.Mode =
                HttpBindingSecurityMode.TransportCredentialOnly;

            httpBinding.Security.Transport.ClientCredentialType =
                HttpClientCredentialType.Ntlm;

            return base.OnConfigureBinding(httpBinding);
        }
    }
}

CorsHandle.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Net.Http;
using System.Threading.Tasks;
using System.Threading;
using System.Net;

namespace DCMAPI
{
    public class CorsHandler : DelegatingHandler
    {
        const string Origin = "Origin";
        const string AccessControlRequestMethod = "Access-Control-Request-Method";
        const string AccessControlRequestHeaders = "Access-Control-Request-Headers";
        const string AccessControlAllowOrigin = "Access-Control-Allow-Origin";
        const string AccessControlAllowMethods = "Access-Control-Allow-Methods";
        const string AccessControlAllowHeaders = "Access-Control-Allow-Headers";

        protected override Task<HttpResponseMessage> SendAsync
            (HttpRequestMessage request, CancellationToken cancellationToken)
        {

            bool isCorsRequest = request.Headers.Contains(Origin);
            bool isPreflightRequest = request.Method == HttpMethod.Options;
            if (isCorsRequest)
            {

                if (isPreflightRequest)
                {
                    HttpResponseMessage response = 
                        new HttpResponseMessage(HttpStatusCode.OK);

                    response.Headers.Add(AccessControlAllowOrigin,
                        request.Headers.GetValues(Origin).First());

                    string accessControlRequestMethod =
                        request.Headers.GetValues(AccessControlRequestMethod)
                        .FirstOrDefault();
                    if (accessControlRequestMethod != null)
                    {
                        response.Headers.Add(
                            AccessControlAllowMethods, accessControlRequestMethod);
                    }

                    string requestedHeaders = string.Join(", ",
                        request.Headers.GetValues(AccessControlRequestHeaders));
                    if (!string.IsNullOrEmpty(requestedHeaders))
                    {
                        response.Headers.Add(AccessControlAllowHeaders,
                            requestedHeaders);
                    }

                    TaskCompletionSource<HttpResponseMessage> tcs = 
                        new TaskCompletionSource<HttpResponseMessage>();
                    tcs.SetResult(response);
                    return tcs.Task;
                }
                else
                {
                    return base.SendAsync(request,
                        cancellationToken).ContinueWith<HttpResponseMessage>(t =>
                    {
                        HttpResponseMessage resp = t.Result;
                        resp.Headers.Add(
                            AccessControlAllowOrigin,
                            request.Headers.GetValues(Origin).First());
                        return resp;
                    });
                }
            }
            else
            {
                return base.SendAsync(request, cancellationToken);
            }
        }
    }
}
4

1 に答える 1

10

これは少し遅れている可能性があり、それがあなたに役立つかどうかはわかりませんが、NTLM と CORS の両方を有効にしたいと考えている他の人に役立つかもしれません。

Web API 2バックエンド REST API に使用しています。そしてAngularJSフロントエンド用。

まず、NTLM を使用しているため、呼び出しでXMLHttpRequest資格情報を送信する必要があります。AngularJs では、次のようにします。

$http.get(url, { withCredentials: true });

次に、バックエンドで CORS を有効にする必要があります ( WebApiConfig.Register())。

        var enableCorsAttribute = new EnableCorsAttribute("*", "*", "*")
            {
                SupportsCredentials = true
            };
        config.EnableCors(enableCorsAttribute);

SupportsCredentials = trueAccess-Control-Allow-Credentials応答に追加することを意味します。

ワイルドカードは、次を許可することを意味します。

  • 任意のオリジンからの CORS
  • すべてのリクエスト ヘッダー
  • 任意の HTTP 動詞 (メソッド)
于 2014-03-25T08:26:11.213 に答える