Obtaining a Facebook auth token for a command-line (desktop) application Obtaining a Facebook auth token for a command-line (desktop) application ruby ruby

Obtaining a Facebook auth token for a command-line (desktop) application


So, I finally got it working, without setting up a web server and doing a callback. The trick, counter-intuitively, was to turn off the "Desktop application" setting and not to request offline_access.

FaceBook::Graph's support for posting videos doesn't seem to work at the moment, so I ended up doing it in Ruby.

fb_auth = FbGraph::Auth.new(APP_ID, APP_SECRET)client = fb_auth.clientclient.redirect_uri = "https://www.facebook.com/connect/login_success.html"if ARGV.length == 0  puts "Go to this URL"  puts client.authorization_uri(:scope => [:publish_stream, :read_stream] )  puts "Then run me again with the code"  exitendif ARGV.length == 1  client.authorization_code = ARGV[0]  access_token = client.access_token! :client_auth_body  File.open("authtoken.txt", "w") { |io| io.write(access_token)  }  exitendfile, title, description = ARGVaccess_token = File.read("authtoken.txt")fb_auth.exchange_token! access_tokenFile.open("authtoken.txt", "w") { |io| io.write(fb_auth.access_token)  }me = FbGraph::Page.new(PAGE_ID, :access_token => access_token)me.video!(    :source => File.new(file),    :title => title,    :description => description)


Problem is in your case that for OAuth you'll need some endpoint URL which is publicly reachable over the Internet for Facebook servers, which can be a no-go for normal client PCs, or a desktop application which is capable of WebViews (and I assume, command line isn't).

Facebook states at https://developers.facebook.com/docs/facebook-login/manually-build-a-login-flow#login that you can build a desktop client login flow, but only via so-called WebViews. Therefore, you'd need to call the OAuth endpoint like this:

https://www.facebook.com/dialog/oauth?client_id={YOUR_APP_ID}&redirect_uri=https://www.facebook.com/connect/login_success.html&response_type=token&scope={YOUR_PERMISSION_LIST}

You then have to inspect the resulting redirected WebView URL as quoted:

When using a desktop app and logging in, Facebook redirects people to the redirect_uri mentioned above and places an access token along with some other metadata (such as token expiry time) in the URI fragment:

https://www.facebook.com/connect/login_success.html#access_token=ACCESS_TOKEN... 

Your app needs to detect this redirect and then read the access token out of the URI using the mechanisms provided by the OS and development framework you are using.

If you want to do this in "hacking mode", I'd recommend to do the following.:

  • As you want to post to a Page, get a Page Access Token and store it locally. This can be done by using the Graph Explorer at the

https://developers.facebook.com/tools/explorer?method=GET&path=me%2Faccounts

endpoint. Remember to give "manage_pages" and "publish_actions" permissions.

curl -v -0 --form title={YOUR_TITLE} --form description={YOUR_DESCRIPTION} --form source=@{YOUR_FULL_FILE_PATH} https://graph-video.facebook.com/{YOUR_PAGE_ID}/videos?access_token={YOUR_ACCESS_TOKEN}

References:

https://developers.facebook.com/docs/graph-api/reference/page/videos/#publishhttps://developers.facebook.com/docs/reference/api/video/


From the facebook video API reference:

An individual Video in the Graph API.

To read a Video, issue an HTTP GET request to /VIDEO_ID with the user_videos permission. This will return videos that the user has uploaded or has been tagged in.

Video POST requests should use graph-video.facebook.com.

So you should be posting to graph-video.facebook.com if you are to upload video.

You also need extended permissions from the user or profile you'll be uploading to, in this case you need video_upload this is going to be requested once only, when the user currently logged in is asked for such permission for the app.

And your endpoint should be:

https://graph-video.facebook.com/me/videos

If you always want to post to a specific user than you'll have to change the endpoint part from /me to the User ID or page ID.

Here's a sample (in PHP):

$app_id = "YOUR_APP_ID";$app_secret = "YOUR_APP_SECRET"; $my_url = "YOUR_POST_LOGIN_URL"; $video_title = "YOUR_VIDEO_TITLE";$video_desc = "YOUR_VIDEO_DESCRIPTION";$code = $_REQUEST["code"];if(empty($code)) {   $dialog_url = "http://www.facebook.com/dialog/oauth?client_id="      . $app_id . "&redirect_uri=" . urlencode($my_url)      . "&scope=publish_stream";    echo("<script>top.location.href='" . $dialog_url . "'</script>");}$token_url = "https://graph.facebook.com/oauth/access_token?client_id="    . $app_id . "&redirect_uri=" . urlencode($my_url)     . "&client_secret=" . $app_secret     . "&code=" . $code;$access_token = file_get_contents($token_url);$post_url = "https://graph-video.facebook.com/me/videos?"    . "title=" . $video_title. "&description=" . $video_desc     . "&". $access_token;echo '<form enctype="multipart/form-data" action=" '.$post_url.' "       method="POST">';echo 'Please choose a file:';echo '<input name="file" type="file">';echo '<input type="submit" value="Upload" />';echo '</form>';

Although I'm concerned about the upload speed if the videos are too big, but I'm guessing your customer has already sorted that out (compress/optimize/short videos etc.)

I've made you a demo here. Go to my website (I own that domain) and try to upload a video. I tried with this one which is a relatively small 4Mb file. Be sure that this script will only try to upload a video, nothing more (to the FB profile you are currently logged in, that is) but, if you are still concerned, copy my snippet, upload it to your own server (with PHP support of course) and create a test app where the site url is that domain and be sure to specify in the $my_url variable your endpoint which is basically the full path to your script receiving responses from facebook:

http://yourdomain.com/testfb.php 

If you still want to do it on a desktop app then you have to go to developer.facebook.com on your app settings:

Settings > Advanced

And look for the first option:

enter image description here

And enable that switch so that facebook allows you to POST from a desktop or native app instead of a web server.

Note: I'm not an expert on Ruby, but the above working PHP code should be pretty obvious and easy to port to it.