3

我有一个创建 pdf 所需的服务类,它需要注入 ControllerContext 才能呈现html => pdf. 此服务通过中间层调用,该中间层不引用 web/mvc 项目。这很好,因为 ninject 完成了所有需要的服务注入等。这就是服务的样子(为了这些目的而简化)

public class PdfCreatorService : AbstractUrlBasedPdfCreatorService
{
    [Inject]
    public ControllerContext ControllerContext { get; set; }

    public override byte[] CreateReport(int reportId)
    {
        var result = new PdfController().CreateReport(reportId);
        using (var it = new ResponseCapture(ControllerContext.RequestContext))
        {
            result.ExecuteResult(ControllerContext);
            return it.ReadAllContents();
        }
    }
}

这是简化的调用堆栈:

Web.HomeController.SendEmailWithPdf(int id) 调用:
MiddleTier.BusinessLogic.SendEmailWithPdf(int id) 调用:
Web.Services.PdfCreatorService.CreateReport(int id)

Ninject 正在到达 PdfCreatorService,没问题,但是:我需要以某种方式将 ControllerContext 从 HomeController(通过中间层)转发到 PdfCreatorService。而中间层不能有任何对 ControllerContext 的引用。

我查看了提供者、工厂、解析器等,但找不到合适的解决方案。

任何帮助表示赞赏!干杯

4

2 回答 2

4

好的。我想出了一个我很满意的解决方案。
这是我检索 ControllerContext 的方式:

public class PdfCreatorService 
{
    [Inject]
    public ControllerContextProvider contextProvider { get; set; }
    [Inject]
    public PdfController pdfController { get; set; }

    public override byte[] CreateReport(int reportId)
    {
        var context = contextProvider.GetControllerContext();
        using (var stream = new ResponseCapture(context.RequestContext))
        {
            // Setup Controller
            var routeData = new RouteData(context.RouteData.Route, context.RouteData.RouteHandler);
            routeData.Values.Add("action", "CreateReport");
            routeData.Values.Add("controller", "Pdf");
            routeData.Values.Add("id", reportId);
            var pdfContext = new ControllerContext(context.HttpContext, routeData, pdfController);

            // Execute Controller
            var result = pdfController.CreateReport(reportId);
            result.ExecuteResult(pdfContext);

            return stream.ReadAllContents();
        }
    }
}

这是在原始控制器中设置上下文的方法:

public abstract class HomeController : Controller
{
    [Inject]
    public ControllerContextProvider ControllerContextProvider { get; set; }

    protected override IAsyncResult BeginExecute(RequestContext requestContext, AsyncCallback callback, object state)
    {
        ControllerContextProvider.GetControllerContext = () => ControllerContext;
        return base.BeginExecute(requestContext, callback, state);
    }
}

这是提供者类的样子:

public class ControllerContextProvider
{
    public Func<ControllerContext> GetControllerContext { get; set; }
}

这就是我绑定它的方式:

public class PortalNinjectModule : NinjectModule
{
    public override void Load()
    {
        Bind<ControllerContextProvider>().ToSelf().InRequestScope();
    }
}

仍然有兴趣看看是否有人有更优雅的解决方案。

于 2013-05-28T13:31:28.127 回答
2

您可以创建一个自定义控制器工厂。

public class MyControllerFactory : DefaultControllerFactory
{
    public override IController CreateController(RequestContext requestContext, string controllerName)
    {
        var controller = base.CreateController(requestContext, controllerName);
        HttpContext.Current.Request["controllerInstance"] = controller;
        return controller;
    }
}

您需要在 global.asax 中注册此控制器:

ControllerBuilder.Current.SetControllerFactory(typeof(MyControllerFactory));

之后,您可以配置 Ninject 以从请求中解析 ControllerContext:

kernel.Bind<ControllerContext>().ToMethod(ctx => ((Controller)HttpContext.Current.Request["controllerInstance"]).ControllerContext);
于 2013-05-28T11:51:45.507 回答