OLD | NEW |
(Empty) | |
| 1 <h1 class="page_title">Tutorial: OAuth</h1> |
| 2 <p> |
| 3 <a href="http://oauth.net/">OAuth</a> is an open protocol that aims to standardi
ze the way desktop and web applications access a user's private data. OAuth prov
ides a mechanism for users to grant access to private data without sharing their
private credentials (username/password). Many sites have started enabling APIs
to use OAuth because of its security and standard set of libraries. |
| 4 </p> |
| 5 <p> |
| 6 This tutorial will walk you through the necessary steps for creating a Google Ch
rome Extension that uses OAuth to access an API. It leverages a library that you
can reuse in your extensions. |
| 7 </p> |
| 8 <p> |
| 9 This tutorial uses the <a href="http://code.google.com/apis/documents/">Google D
ocuments List Data API</a> as an example OAuth-enabled API endpoint. |
| 10 </p> |
| 11 <h2 id="requirements">Requirements</h2> |
| 12 <p> |
| 13 This tutorial expects that you have some experience writing extensions for Googl
e Chrome and some familiarity with the <a href="http://code.google.com/apis/acco
unts/docs/OAuth.html">3-legged OAuth</a> flow. Although you don’t need a backgro
und in the <a href="http://code.google.com/apis/documents/">Google Documents Lis
t Data API</a> (or the other <a href="http://code.google.com/apis/gdata/">Google
Data APIs</a> for that matter), having a understanding of the protocol may be h
elpful. |
| 14 </p> |
| 15 <h2 id="getting-started">Getting started</h2> |
| 16 <p> |
| 17 First, copy over the three library files from the Chromium source tree at <a hre
f="http://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/extensions/docs
/examples/extensions/oauth_contacts/">.../examples/extensions/oauth_contacts/</a
>: |
| 18 </p> |
| 19 <ul> |
| 20 <li><strong><a href="http://src.chromium.org/viewvc/chrome/trunk/src/chrome/comm
on/extensions/docs/examples/extensions/oauth_contacts/chrome_ex_oauth.html?revis
ion=34725&content-type=text/plain">chrome_ex_oauth.html</a></strong> - interstit
ial page for the oauth_callback URL</li> |
| 21 <li><strong><a href="http://src.chromium.org/viewvc/chrome/trunk/src/chrome/comm
on/extensions/docs/examples/extensions/oauth_contacts/chrome_ex_oauth.js?content
-type=text/plain">chrome_ex_oauth.js</a></strong> - core OAuth library</li> |
| 22 <li><strong><a href="http://src.chromium.org/viewvc/chrome/trunk/src/chrome/comm
on/extensions/docs/examples/extensions/oauth_contacts/chrome_ex_oauthsimple.js?c
ontent-type=text/plain">chrome_ex_oauthsimple.js</a></strong> - helpful wrapper
for chrome_ex_oauth.js</li> |
| 23 </ul> |
| 24 <p>Place the three library files in the root of your extension directory (or whe
rever your JavaScript is stored). Then include both .js files in your background
page in the following order:</p> |
| 25 <pre> |
| 26 <script type="text/javascript" src="chrome_ex_oauthsimple.js"></script&
gt; |
| 27 <script type="text/javascript" src="chrome_ex_oauth.js"></script> |
| 28 </pre> |
| 29 <p>Your background page will manage the OAuth flow.</p> |
| 30 <h2 id="oauth-dance">The OAuth dance in an extension</h2> |
| 31 <p> |
| 32 If you are familiar with the OAuth protocol, you'll recall that the OAuth dance
consists of three steps: |
| 33 </p> |
| 34 <ol> |
| 35 <li>fetching an initial request token</li> |
| 36 <li>having the user authorize the request token</li> |
| 37 <li>fetching an access token</li> |
| 38 </ol> |
| 39 <p>In the context of an extension, this flow gets a bit tricky. Namely, there is
no established consumer key/secret between the service provider and the applica
tion. That is, there is no web application URL for the user to be redirected to
after the approval process. |
| 40 </p> |
| 41 <p> |
| 42 Luckily, Google and a few other companies have been working on an <a href="http:
//code.google.com/apis/accounts/docs/OAuthForInstalledApps.html">OAuth for insta
lled applications</a> solution that you can use from an extension environment. I
n the installed applications OAuth dance, the consumer key/secret are ‘anonymous
’/’anonymous’ and you provide an <em>application name</em> for the user to grant
access to (instead of an application URL). The end result is the same: your bac
kground page requests the initial token, opens a new tab to the approval page, a
nd finally makes the asynchronous call for the access token. |
| 43 </p> |
| 44 <h3 id="set-code">Setup code</h3> |
| 45 <p>To initialize the library, create a <code>ChromeExOAuth</code> object in the
background page:</p> |
| 46 <pre> |
| 47 var oauth = ChromeExOAuth.initBackgroundPage({ |
| 48 'request_url': <OAuth request URL>, |
| 49 'authorize_url': <OAuth authorize URL>, |
| 50 'access_url': <OAuth access token URL>, |
| 51 'consumer_key': <OAuth consumer key>, |
| 52 'consumer_secret': <OAuth consumer secret>, |
| 53 'scope': <scope of data access, not used by all OAuth providers>, |
| 54 'app_name': <application name, not used by all OAuth providers> |
| 55 }); |
| 56 </pre> |
| 57 <p>In the case of the Documents List API and Google’s OAuth endpoints, a possibl
e initialization may be:</p> |
| 58 <pre> |
| 59 var oauth = ChromeExOAuth.initBackgroundPage({ |
| 60 'request_url': 'https://www.google.com/accounts/OAuthGetRequestToken', |
| 61 'authorize_url': 'https://www.google.com/accounts/OAuthAuthorizeToken', |
| 62 'access_url': 'https://www.google.com/accounts/OAuthGetAccessToken', |
| 63 'consumer_key': 'anonymous', |
| 64 'consumer_secret': 'anonymous', |
| 65 'scope': 'https://docs.google.com/feeds/', |
| 66 'app_name': 'My Google Docs Extension' |
| 67 }); |
| 68 </pre> |
| 69 <h3 id="request-token">Fetching and authorizing a request token</h3> |
| 70 <p> |
| 71 Once you have your background page set up, call the <code>authorize()</code> fun
ction to begin the OAuth dance and redirect the user to the OAuth provider. The
client library abstracts most of this process, so all you need to do is pass a c
allback to the <code>authorize()</code> function, and a new tab will open and re
direct the user. |
| 72 </p> |
| 73 <pre> |
| 74 oauth.authorize(function() { |
| 75 // ... Ready to fetch private data ... |
| 76 }); |
| 77 </pre> |
| 78 <p> |
| 79 You don't need to provide any additional logic for storing the token and secret,
as this library already stores these values in the browser’s <code>localStorage
</code>. If the library already has an access token stored for the current scope
, then no tab will be opened. In either case, the callback will be called. |
| 80 </p> |
| 81 <h3 id="signed-requests">Sending signed API requests</h3> |
| 82 <p> |
| 83 Once your specified callback is executed, call the <code>sendSignedRequest()</co
de> function to send signed requests to your API endpoint(s). <code>sendSignedRe
quest()</code> takes three arguments: a URI, a callback function, and an optiona
l parameter object. The callback is passed two arguments: the response text and
the <code>XMLHttpRequest</code> object that was used to make the request. |
| 84 </p> |
| 85 <p>This example sends an HTTP <code>GET</code>:</p> |
| 86 <pre> |
| 87 function callback(resp, xhr) { |
| 88 // ... Process text response ... |
| 89 }; |
| 90 function onAuthorized() { |
| 91 var url = 'https://docs.google.com/feeds/default/private/full'; |
| 92 var request = { |
| 93 'method': 'GET', |
| 94 'parameters': {'alt': 'json'} |
| 95 }; |
| 96 // Send: GET https://docs.google.com/feeds/default/private/full?alt=json |
| 97 oauth.sendSignedRequest(url, callback, request); |
| 98 }; |
| 99 oauth.authorize(onAuthorized); |
| 100 </pre> |
| 101 <p>A more complex example using an HTTP <code>POST</code> might look like this:<
/p> |
| 102 <pre> |
| 103 function onAuthorized() { |
| 104 var url = 'https://docs.google.com/feeds/default/private/full'; |
| 105 var request = { |
| 106 'method': 'POST', |
| 107 'headers': { |
| 108 'GData-Version': '3.0', |
| 109 'Content-Type': 'application/atom+xml' |
| 110 }, |
| 111 'parameters': { |
| 112 'alt': 'json' |
| 113 }, |
| 114 'body': 'Data to send' |
| 115 }; |
| 116 // Send: POST https://docs.google.com/feeds/default/private/full?alt=json |
| 117 oauth.sendSignedRequest(url, callback, request); |
| 118 }; |
| 119 </pre> |
| 120 <p> |
| 121 By default, the <code>sendSignedRequest()</code> function sends the <code>oauth_
*</code> parameters in the URL (by calling <code>oauth.signURL()</code>). If you
prefer to send the <code>oauth_*</code> parameters in the <code>Authorization</
code> header (or need direct access to the generated header), use <code>getAutho
rizationHeader()</code>. Its arguments are a URI, an HTTP method, and an optiona
l object of URL query parameters as key/value pairs. |
| 122 </p> |
| 123 <p>Here is the example above using <code>getAuthorizationHeader()</code> and an
<code>XMLHttpRequest</code> object:</p> |
| 124 <pre> |
| 125 function stringify(parameters) { |
| 126 var params = []; |
| 127 for(var p in parameters) { |
| 128 params.push(encodeURIComponent(p) + '=' + |
| 129 encodeURIComponent(parameters[p])); |
| 130 } |
| 131 return params.join('&'); |
| 132 }; |
| 133 function onAuthorized() { |
| 134 var method = 'POST'; |
| 135 var url = 'https://docs.google.com/feeds/default/private/full'; |
| 136 var params = {'alt': 'json'}; |
| 137 var xhr = new XMLHttpRequest(); |
| 138 xhr.onreadystatechange = function(data) { |
| 139 callback(xhr, data); |
| 140 }; |
| 141 xhr.setRequestHeader('GData-Version', '3.0'); |
| 142 xhr.setRequestHeader('Content-Type', 'application/atom+xml'); |
| 143 xhr.setRequestHeader('Authorization', oauth.getAuthorizationHeader(url, method
, params)); |
| 144 xhr.open(method, url + '?' + stringify(params), true); |
| 145 xhr.send('Data to send'); |
| 146 }; |
| 147 </pre> |
| 148 <h2 id="sample-code">Sample code</h2> |
| 149 <p> |
| 150 Sample extensions that use these techniques are available in the Chromium source
tree: |
| 151 </p> |
| 152 <ul> |
| 153 <li><a href="http://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/exten
sions/docs/examples/extensions/gdocs/">.../examples/extensions/gdocs/</a></li> |
| 154 <li><a href="http://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/exten
sions/docs/examples/extensions/oauth_contacts/">.../examples/extensions/oauth_co
ntacts/</a></li> |
| 155 </ul> |
OLD | NEW |