私は、Web APIを使用してASP.NET MVC内でAngularJSプロジェクトに取り組んできました。 angularルーティングされたURLに直接移動するか、ページを更新しようとする場合を除き、非常にうまく機能します。サーバー設定を使用するよりも、これで対処できると思いましたMVCのルーティングエンジン。
現在のWebAPIConfig:
public static class WebApiConfig
{
public static void Register(HttpConfiguration config)
{
// Web API routes
config.MapHttpAttributeRoutes();
config.Routes.MapHttpRoute(
name: "DefaultApi",
routeTemplate: "api/{controller}/{id}",
defaults: new { id = RouteParameter.Optional },
constraints: new { id = @"^[0-9]+$" }
);
config.Routes.MapHttpRoute(
name: "ApiWithActionAndName",
routeTemplate: "api/{controller}/{action}/{name}",
defaults: null,
constraints: new { name = @"^[a-z]+$" }
);
config.Routes.MapHttpRoute(
name: "ApiWithAction",
routeTemplate: "api/{controller}/{action}",
defaults: new { action = "Get" }
);
}
}
現在のRouteConfig:
public class RouteConfig
{
public static void RegisterRoutes(RouteCollection routes)
{
routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
routes.IgnoreRoute(""); //Allow index.html to load
routes.IgnoreRoute("partials/*");
routes.IgnoreRoute("assets/*");
}
}
現在のGlobal.asax.cs:
public class WebApiApplication : HttpApplication
{
protected void Application_Start()
{
AreaRegistration.RegisterAllAreas();
GlobalConfiguration.Configure(WebApiConfig.Register);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
var formatters = GlobalConfiguration.Configuration.Formatters;
formatters.Remove(formatters.XmlFormatter);
GlobalConfiguration.Configuration.Formatters.JsonFormatter.SerializerSettings = new JsonSerializerSettings
{
Formatting = Formatting.Indented,
PreserveReferencesHandling = PreserveReferencesHandling.None,
ReferenceLoopHandling = ReferenceLoopHandling.Ignore,
};
}
}
目標:
/ api/*は引き続きWebAPI、/ partials /、および/ assets /に移動し、すべてファイルシステムに移動します。他のものはすべて/ indexにルーティングされます.html、これはmy Angularシングルページアプリです。
-編集-
私はそれが機能しているようです。これをBOTTOM OFRouteConfig.csに追加しました:
routes.MapPageRoute("Default", "{*anything}", "~/index.html");
そして、ルートweb.configへのこの変更:
<system.web>
...
<compilation debug="true" targetFramework="4.5.1">
<buildProviders>
...
<add extension=".html" type="System.Web.Compilation.PageBuildProvider" /> <!-- Allows for routing everything to ~/index.html -->
...
</buildProviders>
</compilation>
...
</system.web>
ただし、ハックのような匂いがします。これを行うためのより良い方法はありますか?
ワイルドカードセグメントを使用します。
routes.MapRoute(
name: "Default",
url: "{*anything}",
defaults: new { controller = "Home", action = "Index" }
);
よりネイティブなアプローチを提案する
<system.webServer>
<httpErrors errorMode="Custom">
<remove statusCode="404" subStatusCode="-1"/>
<error statusCode="404" prefixLanguageFilePath="" path="/index.cshtml" responseMode="ExecuteURL"/>
</httpErrors>
</system.webServer>
私の場合、これらのアプローチはどれもうまくいきませんでした。私は2エラーメッセージ地獄で立ち往生しました。 このタイプのページは提供されませんまたは何らかの404.
uRLの書き換えが機能しました:
<system.webServer>
<rewrite>
<rules>
<rule name="AngularJS" stopProcessing="true">
<match url="[a-zA-Z]*" />
<conditions logicalGrouping="MatchAll">
<add input="{REQUEST_FILENAME}" matchType="IsFile" negate="true" />
<add input="{REQUEST_FILENAME}" matchType="IsDirectory" negate="true" />
<add input="{REQUEST_URI}" pattern="^/(api)" negate="true" />
</conditions>
<action type="Rewrite" url="/" />
</rule>
</rules>
</rewrite>
...
.js .mapなどのURLを書き換えたくないので、[a-zA-Z]で一致したことに注意してください。
これはVSですぐに機能しましたが、IISでurl-rewriteモジュールをインストールする必要があるかもしれません https://www.iis.net/downloads/Microsoft/url -rewrite
上位のいくつかの回答と同様のアプローチがありましたが、欠点は、誰かがAPI呼び出しを誤って行った場合、より有用なものの代わりにそのインデックスページを戻すことになります。
そこで、/ apiで始まっていないリクエストに対してインデックスページが返されるように更新しました。
//Web Api
GlobalConfiguration.Configure(config =>
{
config.MapHttpAttributeRoutes();
});
//MVC
RouteTable.Routes.Ignore("api/{*anything}");
RouteTable.Routes.MapPageRoute("AnythingNonApi", "{*url}", "~/wwwroot/index.html");
さて、私は_Global.asax.cs
_のRouteConfig.RegisterRoutes(RouteTable.Routes);
呼び出しを削除しましたが、リソースが存在する場合は、入力したURLが提供されるようになりました。 APIヘルプページでも機能します。