Earlier this month a reader asked about AJAX-based validation of addresses. I was able to find a few services, but none that were free. Turned out that my own CFUPS package (riaforge link) actually supported address verification. Unfortunately it was only at the city level. I discovered that UPS does support street level verification as well, but had a heck of a time getting it to work. Luckily another reader (and talk about coincidence), Shane Pitts, figured out the issue and sent in the new code. His modification was released to RIAForge today. At lunch, while trying not to pay attention to the iPhone(y) press conference, I wrote up a quick demo.

Before getting into the AJAXy bits, here is a quick example of how the API works. Note that for development, only addresses in New York can be tested. I picked Epicenter's address, hope they don't mind.

<cfset results = av.streetAddressVerification(address="32 E 31st Street", city="New York", state="NY",postalcode="10016")> <cfdump var="#results#" label="Results">

This results in a one line response that is - for the most part - the exact same address.

In fact, for the most part I think you can assume if you get one response back, your initial response was ok. If you enter an inaccurate response though you get more responses:

<cfset results = av.streetAddressVerification(address="390902 SE 31st Street", city="New York", state="NY",postalcode="10016")> <cfdump var="#results#" label="Results">

So given a simple API where we assume 2-N results means we need to provide correction, here is a quick and ugly demo I created.

<html>

<head> <script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js"></script> <script>

function checkAddy(e) { console.log('running checkAddy'); var street = $.trim($("#street").val()); var city = $.trim($("#city").val()); var state = $.trim($("#state").val()); var zip = $.trim($("#zip").val()); if(street == '' || city == '' || state == '' || zip == '') return; $("#avResult").html("Checking your address for validity...");

$.getJSON("verificationservice.cfc?method=verifyaddress&returnFormat=json", {"street":street, "city":city, "state":state, "zip":zip}, function(res) { if(res.ERROR) { $("#avResult").html("An error occured trying to verify your address: "+res.ERROR); } else { //Ok, we must have RESULTS. If 1, we say its good. Technically it may 'correct' a bit but we don't care //If > 1, we create a list of results so you can pick one to auto correct if(res.RESULTS.DATA.length == 1) $("#avResult").html("Your address verified."); else { var s = "<p>The following addresses were returned as possible corrections. Click the address to correct your form.</p>"; $.each(res.RESULTS.DATA, function(idx,val) { s += "<div class='addyoption'><span class='street'>" + val[0] + "</span><br/> "; s += "<span class='city'>" + val[1] + "</span>, <span class='state'>" + val[2] + "</span><br>" ; s += "<span class='zip'>"+val[3]+"-"+val[4] +"</span></div>";

})

$("#avResult").html(s); } } }) }

function fixAddy(e) { div = $(e.currentTarget); var street = $(".street", div).html(); var city = $(".city", div).html(); var state = $(".state", div).html(); var zip = $(".zip", div).html();

$("#street").val(street); $("#city").val(city); $("#state").val(state); $("#zip").val(zip); $("#avResult").html(""); }

$(document).ready(function() {

$("#street, #city, #state, #zip").change(checkAddy); $("#saveBtn").click(checkAddy);

$(".addyoption").live("click", fixAddy); }) </script>

<style> .addyoption { width: 250px; padding: 5px; background-color: yellow; margin-bottom: 10px; } #avResult { } </style> </head>

<body>

<h2>Shipping Address</h2>

<form> Street: <input type="text" name="street" id="street" value="32 E 31st Street"><br/> City: <input type="text" name="city" id="city" value="New York"><br/> State: <input type="text" name="state" id="state" size="2" maxlength="2" value="NY"><br/> Zip: <input type="text" name="zip" id="zip" value="10016"><br/> <input type="button" id="saveBtn" value="Save"> </form>

<div id="avResult"></div>

</body> </html>

Quite a bit going on here. Let's begin at the bottom. I've got a simple form that asks for street, city, state, and zip. I've got an empty div I'll use for handling results.

Now move up to the document.ready block. I added an event listener to all my form fields. My thinking here was - I want to do address verification as soon as possible. You may enter your state first, your city next, and so on. So I've bound them all (including the button, which would normally be a submit) to a function called checkAddy. Let's go there next.

checkAddy begins by getting all the values. If any are blank, we leave. If not, we call our service. Our service is going to handle calling the UPS API and dealing with any error. As you can see, if we get an ERROR key back, we will report it. Otherwise we assume an array of RESULTS. If the length of the array is one, just assume we are good. (And again, I may be wrong on that.) If the length was more than one, here is where we get fancy. I create a div for each one and output them to screen. Like so:

Now for the cool part. Back in my document.ready block I had set up a listener for clicks on these results: $(".addyoption").live("click", fixAddy). This live call will constantly monitor the DOM so that when the new items are added, the event listener still works. Now you can click on the div and fixAddy will run. This will grab the values from the div and automatically update the form.

How about the service I called? Pretty trivial:

component {

remote function verifyAddress(string street, string city, string state, string zip) { var result = {}; try { var results = application.addressVerification.streetAddressVerification(address=arguments.street, city=arguments.city, state=arguments.state,postalcode=arguments.zip); result.results = results; } catch(any e) { result.error = e.message; }

return result; }

}

Man, I love writing CFCs in script. It's like moving from Michelob Ultra to a fine Stone IPA.

Because UPS tends to be a huge pain in the rear with the API at times, I cannot post a demo. I can however post this cool video of me clicking around the demo. It's better than unicorns farting.

Edited at 5:18PM: I tend to forget jQuery's $.each function. Rey Bango reminded me of this utility and I updated the code to make use of it.
Edited on Saturday: I made a few more improvements based on Bango's feedback. Thanks Rey!