Saturday, March 13, 2010

JAXON "Cloud" Framework

Today we find more developers are creating more and more applications that integrate with third-party services and sites.  As of right now, most of the available development frameworks and resources haven't yet caught up with this trend.  Over the past several months, I have been working closely with colleague under the umbrella of Workperch and have been trying to conceive, plan, execute and evaluate the results.

So far the products we have pushed out made use of third-party APIs as data and using Facebook connect as they provide us with get providing an easy signing up experience and data to play with in the product itself.    Simplifying the amount of time on engineering provides more time for product and experience development.  As the with the stronger engineering background I was stuck with writing the non-fun hard parts of the code.

Part 1 (Union API):
This past month I was becoming very frustrated having to deal with multiple APIs and the time spent required to integrating them in a useful way in the different products we were working on at the time.  For example, dealing with and keeping track of all of the different API credentials and access tokens.  Some of the sites and data providers that provide client libraries are hacked together and not very clean while others return JSON/XML and others only XML (really?).

In an effort to process and created a new service called UnionAPI.com. Union is just a proxy for other APIs, but rather than simply just brokering the API call, I wrote a standardized, simply, light weight client library for Python, PHP and Ruby.  That would be used to interact with the different APIs available on Union.

But, what makes Union useful is a few things:

1) When you log into your account on UnionAPI you enable the APIs you want to use, it bring up a form prompting for the necessity credentials and other configuration options per API enabled.

(Not mine design...)


When making a request to Facebook via Union it is able to inject the credentials into the request before passing it to Facebook, this means, no where in my code, do I need to store these keys.

2) "V-Store" is another tool that works quite well for any APIs that use OAuth 2.0 like Facebook or FourSquare.  When the third-party site sends you back an OAuth Code you can do something pretty tricky with it:
  
    $current_user_id = 102;
    $api = new unionapi("FourSquare");
    $api->store_token_under($current_user_id);
    $api->set("code", $fs_oauth_code);
    $api->set("service", "userToken");
    $api->call();

    $api = new unionapi("Facebook");
    $api->store_token_under($current_user_id);
    $api->set("code", $fb_oauth_code);
    $api->set("service", "userToken");
    $api->call();


See the "store_token_under" method?  This tells Union that it should assign / store the User_ID that I have passed in with the Access Token it is about to received back from the API.

There is no need to worry about saving these access tokens on my server allowing less database changes and code on my side.   UnionAPI will save them and inject the access token when forwarding the request to the API endpoint.

 Let's say I want to get the Check In's for this user "102" I can now just do:

  $api = new unionapi("FourSquare");
  $api->store_use(102);
  $api->set("service", "userCheckins");
  $fs_checkins = $api->call();

  $api = new unionapi("Facebook");
  $api->store_use(102);
  $api->set("service", "userCheckins");
  $fb_checkins = $api->call();

What happens here is that the UnionAPI service will look up the access token for this "102" and injects it into the request before it hits Facebook or FS.  Since, UnionAPI handled the OAuth transaction, I do not have to worry about dealing with the access tokens for that user for both FB and FS.  The UnionAPI service takes care of it all. 

Because UnionAPI is storing all these tokens for my users it can do some pretty nifty things:

  $api = new unionapi("UnionAPI-Social");
  $api->store_use(102);
  $api->set("service", "userCheckins");
  $all_checkins = $api->call();

This will return all checkins for user '102' on any site that Union has stored the Access Token for that user.  

Part 2 (JAXON Framework):

Working on these social related ventures I designed to extend an existing PHP framework that has been put together for doing develop of small scale application intended for easy deployment in any environment .

I decided to extend the framework and its URMin a way that I had a few levels to do something that I have never seen done with its database abstraction layer.

What if I could integrate Union with the database abstractly?

One very common task is pulling the City, State, Lat and Long for a given zip-code.  This is why I there is a  "zipinfo" API on Union, I can pass a zip-code to and it returns all this information associated with it.

I decided that I shouldn't have to make a "special" API call to get this information, when I query DB it should return this information as if it was saved in the database itself.

I ended up extending the database abstraction layer to allow the concept of virtual fields on tables in the database.  The goal here was to lessen the need to write code, so everything is defined from a web-based admin interface.

For example, let's say we have a simple table called "users" that contains 4 fields:


  1. user_id
  2. email_address
  3. password
  4. zip_code  

I am able to log into the admin interface and add a UnionAPI hook on the zipcode field in the users table. 

This is what happens before the hook is added and after:

No Hook:

Running the following SQL:

SELECT * FROM USERS WHERE USER_ID =  1;

Returns:

stdClass Object
(
    [user_id] => 1
    [email_address] => jason@mitchell-networks.com
    [password] => 6f1ed002ab5595859014ebf0951522d9
    [zip_code] => 30341
)



UnionAPI (zipinfo API) Hook added on the "zipcode" field:

SELECT * FROM USERS WHERE USER_ID =  1;
Returns:

stdClass Object
(
    [user_id] => 1
    [email_address] => jason@mitchell-networks.com
    [password] => 6f1ed002ab5595859014ebf0951522d9
    [zip_code] => stdClass Object
        (
            [zipcode] => 30341
            [city] => Atlanta
            [lat] => 33.9013
            [lng] => -84.3058
            [utc] => UTC-5
            [state] => Georgia
        )
)


This data is being pulled from Union in real-time when the query is executed and does not have to be stored in the users table itself.  Right now I made it have two options for caching.  It can be configured to actually have the fields tied to real-field and simply only used to populate the values the first time or it can serialize the values and store them in as JSON in predefined field in the table.

