Ask Ben: Forward ColdFusion Requests To Another Domain

Posted August 21, 2007 at 6:02 PM

Tags: ColdFusion, Ask Ben

I just read your post on Exploring Path Info. I think it is pretty much exactly what I was looking for BUT I am not much of a programmer so I can't implement it. I have to change the domain of a site and need to 301 redirect all the URLs to the new domain... It is not a matter of just forwarding the old domain to the new - I need to forward literally every old URL to every new URL. This all has to do with maintain the search engine rankings in Google. All totaled there are well over 300,000 URLs that need to be redirected.

NOTE: Made one change to the code where query string was returning NULL.

There are a lot of variables that go into this. Things like URL Rewriting and 404 error handling can make this stuff much more complicated and need to be addressed individually. For this demo, I am going to assume that we are dealing with straight-up URLs and that no crazy URL magic is going on.

That being said, all we have to do for this is get the currently requested page URL and then swap out the old domain value for the new domain value using simple string manipulation. There are a few different ways to get the URL parts such as the ColdFusion CGI scope; however, I have been told that those are unreliable going from server to server, so for ease of use, I am going to use the URL String Buffer returned by the underlying ColdFusion Request object. Actually, this is doubly-nice since the StringBuffer already takes care of things like protocol, credentials, and port, so we don't even have to worry about HOW to string together all the URL parts.

