A critical vulnerability was discovered in React Server Components (Next.js). Our systems remain protected but we advise to update packages to newest version. Learn More

Anders Hattestad
Dec 23, 2009
  10448
(0 votes)

QueryRewriteProvider replace query strings with paths

For fun I tried to make a url rewriter that could translate generic query parameters to be part of the path.

So if you had a page /newlist/  and used a query parameter to get to page 2 like this /newslist/?page=2 you would get that parameter as a path like /newslist/page_2/.

The code to replace all links is simple when you base your code on FriendlyUrlRewriteProvider.

From pageRef –> Url

#region from PageRef=>URL
protected override bool ConvertToExternalInternal(UrlBuilder url, object internalObject, System.Text.Encoding toEncoding)
{
    bool status=base.ConvertToExternalInternal(url, internalObject, toEncoding);
    if (status)
    {
        PageReference pageLink = internalObject as PageReference;
        if (pageLink != null)
        {
            List<string> done = new List<string>(); 
            foreach (string key in url.QueryCollection.AllKeys) 
            {
                if (!string.IsNullOrEmpty(key) && !string.IsNullOrEmpty(url.QueryCollection[key]))
                {
                    url.Path += key + "_" + url.QueryCollection[key] + "/";
                    done.Add(key);
                }
            }
            foreach (string d in done)
                url.QueryCollection.Remove(d);
           
        }
    }
    return status;
}
#endregion

This code will replace all links create on a page with the path of that page and the extra “path” from the query.

FromUrl=>Page

When  want to translate the extra path to a query parameter I need to do some small changes in ConvertToInternalInternal function. But the changes are unfortantly a bit inside the base function so I needed to copy the whole function. I also needed to make some adjustments to the helper function GetPageFromStartByPath and GetPageBySegments so I could find last know page from path and get the rest of the path so I could add them to the query collection.

protected static PageReference GetPageBySegments(PageReference rootPageRef, string[] segments, string languageCode, out PageReference lastGood, List<string> segmentsRemoved)
{
    lastGood = PageReference.EmptyReference;
    for (int i = 0; (i < segments.Length) && !PageReference.IsNullOrEmpty(rootPageRef); i++)
    {
        rootPageRef = UrlSegment.GetPageBySegment(rootPageRef, segments[i], languageCode, false);
        if (!PageReference.IsNullOrEmpty(rootPageRef))
            lastGood = rootPageRef;
        else
        {
            for (int x = i; x < segments.Length; x++)
                segmentsRemoved.Add(segments[x]);
        }
    }
    return rootPageRef;
}

Inside the ConvertToInternalIntranal I added this code

if (page == null)
{
    PageReference lastGood = PageReference.EmptyReference;
    List<string> segmentsRemoved = new List<string>();
    page = GetPageFromStartByPath(str, languageBranchAndPath,out lastGood, segmentsRemoved);

    if (page == null && !PageReference.IsNullOrEmpty(lastGood) && segmentsRemoved.Count>0)
    {
        page = DataFactory.Instance.GetPage(lastGood, LanguageSelector.Fallback(ContentLanguage.PreferredCulture.Name, true));
        foreach (string segment in segmentsRemoved)
        {

            string[] parts = segment.Split('_');
            queryCollection.Add(parts[0], HttpContext.Current.Server.UrlDecode( segment.Substring(parts[0].Length + 1)));
        }
    }
}

This code basically find the depths page based on the path, and take the rest of the path’s as query parameters. The code don’t use more cpu than the ordinary friendly rewriter.

 

The full code is here

Dec 23, 2009

Comments

Mårten Berg
Mårten Berg Sep 21, 2010 10:33 AM

Works great. Thanks for sharing.

Please login to comment.
Latest blogs
Beginner's Guide for Optimizely Backend Developers

Developing with Optimizely (formerly Episerver) requires more than just technical know‑how. It’s about respecting the editor’s perspective, ensurin...

MilosR | Dec 10, 2025

Optimizely PaaS Administrator Certification : Free for Everyone

Optimizely has recently launched a free PaaS Administrator Certification. https://academy.optimizely.com/student/activity/2958208-paas-cms-administ...

Madhu | Dec 9, 2025 |

Fixing TinyMCE Initialization Failures in Optimizely CMS: A Hidden Pipeline Issue with .NET SDK Versions

Over the past few weeks, several Optimizely CMS projects began experiencing a puzzling failure: XHtmlString fields stopped initializing TinyMCE in...

Francisco Quintanilla | Dec 9, 2025 |

Jhoose Security Modules v2.6.0 — Added support for Permissions Policy and .NET 10

Version 2.6.0 adds Permissions Policy header support, updates to .NET 10, improved policy management, configurable security settings, and enhanced...

Andrew Markham | Dec 6, 2025 |

Building a 360° Customer Profile With AI: How Opal + Optimizely Unlock Predictive Personalization

Creating truly relevant customer experiences requires more than collecting data—it requires understanding it. Most organizations already have rich...

Sujit Senapati | Dec 4, 2025

Building a Lightweight Optimizely SaaS CMS Solution with 11ty

Modern web development often requires striking a difficult balance between site performance and the flexibility needed by content editors. To addre...

Minesh Shah (Netcel) | Dec 3, 2025