I will also be the first to say this isn't as efficient, but is was just a bit of a hack, but this framework is also designed for rapid prototyping right now and not designed to handle a site with 500 million users.  :-)

A few other examples of the "v-store" feature of UnionAPI are that it can automatically add social fields like "CheckIns" "Statuses" "Photos" to the table,  If the user has "connected" to both Flickr and Facebook the the Photos field will contain information about photos from both sites. 

This framework allows developers to use "cloud" services / APIs and have those services directly integrated into the very database models without the developer having to download any kind of third-party libraries or write one extra line of code.

I plan on releasing it to GitHub, but because the framework relies on my UnionAPI.com product, the framework isn't very useful until I open up UnionAPI to the public, which might be a while. 

Friday, June 9, 2006

What I learned - SeekSpots (2001)

 SeekSpots.com (Age 14 - 2001)



I was brought on to create a PPC search engine for The Spots Networks that would be called SeekSpots.  The Spots Network was a group of sites (shoppingspots.com, datingspots.com, lovespots.com, and others) that made money from referral programs.

SeekSpots - Technical Side:

Web Services - Using XML feeds of search results as back-fill... XML was all the rage.

Merchant Services - Setting up background processing of credit card transactions.

More Ops - I got to setup and maintain lots of Microsoft technology.


This is where SeekSpots.com was being hosted. My parents basement. 

SeekSpots - Business Side:

Connections
When searching for a back-fill provider, I found a search engine called SearchHippo.com which offered results via a free XML feed that I could use.
 
I thought it was quite cool and I e-mailed the creator to let him know that I planned to use SearchHippo to back-fill our own results.   This person turned out to be Kevin Marcus who co-founded InfoSpace.com and became a bit of a mentor and helped me connect to others.   We still keep in touch.

Friday, April 7, 2006

What I learned - My Online Quiz (2001)

My Online Quiz (Age 14 - 2001)


It was 2001 and the "it" thing to do was to create a quiz about yourself (or anything really) and send it to your friends to see how they would do. 

There was one big site doing it, that site was called Quiz Your Friends and after having countless friends send me their quizzes to take, it gave in went to make one myself make one myself.   But, you see Quiz Your Friends site was just terrible, I do mean terrible.  It was full of tacky ads and completely Flash based.   So, I found myself in a dilemma,  I did want to make a quiz, but just didn't want to use Quiz Your Friends.  At this point, I had already seen playing around creating dynamic web applications for a little over a year, and this quiz site seemed like it would be fairly basic application and figured I could just hack out my own version of it.

24 hours later...  I had written a basic application that let users sign up, create an unlimited number of 10 question quizzes, and each quiz had a high score.   I registered the domain MyOnlineQuiz.com and made the application live.   

At this point, I made the first quiz and sent it to 10 friends and waited..  Not only did they all take it, they went in and all make quizzes themselves and sent their quizzes to their friends and so on.  Within hours of making my first quiz, I had 50 users, by the next day it was 200 and the following day it was at 500 users.  This trend continued over the next several years and reached a peak of 2.3 million users at the time I shut it down in 2008.

What did I learn from My Online Quiz?

Technical Side:

Database Design
My database server was a 550MHz / 512MB box running SQL Server 2000 in my parents basement. The Database itself was over 30GB (storing nothing but text).   and I spent tons of time refining queries and storing data in temporary caches that I might need later for that user.

Hacker Proofing
Never half ass code and leave something exposed, hackers will find it and exploit it.   A few months after launch, I became a victim of an SQL Injection Attack that renamed all the quizzes to  "{{You've Been Cracked}}".  

Back Up!
I was extremely lucky that I had setup a daily backup from day one and was quickly able to revert back after the attack after fixing the vulnerability.   If I had not set that up, I would have lost the thousands of quizzes created and all the user accounts.


Business Side:

Setting Trends
When AIM started allowing Limited HTML web-sites to load in peoples profiles.  I realized that I could leverage it do allow people to add a quiz to their AIM profile and their friends could take it right from there.  What still might be the most flattering moment of my life, within a month QuizYourFriends had completely copied it and added the same capability to their application.

The Network Effect / Going Viral
The only goal with this project was just to make a quiz that didn't use QuizYourFriends,   Thinking about a growth model wasn't exactly on my young teenage mind.  However, seeing what was happening organically caused me to start actively thinking and factoring in growth strategies when designing new features and in feature projects.

Create Something You Need
This one goes without saying.  The best products are created by people who see a need and not trying to make money.  Might sure you're solving a problem and not just creating something cool.

User Experience Matters / Analytics Driven
From day one I found myself playing around with how things to functioned and the design.  An accidental blessing was that I had enabled basic logging setup from the start.  

At the time I was just interested in traffic numbers, but I quickly realized the power of analytics.  I discovered things like only X% of people who went to the landing page actually went to the sign up page and created an account (required to make a quiz).

I redesigned the flow to allow the user to start making a quiz from the landing page without having to sign up for an account up front, and I did still required them to do it, but moved it to the end of the flow.  The result was about a 100% increase to quizzes being created.


Allowing users to start making a quiz from the landing page (without an account upfront).


Social Networking
I like to joke that I created Facebook before Facebook (I didn't).   But I did discover early on that quizzes were being created in clusters in friend groups.  This was extremely noticeable with users using their .edu email addresses.   Once I saw one user from a university there were many more to quickly follow from the same university.

This made me think to add a basic social profile to user accounts, including a profile photo and let them tie it either a high-school or college and let other users from that school quickly see what other students had created quizzes also and it was a big hit.