2014-10-01 03:18:38 +08:00
|
|
|
var Request = require('request');
|
|
|
|
var RSA = require('node-bignumber').Key;
|
|
|
|
var hex2b64 = require('node-bignumber').hex2b64;
|
2014-12-17 13:55:43 +08:00
|
|
|
var SteamID = require('steamid');
|
2014-10-01 03:18:38 +08:00
|
|
|
|
2015-06-18 12:22:05 +08:00
|
|
|
require('util').inherits(SteamCommunity, require('events').EventEmitter);
|
|
|
|
|
2014-10-01 03:18:38 +08:00
|
|
|
module.exports = SteamCommunity;
|
|
|
|
|
2014-12-17 13:55:43 +08:00
|
|
|
SteamCommunity.SteamID = SteamID;
|
|
|
|
|
2015-08-05 11:40:10 +08:00
|
|
|
function SteamCommunity(localAddress) {
|
2014-12-17 13:48:19 +08:00
|
|
|
this._jar = Request.jar();
|
2015-07-18 10:26:29 +08:00
|
|
|
this._captchaGid = -1;
|
2015-06-18 12:22:05 +08:00
|
|
|
this.chatState = SteamCommunity.ChatState.Offline;
|
2015-08-05 11:40:10 +08:00
|
|
|
|
2015-12-01 09:39:55 +08:00
|
|
|
var defaults = {
|
|
|
|
"jar": this._jar,
|
2015-12-02 09:57:26 +08:00
|
|
|
"timeout": 50000
|
2015-12-01 09:39:55 +08:00
|
|
|
};
|
|
|
|
|
2015-08-05 11:40:10 +08:00
|
|
|
if(localAddress) {
|
|
|
|
defaults.localAddress = localAddress;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.request = Request.defaults(defaults);
|
2015-07-11 04:00:56 +08:00
|
|
|
|
|
|
|
// English
|
|
|
|
this._jar.setCookie(Request.cookie('Steam_Language=english'), 'https://steamcommunity.com');
|
2015-08-07 12:28:24 +08:00
|
|
|
|
|
|
|
// UTC
|
|
|
|
this._jar.setCookie(Request.cookie('timezoneOffset=0,0'), 'https://steamcommunity.com');
|
2015-12-01 09:39:55 +08:00
|
|
|
|
|
|
|
this._jar.setCookie(Request.cookie("mobileClientVersion=0 (2.1.3)"), "https://steamcommunity.com");
|
|
|
|
this._jar.setCookie(Request.cookie("mobileClient=android"), "https://steamcommunity.com");
|
2014-10-01 03:18:38 +08:00
|
|
|
}
|
|
|
|
|
2014-12-17 13:48:19 +08:00
|
|
|
SteamCommunity.prototype.login = function(details, callback) {
|
2015-05-31 11:45:46 +08:00
|
|
|
if(details.steamguard) {
|
|
|
|
var parts = details.steamguard.split('||');
|
|
|
|
this._jar.setCookie(Request.cookie('steamMachineAuth' + parts[0] + '=' + encodeURIComponent(parts[1])), 'https://steamcommunity.com');
|
2014-10-01 03:18:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
var self = this;
|
2015-12-02 09:57:26 +08:00
|
|
|
|
|
|
|
// headers required to convince steam that we're logging in from a mobile device so that we can get the oAuth data
|
|
|
|
var mobileHeaders = {
|
|
|
|
"X-Requested-With": "com.valvesoftware.android.steam.community",
|
|
|
|
"referer": "https://steamcommunity.com/mobilelogin?oauth_client_id=DE45CD61&oauth_scope=read_profile%20write_profile%20read_client%20write_client",
|
|
|
|
"user-agent": "Mozilla/5.0 (Linux; U; Android 4.1.1; en-us; Google Nexus 4 - 4.1.1 - API 16 - 768x1280 Build/JRO03S) AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Mobile Safari/534.30",
|
|
|
|
"accept": "text/javascript, text/html, application/xml, text/xml, */*"
|
|
|
|
};
|
2015-12-01 09:39:55 +08:00
|
|
|
|
2015-12-02 09:57:26 +08:00
|
|
|
this.request.post("https://steamcommunity.com/login/getrsakey/", {
|
|
|
|
"form": {
|
|
|
|
"username": details.accountName
|
|
|
|
},
|
|
|
|
"headers": mobileHeaders
|
|
|
|
}, function(err, response, body) {
|
2014-10-01 03:18:38 +08:00
|
|
|
if(err) {
|
|
|
|
callback(err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var json;
|
|
|
|
try {
|
|
|
|
json = JSON.parse(body);
|
|
|
|
} catch(e) {
|
|
|
|
callback(e);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var key = new RSA();
|
|
|
|
key.setPublic(json.publickey_mod, json.publickey_exp);
|
|
|
|
|
|
|
|
var form = {
|
2015-07-18 10:26:29 +08:00
|
|
|
"captcha_text": details.captcha || "",
|
|
|
|
"captchagid": self._captchaGid,
|
2014-12-17 13:48:19 +08:00
|
|
|
"emailauth": details.authCode || "",
|
2014-10-01 03:18:38 +08:00
|
|
|
"emailsteamid": "",
|
2014-12-17 13:48:19 +08:00
|
|
|
"password": hex2b64(key.encrypt(details.password)),
|
2015-06-25 01:31:01 +08:00
|
|
|
"remember_login": "true",
|
2014-10-01 03:18:38 +08:00
|
|
|
"rsatimestamp": json.timestamp,
|
2015-07-18 10:57:40 +08:00
|
|
|
"twofactorcode": details.twoFactorCode || "",
|
2015-12-01 09:39:55 +08:00
|
|
|
"username": details.accountName,
|
|
|
|
"oauth_client_id": "DE45CD61",
|
|
|
|
"oauth_scope": "read_profile write_profile read_client write_client",
|
|
|
|
"loginfriendlyname": "#login_emailauth_friendlyname_mobile"
|
2014-10-01 03:18:38 +08:00
|
|
|
};
|
|
|
|
|
2015-06-25 01:31:01 +08:00
|
|
|
self.request.post({
|
|
|
|
"uri": "https://steamcommunity.com/login/dologin/",
|
|
|
|
"json": true,
|
2015-12-02 09:57:26 +08:00
|
|
|
"form": form,
|
|
|
|
"headers": mobileHeaders
|
2015-06-25 01:31:01 +08:00
|
|
|
}, function(err, response, body) {
|
2015-07-18 10:23:20 +08:00
|
|
|
if(self._checkHttpError(err, response, callback)) {
|
2014-10-01 03:18:38 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-06-25 01:31:01 +08:00
|
|
|
if(!body.success && body.emailauth_needed) {
|
2015-07-18 10:57:40 +08:00
|
|
|
// Steam Guard (email)
|
2015-07-18 10:23:20 +08:00
|
|
|
var error = new Error("SteamGuard");
|
2015-07-18 10:07:16 +08:00
|
|
|
error.emaildomain = body.emaildomain;
|
|
|
|
|
2015-07-18 10:23:20 +08:00
|
|
|
callback(error);
|
2015-07-18 10:57:40 +08:00
|
|
|
} else if(!body.success && body.requires_twofactor) {
|
|
|
|
// Steam Guard (app)
|
|
|
|
callback(new Error("SteamGuardMobile"));
|
2015-07-18 10:23:20 +08:00
|
|
|
} else if(!body.success && body.captcha_needed) {
|
|
|
|
var error = new Error("CAPTCHA");
|
|
|
|
error.captchaurl = "https://steamcommunity.com/public/captcha.php?gid=" + body.captcha_gid;
|
|
|
|
|
|
|
|
self._captchaGid = body.captcha_gid;
|
|
|
|
|
2015-07-18 10:07:16 +08:00
|
|
|
callback(error);
|
2015-06-25 01:31:01 +08:00
|
|
|
} else if(!body.success) {
|
2015-07-18 10:23:20 +08:00
|
|
|
callback(new Error(body.message || "Unknown error"));
|
2014-10-01 03:18:38 +08:00
|
|
|
} else {
|
2014-12-17 13:48:19 +08:00
|
|
|
var sessionID = generateSessionID();
|
2015-12-01 09:39:55 +08:00
|
|
|
var oAuth = JSON.parse( body.oauth );
|
2014-12-17 13:48:19 +08:00
|
|
|
self._jar.setCookie(Request.cookie('sessionid=' + sessionID), 'http://steamcommunity.com');
|
2014-10-01 03:18:38 +08:00
|
|
|
|
2015-12-01 09:39:55 +08:00
|
|
|
self.steamID = new SteamID(oAuth.steamid);
|
|
|
|
self.oAuthToken = oAuth.oauth_token;
|
|
|
|
|
2014-12-17 13:48:19 +08:00
|
|
|
var cookies = self._jar.getCookieString("https://steamcommunity.com").split(';').map(function(cookie) {
|
|
|
|
return cookie.trim();
|
|
|
|
});
|
2014-10-01 03:18:38 +08:00
|
|
|
|
2014-12-17 13:48:19 +08:00
|
|
|
// Find the Steam Guard cookie
|
|
|
|
var steamguard = null;
|
|
|
|
for(var i = 0; i < cookies.length; i++) {
|
|
|
|
var parts = cookies[i].split('=');
|
|
|
|
if(parts[0] == 'steamMachineAuth' + self.steamID) {
|
2015-05-31 11:45:46 +08:00
|
|
|
steamguard = self.steamID.toString() + '||' + decodeURIComponent(parts[1]);
|
2014-12-17 13:48:19 +08:00
|
|
|
break;
|
|
|
|
}
|
2014-10-01 03:18:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
callback(null, sessionID, cookies, steamguard);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2014-12-17 13:48:19 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
SteamCommunity.prototype.setCookies = function(cookies) {
|
|
|
|
var self = this;
|
|
|
|
cookies.forEach(function(cookie) {
|
|
|
|
var cookieName = cookie.match(/(.+)=/)[1];
|
|
|
|
if(cookieName == 'steamLogin') {
|
2014-12-17 13:55:43 +08:00
|
|
|
self.steamID = new SteamID(cookie.match(/=(\d+)/)[1]);
|
2014-12-17 13:48:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
self._jar.setCookie(Request.cookie(cookie), (cookieName.match(/^steamMachineAuth/) || cookieName.match(/Secure$/) ? "https://" : "http://") + "steamcommunity.com");
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
SteamCommunity.prototype.getSessionID = function() {
|
|
|
|
var cookies = this._jar.getCookieString("http://steamcommunity.com").split(';');
|
|
|
|
for(var i = 0; i < cookies.length; i++) {
|
|
|
|
var match = cookies[i].trim().match(/([^=]+)=(.+)/);
|
|
|
|
if(match[1] == 'sessionid') {
|
|
|
|
return decodeURIComponent(match[2]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var sessionID = generateSessionID();
|
|
|
|
this._jar.setCookie(Request.cookie('sessionid=' + sessionID), "http://steamcommunity.com");
|
|
|
|
return sessionID;
|
|
|
|
};
|
|
|
|
|
|
|
|
function generateSessionID() {
|
|
|
|
return Math.floor(Math.random() * 1000000000);
|
2015-11-27 14:28:31 +08:00
|
|
|
}
|
2014-12-17 14:48:17 +08:00
|
|
|
|
2015-05-11 13:07:14 +08:00
|
|
|
SteamCommunity.prototype.getWebApiKey = function(domain, callback) {
|
|
|
|
var self = this;
|
2015-09-03 05:19:12 +08:00
|
|
|
this.request({
|
|
|
|
"uri": "https://steamcommunity.com/dev/apikey",
|
|
|
|
"followRedirect": false
|
|
|
|
}, function(err, response, body) {
|
2015-09-03 05:28:13 +08:00
|
|
|
if(self._checkHttpError(err, response, callback)) {
|
2015-07-18 10:23:20 +08:00
|
|
|
return;
|
2015-05-11 13:07:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if(body.match(/<h2>Access Denied<\/h2>/)) {
|
2015-10-07 06:53:25 +08:00
|
|
|
return callback(new Error("Access Denied"));
|
2015-05-11 13:07:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
var match = body.match(/<p>Key: ([0-9A-F]+)<\/p>/);
|
|
|
|
if(match) {
|
|
|
|
// We already have an API key registered
|
|
|
|
callback(null, match[1]);
|
|
|
|
} else {
|
|
|
|
// We need to register a new API key
|
2015-06-10 02:45:07 +08:00
|
|
|
self.request.post('https://steamcommunity.com/dev/registerkey', {
|
2015-05-11 13:07:14 +08:00
|
|
|
"form": {
|
|
|
|
"domain": domain,
|
2015-05-29 13:03:07 +08:00
|
|
|
"agreeToTerms": "agreed",
|
|
|
|
"sessionid": self.getSessionID(),
|
|
|
|
"Submit": "Register"
|
2015-05-11 13:07:14 +08:00
|
|
|
}
|
|
|
|
}, function(err, response, body) {
|
2015-09-03 05:19:12 +08:00
|
|
|
if(self._checkHttpError(err, response, callback)) {
|
|
|
|
return;
|
2015-05-11 13:07:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
self.getWebApiKey(domain, callback);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-05-12 03:48:00 +08:00
|
|
|
SteamCommunity.prototype.parentalUnlock = function(pin, callback) {
|
2015-06-10 02:45:07 +08:00
|
|
|
this.request.post("https://steamcommunity.com/parental/ajaxunlock", {
|
2015-05-12 03:48:00 +08:00
|
|
|
"json": true,
|
|
|
|
"form": {
|
|
|
|
"pin": pin
|
|
|
|
}
|
|
|
|
}, function(err, response, body) {
|
2015-05-12 03:56:41 +08:00
|
|
|
if(!callback) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-07-18 10:23:20 +08:00
|
|
|
if(self._checkHttpError(err, response, callback)) {
|
|
|
|
return;
|
2015-05-12 03:48:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if(!body || typeof body.success !== 'boolean') {
|
|
|
|
return callback("Invalid response");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!body.success) {
|
|
|
|
return callback("Incorrect PIN");
|
|
|
|
}
|
|
|
|
|
|
|
|
callback();
|
|
|
|
}.bind(this));
|
|
|
|
};
|
|
|
|
|
2015-05-26 01:37:01 +08:00
|
|
|
SteamCommunity.prototype.getNotifications = function(callback) {
|
2015-11-11 12:46:17 +08:00
|
|
|
var self = this;
|
2015-06-10 02:45:07 +08:00
|
|
|
this.request.get("https://steamcommunity.com/actions/RefreshNotificationArea", function(err, response, body) {
|
2015-07-18 10:23:20 +08:00
|
|
|
if(self._checkHttpError(err, response, callback)) {
|
|
|
|
return;
|
2015-05-26 01:37:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
var notifications = {
|
|
|
|
"comments": 0,
|
|
|
|
"items": 0,
|
|
|
|
"invites": 0,
|
|
|
|
"gifts": 0,
|
|
|
|
"chat": 0,
|
|
|
|
"trades": 0
|
|
|
|
};
|
|
|
|
|
|
|
|
var items = {
|
|
|
|
"comments": /(\d+) new comments?/,
|
|
|
|
"items": /(\d+) new items? in your inventory/,
|
|
|
|
"invites": /(\d+) new invites?/,
|
|
|
|
"gifts": /(\d+) new gifts?/,
|
|
|
|
"chat": /(\d+) unread chat messages?/,
|
|
|
|
"trades": /(\d+) new trade notifications?/
|
|
|
|
};
|
|
|
|
|
|
|
|
var match;
|
|
|
|
for(var i in items) {
|
|
|
|
if(match = body.match(items[i])) {
|
|
|
|
notifications[i] = parseInt(match[1], 10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
callback(null, notifications);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-05-26 01:20:47 +08:00
|
|
|
SteamCommunity.prototype.resetItemNotifications = function(callback) {
|
2015-12-01 09:39:55 +08:00
|
|
|
var self = this;
|
2015-06-10 02:45:07 +08:00
|
|
|
this.request.get("https://steamcommunity.com/my/inventory", function(err, response, body) {
|
2015-05-26 01:20:47 +08:00
|
|
|
if(!callback) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-07-18 10:23:20 +08:00
|
|
|
if(self._checkHttpError(err, response, callback)) {
|
|
|
|
return;
|
2015-05-26 01:20:47 +08:00
|
|
|
}
|
2015-07-18 10:23:20 +08:00
|
|
|
|
|
|
|
callback(null);
|
2015-05-26 01:20:47 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-07-18 09:10:08 +08:00
|
|
|
SteamCommunity.prototype.loggedIn = function(callback) {
|
|
|
|
this.request("https://steamcommunity.com/my", {"followRedirect": false}, function(err, response, body) {
|
2015-07-18 09:19:43 +08:00
|
|
|
if(err || (response.statusCode != 302 && response.statusCode != 403)) {
|
2015-07-18 10:23:20 +08:00
|
|
|
callback(err || new Error("HTTP error " + response.statusCode));
|
2015-07-18 09:10:08 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-07-18 09:19:43 +08:00
|
|
|
if(response.statusCode == 403) {
|
|
|
|
callback(null, true, true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
callback(null, !!response.headers.location.match(/steamcommunity\.com(\/(id|profiles)\/[^\/]+)\/?/), false);
|
2015-07-18 09:10:08 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2014-12-17 15:04:24 +08:00
|
|
|
SteamCommunity.prototype._checkCommunityError = function(html, callback) {
|
|
|
|
if(html.match(/<h1>Sorry!<\/h1>/)) {
|
|
|
|
var match = html.match(/<h3>(.+)<\/h3>/);
|
2015-07-18 10:23:20 +08:00
|
|
|
callback(new Error(match ? match[1] : "Unknown error occurred"));
|
2014-12-17 15:04:24 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2014-12-17 16:07:49 +08:00
|
|
|
SteamCommunity.prototype._myProfile = function(endpoint, form, callback) {
|
|
|
|
var self = this;
|
2015-06-10 02:45:07 +08:00
|
|
|
this.request("https://steamcommunity.com/my", {"followRedirect": false}, function(err, response, body) {
|
2014-12-17 16:07:49 +08:00
|
|
|
if(err || response.statusCode != 302) {
|
|
|
|
callback(err || "HTTP error " + response.statusCode);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var match = response.headers.location.match(/steamcommunity\.com(\/(id|profiles)\/[^\/]+)\/?/);
|
|
|
|
if(!match) {
|
|
|
|
callback("Can't get profile URL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-06-10 02:45:07 +08:00
|
|
|
(form ? self.request.post : self.request)("https://steamcommunity.com" + match[1] + "/" + endpoint, form ? {"form": form} : {}, callback);
|
2014-12-17 16:07:49 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-09-03 05:28:13 +08:00
|
|
|
SteamCommunity.prototype._checkHttpError = function(err, response, callback) {
|
2015-07-18 10:23:20 +08:00
|
|
|
if(err) {
|
|
|
|
callback(err);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-09-03 05:28:13 +08:00
|
|
|
if(response.statusCode >= 300 && response.statusCode <= 399 && response.headers.location.indexOf('/login') != -1) {
|
|
|
|
callback(new Error("Not Logged In"));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(response.statusCode >= 400) {
|
2015-07-18 10:23:20 +08:00
|
|
|
var error = new Error("HTTP error " + response.statusCode);
|
|
|
|
error.code = response.statusCode;
|
|
|
|
callback(error);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2015-08-05 11:37:16 +08:00
|
|
|
require('./components/chat.js');
|
|
|
|
require('./components/profile.js');
|
|
|
|
require('./components/market.js');
|
|
|
|
require('./components/groups.js');
|
2015-08-14 11:57:08 +08:00
|
|
|
require('./components/users.js');
|
2015-09-10 00:04:28 +08:00
|
|
|
require('./components/inventoryhistory.js');
|
2015-11-27 14:28:31 +08:00
|
|
|
require('./components/webapi.js');
|
2015-11-27 14:45:58 +08:00
|
|
|
require('./components/twofactor.js');
|
2015-06-29 08:19:42 +08:00
|
|
|
require('./classes/CMarketItem.js');
|
2015-07-11 03:46:29 +08:00
|
|
|
require('./classes/CMarketSearchResult.js');
|
2014-12-17 14:48:17 +08:00
|
|
|
require('./classes/CSteamGroup.js');
|
2015-04-26 02:46:25 +08:00
|
|
|
require('./classes/CSteamUser.js');
|