1

Web APIで基本Http認証を実装する属性をリファクタリングして、次のようにDIを設定しました。

 public class BasicHttpAuthAttribute : ActionFilterAttribute
    {
        private readonly ILoginManager _manager;

        public BasicHttpAuthAttribute(ILoginManager manager)
        {
            this._manager = manager;
        }
        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            if (actionContext.Request.Headers.Authorization == null)
            {
                actionContext.Response = actionContext.Request.CreateResponse(HttpStatusCode.Unauthorized);
                actionContext.Response.Content = new StringContent("Missing Auth-Token");
            }
            else
            {

                var authToken = actionContext.Request.Headers.Authorization.Parameter;
                var decodedToken = Encoding.UTF8.GetString(Convert.FromBase64String(authToken));

                string userName = decodedToken.Substring(0, decodedToken.IndexOf(":"));
                string password = decodedToken.Substring(decodedToken.IndexOf(":") + 1);


                UserInfo user;


                if (_manager.LoginPasswordMatch(userName, password, out user))
                {
                    var apiUser = new ApiUser(user.UserID);
                    HttpContext.Current.User = new GenericPrincipal(new ApiIdentity(apiUser), new string[]{});

                    base.OnActionExecuting(actionContext);

                }
                else
                {
                    actionContext.Response = actionContext.Request.CreateResponse(HttpStatusCode.Unauthorized);
                    actionContext.Response.Content = new StringContent("Invalid username or password");
                }



            }
        }
    }

リファクタリングの前は、OnActionExecuting内にLoginManagerのインスタンス(それ自体にはDIがないため、ctorを使用してインスタンスを作成できました)を作成していました。リファクタリング後の問題は、WebApiメソッドにフィルターを適用すると、そこで引数が必要になるため、ビルドが失敗することです。

この場合、LoginManager自体がコンストラクターでILoginRepositoryを取得するため、DIを実装するにはどうすればよいですか?それも可能ですか?

4

3 に答える 3

2

Web API での属性の処理方法は、予想とは少し異なるため、これを行うのは簡単ではありません。まず第一に、それらはさまざまな時点で作成され、第二にキャッシュされます。

そうは言っても、コンストラクターを介した注入ではなく、DI を実現する最も簡単な方法は、処理時に選択した DI フレームワークを呼び出し、依存関係を取得することonActionExecutingです。

var s = GlobalConfiguration.Configuration.DependencyResolver.GetService(typeof(IService));
于 2012-09-26T13:08:26.133 に答える
1

最も簡単なオプションは、コンストラクターがパラメーターを必要としないようにすることですが、代わりに別の形式の注入を使用します。正確な方法は、使用している DI コンテナーによって異なります。たとえば、Unity では、デフォルトのコンストラクターを作成するだけです。別のパブリック メソッドを作成し (私は通常、わかりやすくするために Initialize と呼んでいます)、[InjectionMethod] 属性で装飾します。次に、コンストラクター内で、container.BuildUp(this); を呼び出すだけです。これにより、基本的に MVC は必要に応じて既定のコンストラクターを呼び出すことができますが、InjectionMethod は常に構築プロセスの一部として自動的に呼び出されます。

他の DI コンテナーで同じことを行う同様の方法がありますが、何を使用しているかを知らなくても、Unit の例を説明するのが最も簡単です。

于 2012-09-26T13:01:27.603 に答える