Now, we just need a centralized place in the site so that all incoming ColdFusion URLs are redirected. NOTE that I say ColdFusion requests since Image and other File requests are not handled through the ColdFusion application server. I only know this aspect (the ColdFusion aspect). The Application.cfc or Application.cfm file serves as the easiest place to put this code:

 Launch code in new window » Download code as text file »

  • <!--- Get the request object for the current page. --->
  • <cfset objRequest = GetPageContext().GetRequest() />
  •  
  • <!---
  • Get the requested URL string buffer (this will NOT
  • contain the query string).
  • --->
  • <cfset objURLBuffer = objRequest.GetRequestURL() />
  •  
  • <!--- Get the query string for the current request. --->
  • <cfset strQueryString = objRequest.GetQueryString() />
  •  
  •  
  • <!---
  • Append the query string to the URL buffer if one exists
  • for the current request. If no query string was present,
  • the GetQueryString() method call will return NULL.
  • --->
  • <cfif StructKeyExists( VARIABLES, "strQueryString" )>
  •  
  • <cfset objURLBuffer.Append( "?" & strQueryString ) />
  •  
  • </cfif>
  •  
  •  
  • <!---
  • ASSERT: At this point, our URL string buffer holds the
  • full URL of the page that is currently being requested.
  • Now that we have that, we need to forward users to
  • another domain.
  • --->
  •  
  •  
  • <!--- Convert the URL buffer to a string. --->
  • <cfset strURL = objURLBuffer.ToString() />
  •  
  • <!---
  • Instead of just blindly replacing domain values, make
  • sure that it exists (if for some reason, something goes
  • wrong, you might create an infinite loop of forwarding
  • (if the domain never changes).
  •  
  • CAUTION: I am using a regular expression here in case
  • you want to hanlde thinks like www or what not. If you
  • don't care about that, just use FindNoCase() and remove
  • the \b constructs.
  • --->
  • <cfif REFindNoCase( "\bOLD_DOMAIN\b", strURL )>
  •  
  • <!---
  • We found the OLD domain in the current URL. Now,
  • let's replace it with the NEW domain.
  •  
  • CAUTION: I am using regular expression here in case
  • you want to handle things differently. If you don't
  • care about RegEx, just use a ReplaceNoCase() instead.
  • --->
  • <cfset strNewURL = strURL.ReplaceFirst(
  • "\bswoop\b",
  • "NEW_DOMAIN"
  • ) />
  •  
  • <!--- Return the permanent redirect to new URL. --->
  • <cfheader
  • statuscode="301"
  • statustext="Moved Permanently"
  • />
  •  
  • <!--- Return new URL. --->
  • <cfheader
  • name="Location"
  • value="#strNewURL#"
  • />
  •  
  •  
  • <cfelse>
  •  
  •  
  • <!---
  • Something went wrong with our code and we are not
  • finding the old domain. Return a 500 error.
  • --->
  • <cfheader
  • statuscode="500"
  • statustext="Internal Server Error"
  • />
  •  
  • <!--- Clear the content buffer and return HTML. --->
  • <cfcontent
  • type="text/html"
  • reset="true"
  • />
  •  
  • <!---
  • Even though something went wrong, provide the user
  • with a fail-safe so that they can at least get to
  • the new domain.
  • --->
  • <p>
  • There was an error with the current page request.
  • Please access our new site
  • <a href="http://NEW_DOMAIN_HOME_PAGE">here</a>.
  • </p>
  •  
  • </cfif>
  •  
  • <!--- Abort out of page. --->
  • <cfabort />

You will notice that in my redirect code, I am not just blindly swapping domain values and returning the headers; first I am checking to see if the domain is found. Once you get your code up and running, this shouldn't be a problem, but, at least for testing, this will help prevent and infinite redirect loop from occurring.

Also, just incase something does go wrong, I am returning a 500 error and giving the user a last-chance link to get to the new site. Hope this helps a bit.

Download Code Snippet ZIP File

Post Comment  |  Ask Ben  |  Permalink  |  Other Searches  |  Print Page





Reader Comments

Aug 22, 2007 at 7:59 AM // reply »
9 Comments

Alternative and in my opinion preferred way is to use mod_rewrite if using Apache. Couple fo lines in .htaccess will take care of whole thing (including images and other files)

RewriteEngine On

RewriteRule ^(.*)$ http://www.new.com/$1 [R=301,L]

IIS has something similar, I don't know exact solution for it.


Aug 22, 2007 at 9:02 AM // reply »
6,516 Comments

@Tero,

That looks pretty nice. Unfortunately, I don't have access to any rewriting filters where I host / test, so I can't go this route. In your example, maybe I am just not seeing it, but where are you matching the old domain name? It appears as if the current domain name isn't even part of the URL that is exposed to MOD Rewrite? Also, will $1 start with a forward slash?


Aug 22, 2007 at 9:50 AM // reply »
9 Comments

That would go to .htaccess in your webroot; only requests for that domain will be served there and thus matching is done earlier by Apache virtual hosts setup.

If your host allows you to use .htaccess and they have mod_rewrite enabled this would work for you - I would think that's quite common but it has been ages since I last used anything but dedicated servers one way or another.

I think you are right about forward slash, I just googled for example and copied what they had (http://www.tamingthebeast.net/articles3/changing-domain-names.htm)


Aug 23, 2007 at 1:13 PM // reply »
1 Comments

Ben, I'd be surprised if your current host does not allow .htaccess. I think that almost all hosts allow it. I know plenty of people who use it to help set up they blogs address string in wordpress. It has become a quite common feature with most webhosting.


Aug 23, 2007 at 1:21 PM // reply »
6,516 Comments

@Dave,

To be honest, I know nothing about htaccess files. In general, other than a 404 handler, I don't know much about anything that requires additional CFAdmin or IIS setup. I don't have access to that stuff, so I find ways of doing stuff without it.


Dec 11, 2007 at 8:13 AM // reply »
3 Comments

You're using a very commendable process to go about this. Scanning for the site first, etc. I can imagine the nightmare the original asker of the question must have gone through trying to do everything one by one like that. Surely there must be an easier way! You do have to assume, though, that no URL magic is going on - so, you still need to probably at least check one by one.


Post Comment  |  Ask Ben

Recent Blog Comments
Nov 21, 2009 at 1:13 PM
My First ColdFusion Builder Extension - Encrypting And Decrypting CFM / CFC Files
@Ben, Because I am pedantic, I just want to make sure that everyone knows there is absolutely no encryption going on. There is only encoding and obfuscation. The cfencode tool only obfuscates your C ... read »
Nov 21, 2009 at 12:28 PM
Using ColdFusion Structures To Remove Duplicate List Values
@Jody I can't seem to get your code sample to work. If you are still having problems, try this code out and see if it gets you what you wanted. <!--- Comma delimited list with various duplicates ... read »
Nov 21, 2009 at 11:03 AM
Groovy Operator Overloading Does Not Work In The ColdFusion Context
Hi Ben, Thanks for this informative post. Now I am reading ur old posts too ... read »
Nov 21, 2009 at 10:56 AM
HostMySite.com Has The Best ColdFusion Hosting
@Mehul, Yes very nice people, however several downtimes per day which was not acceptable. Hence we had to move out. I am glad you are having good luck with them so far. ... read »
Nov 20, 2009 at 11:32 PM
Five Months Without Hungarian Notation And I'm Loving It
I've used headless camel case for years for not only ColdFusion variables, but also SQL tables and fields... pretty much everything involving code. I also subscribe to the "don't abbreviate and clea ... read »
Nov 20, 2009 at 11:00 PM
Five Months Without Hungarian Notation And I'm Loving It
@Marcel, Yeah, I always err on the side of longer but more readable variable names. As for the camel casing of CF methods and the headless camel casing of custom items, I get around this by always ... read »
Nov 20, 2009 at 10:56 PM
Five Months Without Hungarian Notation And I'm Loving It
I use the following and love it: my.namespace.MyComponents.functionMethodsOrUDF() CONSTANT_VALUES_OR_PROPERTIES One thing I always try is to CamelCaseBuiltInColdFusionFunctions() so others can tell ... read »