Skip to main content

webservices - Oauth for Khan Academy API


About the project


I’m am building an online dashboard for my students that shows their up-to-the-minute Khan Academy progress. This will require access to the Khan Academy API. I’d like to build the whole thing in Mathematica.


Last school year I downloaded 13 unique spreadsheets and built some kludgy [Visual Basic][2] to bring it all together. This year, I'd like to make use of Khan's API and automate it all. Here’s an example report: Dropbox - khanReport.xlsx


What’s working:


I can make unauthenticated calls with success:


badges = Dataset[URLExecute["http://www.khanacademy.org/api/v1/badges”]]



I followed this Create A Random Nonce String Using JavaScript and made


choices = {“A”, “B”, “C”, “D”, "E", "F", "G", "H", "I", "J", "K", "L", 
"M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y",
"Z", "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l",
"m", "n", "o", "p", "q", "r", "t", "s", "u", "v", "w", "x", "y",
"z", 0, 1, 2, 3, 4, 5, 6, 7, 8, 9};

MakeNONCE[vals0_, bits0_] :=
Module[{vals = vals0, bits = bits0, result = ""},
For[i = 0, i < bits - 1, i++,

result = StringJoin[result, ToString[RandomChoice[vals]]]];
result
]

MakeNONCE[choices, 64]

I’ve written this, which I believe is formatted correctly but I still cannot generate a few values:


params = {"oauth_consumer_key" -> "7cPJxRmA5ybuaNQz", 
"oauth_nonce" -> MakeNONCE[choices, 64], "oauth_version" -> "1.0",
"oauth_signature" -> "abc",

"oauth_signature_method" -> "HMAC-SHA1",
"oauth_timestamp" -> UnixTime[]};
URLExecute["https://www.khanacademy.org/api/auth2/request_token", \
params]

What I still need help with:



  1. I cannot generate the signature correctly. I know that Hash[stuff,”SHA”] is close, but I’m uncertain what stuff needs to be included in the hash.

  2. Does my nonce code look right? Does it make sense? I’m certain there are built-in functions that would make this easy, but I couldn’t make them work.




Answer



Update: Full Flow


So after much wrangling here is the full OAuth 1.0 flow.


(* Get the token *)

$consumerKey = "key";
$consumerSecret = "secret";

$oauthTokenResponseString =
URLRead[

OAuthSigning`Private`HMACSha1SignatureService[
"http://www.khanacademy.org/api/auth2/request_token",
"HMAC", "GET",
$consumerKey,
$consumerSecret,
""(*token key*),
""(*token secret*)],
"Content"
]


$oauthTokenParams =
Association@URLQueryDecode@$oauthTokenResponseString;

(* Authorize it using the ones own account info (only works for the user who registered the keys) *)

URLRead[
HTTPRequest["https://www.khanacademy.org/api/auth2/authorize",
<|
"Method" -> "POST",
"Body" ->

{
"oauth_token" -> $oauthTokenParams["oauth_token"],
"identifier" -> $username,
"password" -> $password}
|>
],
"Content"
]

(* Obtain the request token proper *)


$accessTokenResponseString =
URLRead[
HTTPRequest[
OAuthSigning`Private`HMACSha1SignatureService[
"http://www.khanacademy.org/api/auth2/access_token",
"HMAC",
"GET",
$consumerKey,
$consumerSecret,

$oauthTokenParams["oauth_token"],
$oauthTokenParams["oauth_token_secret"]
]
],
"Content"
];

$acessTokenParams =
Association@URLQueryDecode@$accessTokenResponseString;


(* Make authenticated calls *)

Import[
HTTPRequest[
OAuthSigning`Private`HMACSha1SignatureService[
callEndpoint,
"HMAC", "GET",
$consumerKey, $consumerSecret,
$accessTokenParams["oauth_token"],
$accessTokenParams["oauth_token_secret"]

]
],
"RawJSON"
]



Update: now with docs*


Here are some docs for the ServiceConnections, OAuthClient, KeyClient, and OAuthSigning packages.


*Docs is a perhaps too strong a word for what they really are. Mostly they're stubs and things to try.





Mathematica has an oauth client but it's buried and undocumented. I tend to use it through the ServiceConnect framework but obviously my experience there will not help you as that's a) overkill for what you want to do and b) using oauth 2.


On the other hand there is oauth 1 support and some spelunking found me: OAuthSigning`Private`HMACSha1SignatureService (sorry about how mediocre that doc page is).


It takes an argument pattern like this per the docs:


