Создание канонических URL-адресов, включая идентификатор и заголовок - PullRequest
1 голос
/ 18 октября 2011

Я хочу повторить, что StackOverflow делает с его URL.

Например:

Скрытые возможности C #? - ( Скрытые функции C #? )

или

Скрытые возможности C #? - ( Скрытые функции C #? )

Приведет вас на ту же страницу, но когда они возвращаются в браузер, всегда возвращается первая.

Как реализовать изменение, чтобы возвращался больший URL?

1 Ответ

4 голосов
/ 18 октября 2011

Способ, который я обработал ранее, состоит в том, чтобы два маршрута были зарегистрированы в этом порядке

routes.MapRoute(
    null,
    "questions/{id}/{title}",
    new { controller = "Questions", action = "Index" },
    new { id = @"\d+", title = @"[\w\-]*" });

routes.MapRoute(
    null,
    "questions/{id}",
    new { controller = "Questions", action = "Index" },
    new { id = @"\d+" });

теперь в действии контроллера,

public class QuestionsController 
{
    private readonly IQuestionRepository _questionRepo;

    public QuestionsController(IQuestionRepository questionRepo)
    {
        _questionRepo = questionRepo;
    }

    public ActionResult Index(int id, string title)
    {
        var question = _questionRepo.Get(id);

        if (string.IsNullOrWhiteSpace(title) || title != question.Title.ToSlug())
        {
            return RedirectToAction("Index", new { id, title = question.Title.ToSlug() }).AsMovedPermanently();
        }

        return View(question);
    }
}

Мы будем постоянноперенаправить на URL, который содержит заголовок (строчный заголовок с дефисами в качестве разделителей), если у нас есть только идентификатор.Мы также удостоверимся, что переданный заголовок является правильным, проверив его по slugged-версии заголовка вопроса, создав тем самым канонический URL-адрес для вопроса, который содержит и идентификатор, и правильный заголовок slug.

Aпара помощников использовала

public static class PermanentRedirectionExtensions
{
    public static PermanentRedirectToRouteResult AsMovedPermanently
        (this RedirectToRouteResult redirection)
    {
        return new PermanentRedirectToRouteResult(redirection);
    }
}

public class PermanentRedirectToRouteResult : ActionResult
{
    public RedirectToRouteResult Redirection { get; private set; }
    public PermanentRedirectToRouteResult(RedirectToRouteResult redirection)
    {
        this.Redirection = redirection;
    }
    public override void ExecuteResult(ControllerContext context)
    {
        // After setting up a normal redirection, switch it to a 301
        Redirection.ExecuteResult(context);
        context.HttpContext.Response.StatusCode = 301;
        context.HttpContext.Response.Status = "301 Moved Permanently";
    }
}

public static class StringExtensions
{
    private static readonly Encoding Encoding = Encoding.GetEncoding("Cyrillic");

    public static string RemoveAccent(this string value)
    {
        byte[] bytes = Encoding.GetBytes(value);
        return Encoding.ASCII.GetString(bytes);
    }

    public static string ToSlug(this string value)
    {
        if (string.IsNullOrWhiteSpace(value))
        {
            return string.Empty;
        }

        var str = value.RemoveAccent().ToLowerInvariant();

        str = Regex.Replace(str, @"[^a-z0-9\s-]", "");

        str = Regex.Replace(str, @"\s+", " ").Trim();

        str = str.Substring(0, str.Length <= 200 ? str.Length : 200).Trim();

        str = Regex.Replace(str, @"\s", "-");

        str = Regex.Replace(str, @"-+", "-");

        return str;
    }
}
...