OAuthSigning`Private`HMACSha1SignatureService[
url,
signatureMethod,
httpVerb,
consumerKey, consumerSecret,
tokenKey, tokenSecret
]


then subbing in some values from you + the Khan Academy API docs:


Needs["OAuthSigning`"]

OAuthSigning`Private`HMACSha1SignatureService[
"https://www.khanacademy.org/api/auth2/request_token",
"HMAC-SHA1",
"POST",
"7cPJxRmA5ybuaNQz",
"secret",

"t4632213944267176",
"4tsVQH6L5n2TGm8R"
]

And... it crashes my kernel without fail.


Digging in the actual implementation file we find we really want to use "HMAC" not "HMAC-SHA1" and with that knowledge:


OAuthSigning`Private`HMACSha1SignatureService[
"https://www.khanacademy.org/api/auth2/request_token",
"HMAC",
"POST",

"7cPJxRmA5ybuaNQz",
"secret",
"t4632213944267176",
"4tsVQH6L5n2TGm8R"
]

"https://www.khanacademy.org/api/auth2/request_token?oauth_consumer_\
key=7cPJxRmA5ybuaNQz&oauth_nonce=S994QECCq0o50QBDrj26&oauth_signature_\
method=HMAC-SHA1&oauth_timestamp=1500597534&oauth_token=\
t4632213944267176&oauth_version=1.0&oauth_signature=\

K4RxAhNuEHKn8aA3QMcnvTs1n9s%3D"

Voilà


There's your request URL.


The OAuthClient` package uses a special wrapper to hold the tokenKey and tokenSecret but it's probably easiest just to go straight to this unless you want to fill in all of the flow details.


Comments

Popular posts from this blog

front end - keyboard shortcut to invoke Insert new matrix

I frequently need to type in some matrices, and the menu command Insert > Table/Matrix > New... allows matrices with lines drawn between columns and rows, which is very helpful. I would like to make a keyboard shortcut for it, but cannot find the relevant frontend token command (4209405) for it. Since the FullForm[] and InputForm[] of matrices with lines drawn between rows and columns is the same as those without lines, it's hard to do this via 3rd party system-wide text expanders (e.g. autohotkey or atext on mac). How does one assign a keyboard shortcut for the menu item Insert > Table/Matrix > New... , preferably using only mathematica? Thanks! Answer In the MenuSetup.tr (for linux located in the $InstallationDirectory/SystemFiles/FrontEnd/TextResources/X/ directory), I changed the line MenuItem["&New...", "CreateGridBoxDialog"] to read MenuItem["&New...", "CreateGridBoxDialog", MenuKey["m", Modifiers-...

How to thread a list

I have data in format data = {{a1, a2}, {b1, b2}, {c1, c2}, {d1, d2}} Tableform: I want to thread it to : tdata = {{{a1, b1}, {a2, b2}}, {{a1, c1}, {a2, c2}}, {{a1, d1}, {a2, d2}}} Tableform: And I would like to do better then pseudofunction[n_] := Transpose[{data2[[1]], data2[[n]]}]; SetAttributes[pseudofunction, Listable]; Range[2, 4] // pseudofunction Here is my benchmark data, where data3 is normal sample of real data. data3 = Drop[ExcelWorkBook[[Column1 ;; Column4]], None, 1]; data2 = {a #, b #, c #, d #} & /@ Range[1, 10^5]; data = RandomReal[{0, 1}, {10^6, 4}]; Here is my benchmark code kptnw[list_] := Transpose[{Table[First@#, {Length@# - 1}], Rest@#}, {3, 1, 2}] &@list kptnw2[list_] := Transpose[{ConstantArray[First@#, Length@# - 1], Rest@#}, {3, 1, 2}] &@list OleksandrR[list_] := Flatten[Outer[List, List@First[list], Rest[list], 1], {{2}, {1, 4}}] paradox2[list_] := Partition[Riffle[list[[1]], #], 2] & /@ Drop[list, 1] RM[list_] := FoldList[Transpose[{First@li...

plotting - How to draw lines between specified dots on ListPlot?

I would like to create a plot where I have unconnected dots and some connected. So far, I have figured out how to draw the dots. My code is the following: ListPlot[{{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4,13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full] I have thought using ListLinePlot command, but I don't know how to specify to the command to draw only selected lines between the dots. Do have any suggestions/hints on how to do that? Thank you. Answer One possibility would be to use Epilog with Line : ListPlot[ {{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4, 13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full, Epilog -> { Line[ ...