2017-04-17 05:10:45 -04:00
|
|
|
import lbryio from './lbryio.js';
|
2016-12-09 01:54:18 -05:00
|
|
|
import lighthouse from './lighthouse.js';
|
2017-03-08 01:23:07 -05:00
|
|
|
import jsonrpc from './jsonrpc.js';
|
2017-04-18 15:14:42 -04:00
|
|
|
import lbryuri from './lbryuri.js';
|
2017-04-13 14:52:26 -04:00
|
|
|
import {getLocal, getSession, setSession, setLocal} from './utils.js';
|
2016-12-09 01:54:18 -05:00
|
|
|
|
2017-04-27 02:52:14 -04:00
|
|
|
const {remote, ipcRenderer} = require('electron');
|
2017-03-08 03:56:34 -05:00
|
|
|
const menu = remote.require('./menu/main-menu');
|
|
|
|
|
2017-03-14 23:05:07 -04:00
|
|
|
/**
|
|
|
|
* Records a publish attempt in local storage. Returns a dictionary with all the data needed to
|
|
|
|
* needed to make a dummy claim or file info object.
|
|
|
|
*/
|
2017-04-12 19:24:04 -04:00
|
|
|
function savePendingPublish({name, channel_name}) {
|
2017-04-18 15:14:42 -04:00
|
|
|
let uri;
|
2017-04-13 20:56:58 -04:00
|
|
|
if (channel_name) {
|
2017-04-18 15:14:42 -04:00
|
|
|
uri = lbryuri.build({name: channel_name, path: name}, false);
|
2017-04-13 20:56:58 -04:00
|
|
|
} else {
|
2017-04-18 15:14:42 -04:00
|
|
|
uri = lbryuri.build({name: name}, false);
|
2017-04-13 20:56:58 -04:00
|
|
|
}
|
2017-03-14 23:05:07 -04:00
|
|
|
const pendingPublishes = getLocal('pendingPublishes') || [];
|
|
|
|
const newPendingPublish = {
|
2017-04-12 19:24:04 -04:00
|
|
|
name, channel_name,
|
2017-04-18 15:14:42 -04:00
|
|
|
claim_id: 'pending_claim_' + uri,
|
|
|
|
txid: 'pending_' + uri,
|
2017-03-14 23:05:07 -04:00
|
|
|
nout: 0,
|
2017-04-18 15:14:42 -04:00
|
|
|
outpoint: 'pending_' + uri + ':0',
|
2017-03-14 23:05:07 -04:00
|
|
|
time: Date.now(),
|
|
|
|
};
|
|
|
|
setLocal('pendingPublishes', [...pendingPublishes, newPendingPublish]);
|
|
|
|
return newPendingPublish;
|
|
|
|
}
|
|
|
|
|
2017-04-12 19:24:04 -04:00
|
|
|
/**
|
|
|
|
* If there is a pending publish with the given name or outpoint, remove it.
|
|
|
|
* A channel name may also be provided along with name.
|
|
|
|
*/
|
|
|
|
function removePendingPublishIfNeeded({name, channel_name, outpoint}) {
|
|
|
|
function pubMatches(pub) {
|
|
|
|
return pub.outpoint === outpoint || (pub.name === name && (!channel_name || pub.channel_name === channel_name));
|
|
|
|
}
|
|
|
|
|
|
|
|
setLocal('pendingPublishes', getPendingPublishes().filter(pub => !pubMatches(pub)));
|
2017-03-14 23:05:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the current list of pending publish attempts. Filters out any that have timed out and
|
|
|
|
* removes them from the list.
|
|
|
|
*/
|
|
|
|
function getPendingPublishes() {
|
|
|
|
const pendingPublishes = getLocal('pendingPublishes') || [];
|
2017-04-12 19:24:04 -04:00
|
|
|
const newPendingPublishes = pendingPublishes.filter(pub => Date.now() - pub.time <= lbry.pendingPublishTimeout);
|
2017-03-14 23:05:07 -04:00
|
|
|
setLocal('pendingPublishes', newPendingPublishes);
|
2017-04-12 19:24:04 -04:00
|
|
|
return newPendingPublishes;
|
2017-03-14 23:05:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-04-12 19:24:04 -04:00
|
|
|
* Gets a pending publish attempt by its name or (fake) outpoint. A channel name can also be
|
|
|
|
* provided along withe the name. If no pending publish is found, returns null.
|
2017-03-14 23:05:07 -04:00
|
|
|
*/
|
2017-04-12 19:24:04 -04:00
|
|
|
function getPendingPublish({name, channel_name, outpoint}) {
|
2017-03-14 23:05:07 -04:00
|
|
|
const pendingPublishes = getPendingPublishes();
|
2017-04-12 19:24:04 -04:00
|
|
|
return pendingPublishes.find(
|
|
|
|
pub => pub.outpoint === outpoint || (pub.name === name && (!channel_name || pub.channel_name === channel_name))
|
|
|
|
) || null;
|
2017-03-14 23:05:07 -04:00
|
|
|
}
|
|
|
|
|
2017-04-12 19:24:04 -04:00
|
|
|
function pendingPublishToDummyClaim({channel_name, name, outpoint, claim_id, txid, nout}) {
|
2017-04-13 20:56:58 -04:00
|
|
|
return {name, outpoint, claim_id, txid, nout, channel_name};
|
2017-03-14 23:05:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function pendingPublishToDummyFileInfo({name, outpoint, claim_id}) {
|
2017-04-13 20:56:58 -04:00
|
|
|
return {name, outpoint, claim_id, metadata: null};
|
2017-03-14 23:05:07 -04:00
|
|
|
}
|
2017-04-13 20:56:58 -04:00
|
|
|
window.pptdfi = pendingPublishToDummyFileInfo;
|
2017-03-14 23:05:07 -04:00
|
|
|
|
2017-03-08 01:23:07 -05:00
|
|
|
let lbry = {
|
2016-03-14 18:05:24 -04:00
|
|
|
isConnected: false,
|
2016-03-15 12:05:11 -04:00
|
|
|
rootPath: '.',
|
2016-08-18 02:57:19 -04:00
|
|
|
daemonConnectionString: 'http://localhost:5279/lbryapi',
|
2016-12-06 14:58:51 -05:00
|
|
|
webUiUri: 'http://localhost:5279',
|
2017-01-23 23:31:06 -05:00
|
|
|
peerListTimeout: 6000,
|
2017-03-14 23:05:07 -04:00
|
|
|
pendingPublishTimeout: 20 * 60 * 1000,
|
2016-03-14 18:05:24 -04:00
|
|
|
colors: {
|
|
|
|
primary: '#155B4A'
|
2016-08-22 15:07:41 -04:00
|
|
|
},
|
2016-08-22 15:19:04 -04:00
|
|
|
defaultClientSettings: {
|
|
|
|
showNsfw: false,
|
2017-01-20 08:54:33 -05:00
|
|
|
showUnavailable: true,
|
2016-12-29 04:17:10 -05:00
|
|
|
debug: false,
|
|
|
|
useCustomLighthouseServers: false,
|
|
|
|
customLighthouseServers: [],
|
2017-03-08 18:34:07 -05:00
|
|
|
showDeveloperMenu: false,
|
2016-08-22 15:19:04 -04:00
|
|
|
}
|
2016-03-14 18:05:24 -04:00
|
|
|
};
|
|
|
|
|
2016-08-18 02:57:19 -04:00
|
|
|
lbry.call = function (method, params, callback, errorCallback, connectFailedCallback) {
|
2017-05-03 13:54:36 -04:00
|
|
|
jsonrpc.call(lbry.daemonConnectionString, method, params, callback, errorCallback, connectFailedCallback);
|
2016-08-18 02:57:19 -04:00
|
|
|
}
|
|
|
|
|
2016-03-14 18:05:24 -04:00
|
|
|
//core
|
2017-04-09 11:06:23 -04:00
|
|
|
lbry._connectPromise = null;
|
|
|
|
lbry.connect = function() {
|
|
|
|
if (lbry._connectPromise === null) {
|
|
|
|
|
|
|
|
lbry._connectPromise = new Promise((resolve, reject) => {
|
|
|
|
|
|
|
|
// Check every half second to see if the daemon is accepting connections
|
|
|
|
function checkDaemonStarted(tryNum = 0) {
|
|
|
|
lbry.isDaemonAcceptingConnections(function (runningStatus) {
|
|
|
|
if (runningStatus) {
|
|
|
|
resolve(true);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (tryNum <= 600) { // Move # of tries into constant or config option
|
|
|
|
setTimeout(function () {
|
|
|
|
checkDaemonStarted(tryNum + 1);
|
|
|
|
}, tryNum < 100 ? 200 : 1000);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
reject(new Error("Unable to connect to LBRY"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2016-03-24 03:08:39 -04:00
|
|
|
}
|
2017-04-09 11:06:23 -04:00
|
|
|
|
|
|
|
checkDaemonStarted();
|
2016-03-24 03:08:39 -04:00
|
|
|
});
|
2016-03-24 21:20:58 -04:00
|
|
|
}
|
2017-04-09 11:06:23 -04:00
|
|
|
|
|
|
|
return lbry._connectPromise;
|
2016-03-14 18:05:24 -04:00
|
|
|
}
|
|
|
|
|
2017-04-30 20:15:21 -04:00
|
|
|
|
|
|
|
//kill this but still better than document.title =, which this replaced
|
|
|
|
lbry.setTitle = function(title) {
|
|
|
|
document.title = title + " - LBRY";
|
|
|
|
}
|
|
|
|
|
|
|
|
//kill this with proper routing
|
|
|
|
lbry.back = function() {
|
|
|
|
if (window.history.length > 1) {
|
|
|
|
window.history.back();
|
|
|
|
} else {
|
|
|
|
window.location.href = "?discover";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-18 09:29:47 -06:00
|
|
|
lbry.isDaemonAcceptingConnections = function (callback) {
|
|
|
|
// Returns true/false whether the daemon is at a point it will start returning status
|
|
|
|
lbry.call('status', {}, () => callback(true), null, () => callback(false))
|
2016-03-24 21:20:58 -04:00
|
|
|
};
|
|
|
|
|
2016-07-04 11:29:58 -04:00
|
|
|
lbry.checkFirstRun = function(callback) {
|
|
|
|
lbry.call('is_first_run', {}, callback);
|
|
|
|
}
|
|
|
|
|
2017-04-10 13:44:54 -04:00
|
|
|
lbry.getUnusedAddress = function(callback) {
|
|
|
|
lbry.call('wallet_unused_address', {}, callback);
|
2016-07-04 11:29:58 -04:00
|
|
|
}
|
|
|
|
|
2017-01-12 21:03:34 -05:00
|
|
|
lbry.checkAddressIsMine = function(address, callback) {
|
2016-09-02 01:34:30 -04:00
|
|
|
lbry.call('address_is_mine', {address: address}, callback);
|
|
|
|
}
|
|
|
|
|
2016-08-22 14:13:41 -04:00
|
|
|
lbry.getDaemonSettings = function(callback) {
|
2016-04-03 08:45:33 -04:00
|
|
|
lbry.call('get_settings', {}, callback);
|
2016-08-22 14:13:41 -04:00
|
|
|
}
|
2016-04-03 08:45:33 -04:00
|
|
|
|
2016-08-22 14:13:41 -04:00
|
|
|
lbry.setDaemonSettings = function(settings, callback) {
|
2016-04-03 08:45:33 -04:00
|
|
|
lbry.call('set_settings', settings, callback);
|
2016-08-22 14:13:41 -04:00
|
|
|
}
|
2017-05-03 23:44:08 -04:00
|
|
|
|
2016-08-22 14:13:41 -04:00
|
|
|
lbry.setDaemonSetting = function(setting, value, callback) {
|
|
|
|
var setSettingsArgs = {};
|
|
|
|
setSettingsArgs[setting] = value;
|
|
|
|
lbry.call('set_settings', setSettingsArgs, callback)
|
|
|
|
}
|
2016-04-03 08:45:33 -04:00
|
|
|
|
2016-08-22 15:07:41 -04:00
|
|
|
|
2017-04-10 13:44:54 -04:00
|
|
|
lbry.getBalance = function(callback) {
|
|
|
|
lbry.call("wallet_balance", {}, callback);
|
2016-03-14 18:05:24 -04:00
|
|
|
}
|
|
|
|
|
2017-04-10 13:44:54 -04:00
|
|
|
lbry.sendToAddress = function(amount, address, callback, errorCallback) {
|
2016-07-22 10:33:05 -05:00
|
|
|
lbry.call("send_amount_to_address", { "amount" : amount, "address": address }, callback, errorCallback);
|
2016-07-18 17:40:15 -05:00
|
|
|
}
|
|
|
|
|
2016-07-28 18:02:24 -04:00
|
|
|
lbry.getClaimInfo = function(name, callback) {
|
2017-04-10 13:44:54 -04:00
|
|
|
if (!name) {
|
|
|
|
throw new Error(`Name required.`);
|
|
|
|
}
|
2016-07-28 18:02:24 -04:00
|
|
|
lbry.call('get_claim_info', { name: name }, callback);
|
|
|
|
}
|
|
|
|
|
2016-09-16 13:45:33 -04:00
|
|
|
lbry.getMyClaim = function(name, callback) {
|
2017-01-25 00:34:18 -05:00
|
|
|
lbry.call('claim_list_mine', {}, (claims) => {
|
|
|
|
callback(claims.find((claim) => claim.name == name) || null);
|
|
|
|
});
|
2016-09-16 13:45:33 -04:00
|
|
|
}
|
|
|
|
|
2016-12-07 16:29:04 -05:00
|
|
|
lbry.getPeersForBlobHash = function(blobHash, callback) {
|
2017-01-23 23:31:06 -05:00
|
|
|
let timedOut = false;
|
|
|
|
const timeout = setTimeout(() => {
|
|
|
|
timedOut = true;
|
|
|
|
callback([]);
|
|
|
|
}, lbry.peerListTimeout);
|
|
|
|
|
|
|
|
lbry.call('peer_list', { blob_hash: blobHash }, function(peers) {
|
|
|
|
if (!timedOut) {
|
|
|
|
clearTimeout(timeout);
|
|
|
|
callback(peers);
|
|
|
|
}
|
|
|
|
});
|
2016-12-07 16:29:04 -05:00
|
|
|
}
|
|
|
|
|
2017-04-13 14:57:12 -04:00
|
|
|
/**
|
|
|
|
* Takes a LBRY URI; will first try and calculate a total cost using
|
|
|
|
* Lighthouse. If Lighthouse can't be reached, it just retrives the
|
|
|
|
* key fee.
|
|
|
|
*
|
|
|
|
* Returns an object with members:
|
|
|
|
* - cost: Number; the calculated cost of the name
|
|
|
|
* - includes_data: Boolean; indicates whether or not the data fee info
|
|
|
|
* from Lighthouse is included.
|
|
|
|
*/
|
2017-04-13 15:37:41 -04:00
|
|
|
lbry.costPromiseCache = {}
|
2017-04-18 15:14:42 -04:00
|
|
|
lbry.getCostInfo = function(uri) {
|
|
|
|
if (lbry.costPromiseCache[uri] === undefined) {
|
|
|
|
lbry.costPromiseCache[uri] = new Promise((resolve, reject) => {
|
2017-04-17 16:22:08 -04:00
|
|
|
const COST_INFO_CACHE_KEY = 'cost_info_cache';
|
2017-04-13 15:37:41 -04:00
|
|
|
let costInfoCache = getSession(COST_INFO_CACHE_KEY, {})
|
2017-04-13 14:57:12 -04:00
|
|
|
|
2017-04-17 16:22:08 -04:00
|
|
|
function cacheAndResolve(cost, includesData) {
|
2017-04-18 15:14:42 -04:00
|
|
|
costInfoCache[uri] = {cost, includesData};
|
2017-04-17 16:22:08 -04:00
|
|
|
setSession(COST_INFO_CACHE_KEY, costInfoCache);
|
|
|
|
resolve({cost, includesData});
|
|
|
|
}
|
|
|
|
|
2017-04-18 15:14:42 -04:00
|
|
|
if (!uri) {
|
2017-04-17 16:22:08 -04:00
|
|
|
return reject(new Error(`URI required.`));
|
2017-04-13 15:37:41 -04:00
|
|
|
}
|
2017-04-13 14:57:12 -04:00
|
|
|
|
2017-04-18 15:14:42 -04:00
|
|
|
if (costInfoCache[uri] && costInfoCache[uri].cost) {
|
|
|
|
return resolve(costInfoCache[uri])
|
2017-04-13 15:37:41 -04:00
|
|
|
}
|
2017-04-12 01:02:29 -04:00
|
|
|
|
2017-04-18 15:14:42 -04:00
|
|
|
function getCost(uri, size) {
|
|
|
|
lbry.stream_cost_estimate({uri, ... size !== null ? {size} : {}}).then((cost) => {
|
2017-04-17 16:22:08 -04:00
|
|
|
cacheAndResolve(cost, size !== null);
|
2017-04-13 15:37:41 -04:00
|
|
|
}, reject);
|
|
|
|
}
|
2017-04-10 21:24:35 -04:00
|
|
|
|
2017-04-18 15:14:42 -04:00
|
|
|
function getCostGenerous(uri) {
|
2017-04-17 16:22:08 -04:00
|
|
|
// If generous is on, the calculation is simple enough that we might as well do it here in the front end
|
2017-04-18 15:14:42 -04:00
|
|
|
lbry.resolve({uri: uri}).then((resolutionInfo) => {
|
2017-05-09 23:06:48 -04:00
|
|
|
if (!resolutionInfo) {
|
|
|
|
return reject(new Error("Unused URI"));
|
|
|
|
}
|
2017-04-17 16:22:08 -04:00
|
|
|
const fee = resolutionInfo.claim.value.stream.metadata.fee;
|
|
|
|
if (fee === undefined) {
|
|
|
|
cacheAndResolve(0, true);
|
|
|
|
} else if (fee.currency == 'LBC') {
|
|
|
|
cacheAndResolve(fee.amount, true);
|
|
|
|
} else {
|
|
|
|
lbryio.getExchangeRates().then(({lbc_usd}) => {
|
|
|
|
cacheAndResolve(fee.amount / lbc_usd, true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-04-18 15:14:42 -04:00
|
|
|
const uriObj = lbryuri.parse(uri);
|
2017-04-13 15:37:41 -04:00
|
|
|
const name = uriObj.path || uriObj.name;
|
|
|
|
|
2017-04-17 16:22:08 -04:00
|
|
|
lbry.settings_get({allow_cached: true}).then(({is_generous_host}) => {
|
|
|
|
if (is_generous_host) {
|
2017-04-18 15:14:42 -04:00
|
|
|
return getCostGenerous(uri);
|
2017-04-13 15:37:41 -04:00
|
|
|
}
|
2017-04-17 16:22:08 -04:00
|
|
|
|
|
|
|
lighthouse.get_size_for_name(name).then((size) => {
|
|
|
|
if (size) {
|
|
|
|
getCost(name, size);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
getCost(name, null);
|
|
|
|
}
|
|
|
|
}, () => {
|
2017-04-13 15:37:41 -04:00
|
|
|
getCost(name, null);
|
2017-04-17 16:22:08 -04:00
|
|
|
});
|
2017-04-13 15:37:41 -04:00
|
|
|
});
|
2017-04-13 14:52:26 -04:00
|
|
|
});
|
2017-04-13 15:37:41 -04:00
|
|
|
}
|
2017-04-18 15:14:42 -04:00
|
|
|
return lbry.costPromiseCache[uri];
|
2016-12-09 01:54:18 -05:00
|
|
|
}
|
|
|
|
|
2016-11-09 06:27:07 -05:00
|
|
|
lbry.getMyClaims = function(callback) {
|
|
|
|
lbry.call('get_name_claims', {}, callback);
|
|
|
|
}
|
|
|
|
|
2017-03-08 02:17:16 -05:00
|
|
|
lbry.removeFile = function(outpoint, deleteTargetFile=true, callback) {
|
|
|
|
this._removedFiles.push(outpoint);
|
2017-04-30 00:02:25 +07:00
|
|
|
// this._updateFileInfoSubscribers(outpoint);
|
2017-01-17 04:04:29 -05:00
|
|
|
|
2017-03-08 02:17:16 -05:00
|
|
|
lbry.file_delete({
|
|
|
|
outpoint: outpoint,
|
2016-05-30 08:54:08 -04:00
|
|
|
delete_target_file: deleteTargetFile,
|
2017-03-08 02:17:16 -05:00
|
|
|
}).then(callback);
|
2016-05-30 08:54:08 -04:00
|
|
|
}
|
|
|
|
|
2016-09-16 11:17:12 -04:00
|
|
|
lbry.getFileInfoWhenListed = function(name, callback, timeoutCallback, tryNum=0) {
|
2017-02-16 20:50:09 -05:00
|
|
|
function scheduleNextCheckOrTimeout() {
|
2017-02-01 04:08:09 -05:00
|
|
|
if (timeoutCallback && tryNum > 200) {
|
2016-09-16 11:17:12 -04:00
|
|
|
timeoutCallback();
|
2017-02-01 04:08:09 -05:00
|
|
|
} else {
|
2017-02-16 20:50:09 -05:00
|
|
|
setTimeout(() => lbry.getFileInfoWhenListed(name, callback, timeoutCallback, tryNum + 1), 250);
|
2016-09-16 11:17:12 -04:00
|
|
|
}
|
2017-02-16 20:50:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Calls callback with file info when it appears in the lbrynet file manager.
|
|
|
|
// If timeoutCallback is provided, it will be called if the file fails to appear.
|
2017-03-08 01:23:07 -05:00
|
|
|
lbry.file_list({name: name}).then(([fileInfo]) => {
|
2017-02-16 20:50:09 -05:00
|
|
|
if (fileInfo) {
|
|
|
|
callback(fileInfo);
|
|
|
|
} else {
|
|
|
|
scheduleNextCheckOrTimeout();
|
|
|
|
}
|
|
|
|
}, () => scheduleNextCheckOrTimeout());
|
2016-09-16 11:17:12 -04:00
|
|
|
}
|
|
|
|
|
2017-03-14 23:05:07 -04:00
|
|
|
/**
|
|
|
|
* Publishes a file. The optional fileListedCallback is called when the file becomes available in
|
|
|
|
* lbry.file_list() during the publish process.
|
|
|
|
*
|
|
|
|
* This currently includes a work-around to cache the file in local storage so that the pending
|
|
|
|
* publish can appear in the UI immediately.
|
|
|
|
*/
|
2016-09-16 11:17:12 -04:00
|
|
|
lbry.publish = function(params, fileListedCallback, publishedCallback, errorCallback) {
|
2017-03-14 23:05:07 -04:00
|
|
|
lbry.call('publish', params, (result) => {
|
|
|
|
if (returnedPending) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
clearTimeout(returnPendingTimeout);
|
|
|
|
publishedCallback(result);
|
|
|
|
}, (err) => {
|
|
|
|
if (returnedPending) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
clearTimeout(returnPendingTimeout);
|
|
|
|
errorCallback(err);
|
|
|
|
});
|
|
|
|
|
|
|
|
let returnedPending = false;
|
|
|
|
// Give a short grace period in case publish() returns right away or (more likely) gives an error
|
|
|
|
const returnPendingTimeout = setTimeout(() => {
|
|
|
|
returnedPending = true;
|
|
|
|
|
|
|
|
if (publishedCallback) {
|
2017-04-13 20:56:58 -04:00
|
|
|
savePendingPublish({name: params.name, channel_name: params.channel_name});
|
2017-03-14 23:05:07 -04:00
|
|
|
publishedCallback(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fileListedCallback) {
|
2017-04-12 19:24:04 -04:00
|
|
|
const {name, channel_name} = params;
|
2017-04-13 20:56:58 -04:00
|
|
|
savePendingPublish({name: params.name, channel_name: params.channel_name});
|
2017-03-14 23:05:07 -04:00
|
|
|
fileListedCallback(true);
|
|
|
|
}
|
|
|
|
}, 2000);
|
|
|
|
|
|
|
|
//lbry.getFileInfoWhenListed(params.name, function(fileInfo) {
|
|
|
|
// fileListedCallback(fileInfo);
|
|
|
|
//});
|
2016-07-15 08:01:51 -04:00
|
|
|
}
|
|
|
|
|
2016-04-12 06:30:25 -04:00
|
|
|
|
2016-08-22 15:07:41 -04:00
|
|
|
lbry.getClientSettings = function() {
|
|
|
|
var outSettings = {};
|
|
|
|
for (let setting of Object.keys(lbry.defaultClientSettings)) {
|
|
|
|
var localStorageVal = localStorage.getItem('setting_' + setting);
|
|
|
|
outSettings[setting] = (localStorageVal === null ? lbry.defaultClientSettings[setting] : JSON.parse(localStorageVal));
|
|
|
|
}
|
|
|
|
return outSettings;
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.getClientSetting = function(setting) {
|
|
|
|
var localStorageVal = localStorage.getItem('setting_' + setting);
|
2017-04-09 11:06:23 -04:00
|
|
|
if (setting == 'showDeveloperMenu')
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-22 15:07:41 -04:00
|
|
|
return (localStorageVal === null ? lbry.defaultClientSettings[setting] : JSON.parse(localStorageVal));
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.setClientSettings = function(settings) {
|
|
|
|
for (let setting of Object.keys(settings)) {
|
|
|
|
lbry.setClientSetting(setting, settings[setting]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.setClientSetting = function(setting, value) {
|
|
|
|
return localStorage.setItem('setting_' + setting, JSON.stringify(value));
|
|
|
|
}
|
|
|
|
|
2016-11-30 00:23:45 -05:00
|
|
|
lbry.getSessionInfo = function(callback) {
|
|
|
|
lbry.call('get_lbry_session_info', {}, callback);
|
|
|
|
}
|
2016-08-22 15:07:41 -04:00
|
|
|
|
2016-04-23 08:19:15 -04:00
|
|
|
lbry.reportBug = function(message, callback) {
|
2017-01-02 15:43:34 -05:00
|
|
|
lbry.call('report_bug', {
|
2016-04-23 08:19:15 -04:00
|
|
|
message: message
|
|
|
|
}, callback);
|
|
|
|
}
|
|
|
|
|
2016-03-14 18:05:24 -04:00
|
|
|
//utilities
|
|
|
|
lbry.formatCredits = function(amount, precision)
|
|
|
|
{
|
2016-08-19 03:24:12 -04:00
|
|
|
return amount.toFixed(precision || 1).replace(/\.?0+$/, '');
|
2016-03-14 18:05:24 -04:00
|
|
|
}
|
|
|
|
|
2016-09-01 03:28:07 -04:00
|
|
|
lbry.formatName = function(name) {
|
2016-11-15 01:31:28 -06:00
|
|
|
// Converts LBRY name to standard format (all lower case, no special characters, spaces replaced by dashes)
|
2016-11-16 13:56:04 -06:00
|
|
|
name = name.replace('/\s+/g', '-');
|
2016-11-15 01:31:28 -06:00
|
|
|
name = name.toLowerCase().replace(/[^a-z0-9\-]/g, '');
|
|
|
|
return name;
|
2016-09-01 03:28:07 -04:00
|
|
|
}
|
|
|
|
|
2016-11-18 04:00:03 -05:00
|
|
|
lbry.nameIsValid = function(name, checkCase=true) {
|
|
|
|
const regexp = new RegExp('^[a-z0-9-]+$', checkCase ? '' : 'i');
|
|
|
|
return regexp.test(name);
|
|
|
|
}
|
|
|
|
|
2016-03-14 18:05:24 -04:00
|
|
|
lbry.loadJs = function(src, type, onload)
|
|
|
|
{
|
|
|
|
var lbryScriptTag = document.getElementById('lbry'),
|
|
|
|
newScriptTag = document.createElement('script'),
|
|
|
|
type = type || 'text/javascript';
|
2016-08-07 20:57:12 -04:00
|
|
|
|
2016-03-14 18:05:24 -04:00
|
|
|
newScriptTag.src = src;
|
|
|
|
newScriptTag.type = type;
|
|
|
|
if (onload)
|
|
|
|
{
|
2016-11-22 14:28:16 -06:00
|
|
|
newScriptTag.onload = onload;
|
2016-03-14 18:05:24 -04:00
|
|
|
}
|
|
|
|
lbryScriptTag.parentNode.insertBefore(newScriptTag, lbryScriptTag);
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.imagePath = function(file)
|
2016-08-07 20:57:12 -04:00
|
|
|
{
|
|
|
|
return lbry.rootPath + '/img/' + file;
|
2016-03-14 18:05:24 -04:00
|
|
|
}
|
2016-04-20 04:46:36 -04:00
|
|
|
|
2016-09-02 04:48:01 -04:00
|
|
|
lbry.getMediaType = function(contentType, fileName) {
|
|
|
|
if (contentType) {
|
|
|
|
return /^[^/]+/.exec(contentType);
|
2016-09-08 04:35:41 -04:00
|
|
|
} else if (fileName) {
|
|
|
|
var dotIndex = fileName.lastIndexOf('.');
|
2016-09-02 04:48:01 -04:00
|
|
|
if (dotIndex == -1) {
|
|
|
|
return 'unknown';
|
|
|
|
}
|
|
|
|
|
2016-09-08 04:35:41 -04:00
|
|
|
var ext = fileName.substr(dotIndex + 1);
|
2016-09-02 04:48:01 -04:00
|
|
|
if (/^mp4|mov|m4v|flv|f4v$/i.test(ext)) {
|
|
|
|
return 'video';
|
|
|
|
} else if (/^mp3|m4a|aac|wav|flac|ogg$/i.test(ext)) {
|
|
|
|
return 'audio';
|
|
|
|
} else if (/^html|htm|pdf|odf|doc|docx|md|markdown|txt$/i.test(ext)) {
|
|
|
|
return 'document';
|
|
|
|
} else {
|
|
|
|
return 'unknown';
|
|
|
|
}
|
2016-09-08 04:35:41 -04:00
|
|
|
} else {
|
|
|
|
return 'unknown';
|
2016-05-16 04:16:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-20 04:46:36 -04:00
|
|
|
lbry.stop = function(callback) {
|
|
|
|
lbry.call('stop', {}, callback);
|
|
|
|
};
|
2016-05-16 04:16:40 -04:00
|
|
|
|
2017-01-12 21:03:34 -05:00
|
|
|
lbry.fileInfo = {};
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry._subscribeIdCount = 0;
|
2017-01-12 21:03:34 -05:00
|
|
|
lbry._fileInfoSubscribeCallbacks = {};
|
2017-04-11 22:01:45 -04:00
|
|
|
lbry._fileInfoSubscribeInterval = 500000;
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry._balanceSubscribeCallbacks = {};
|
|
|
|
lbry._balanceSubscribeInterval = 5000;
|
2017-01-17 04:04:29 -05:00
|
|
|
lbry._removedFiles = [];
|
2017-03-08 02:17:16 -05:00
|
|
|
lbry._claimIdOwnershipCache = {};
|
2017-01-12 21:03:34 -05:00
|
|
|
|
|
|
|
lbry._updateClaimOwnershipCache = function(claimId) {
|
2017-01-17 05:06:39 -05:00
|
|
|
lbry.getMyClaims((claimInfos) => {
|
|
|
|
lbry._claimIdOwnershipCache[claimId] = !!claimInfos.reduce(function(match, claimInfo) {
|
2017-01-12 21:03:34 -05:00
|
|
|
return match || claimInfo.claim_id == claimId;
|
2017-04-25 21:22:44 -04:00
|
|
|
}, false);
|
2017-01-12 21:03:34 -05:00
|
|
|
});
|
2017-04-11 22:01:45 -04:00
|
|
|
|
2017-01-12 21:03:34 -05:00
|
|
|
};
|
|
|
|
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry._updateFileInfoSubscribers = function(outpoint) {
|
2017-03-08 02:17:16 -05:00
|
|
|
const callSubscribedCallbacks = (outpoint, fileInfo) => {
|
2017-03-27 01:44:13 -04:00
|
|
|
for (let callback of Object.values(this._fileInfoSubscribeCallbacks[outpoint])) {
|
2017-01-17 05:11:21 -05:00
|
|
|
callback(fileInfo);
|
|
|
|
}
|
2017-01-17 04:04:29 -05:00
|
|
|
}
|
|
|
|
|
2017-03-08 02:17:16 -05:00
|
|
|
if (lbry._removedFiles.includes(outpoint)) {
|
|
|
|
callSubscribedCallbacks(outpoint, false);
|
2017-01-17 04:04:29 -05:00
|
|
|
} else {
|
2017-03-14 12:15:21 -04:00
|
|
|
lbry.file_list({
|
|
|
|
outpoint: outpoint,
|
|
|
|
full_status: true,
|
|
|
|
}).then(([fileInfo]) => {
|
2017-01-17 04:04:29 -05:00
|
|
|
if (fileInfo) {
|
|
|
|
if (this._claimIdOwnershipCache[fileInfo.claim_id] === undefined) {
|
|
|
|
this._updateClaimOwnershipCache(fileInfo.claim_id);
|
|
|
|
}
|
|
|
|
fileInfo.isMine = !!this._claimIdOwnershipCache[fileInfo.claim_id];
|
2017-01-12 21:03:34 -05:00
|
|
|
}
|
2017-01-17 04:04:29 -05:00
|
|
|
|
2017-03-08 02:17:16 -05:00
|
|
|
callSubscribedCallbacks(outpoint, fileInfo);
|
2017-01-12 21:03:34 -05:00
|
|
|
});
|
2017-01-17 04:04:29 -05:00
|
|
|
}
|
|
|
|
|
2017-03-08 02:17:16 -05:00
|
|
|
if (Object.keys(this._fileInfoSubscribeCallbacks[outpoint]).length) {
|
2017-01-12 23:05:43 -05:00
|
|
|
setTimeout(() => {
|
2017-03-26 14:30:18 -04:00
|
|
|
this._updateFileInfoSubscribers(outpoint);
|
2017-01-12 23:05:43 -05:00
|
|
|
}, lbry._fileInfoSubscribeInterval);
|
|
|
|
}
|
2017-01-12 21:03:34 -05:00
|
|
|
}
|
|
|
|
|
2017-03-08 02:17:16 -05:00
|
|
|
lbry.fileInfoSubscribe = function(outpoint, callback) {
|
|
|
|
if (!lbry._fileInfoSubscribeCallbacks[outpoint])
|
2017-01-12 21:03:34 -05:00
|
|
|
{
|
2017-03-08 02:17:16 -05:00
|
|
|
lbry._fileInfoSubscribeCallbacks[outpoint] = {};
|
2017-01-12 21:03:34 -05:00
|
|
|
}
|
|
|
|
|
2017-03-26 14:30:18 -04:00
|
|
|
const subscribeId = ++lbry._subscribeIdCount;
|
2017-03-08 02:17:16 -05:00
|
|
|
lbry._fileInfoSubscribeCallbacks[outpoint][subscribeId] = callback;
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry._updateFileInfoSubscribers(outpoint);
|
2017-01-12 21:03:34 -05:00
|
|
|
return subscribeId;
|
|
|
|
}
|
|
|
|
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry.fileInfoUnsubscribe = function(outpoint, subscribeId) {
|
|
|
|
delete lbry._fileInfoSubscribeCallbacks[outpoint][subscribeId];
|
|
|
|
}
|
|
|
|
|
2017-04-20 20:31:52 -04:00
|
|
|
lbry._balanceUpdateInterval = null;
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry._updateBalanceSubscribers = function() {
|
|
|
|
lbry.get_balance().then(function(balance) {
|
2017-03-27 01:44:13 -04:00
|
|
|
for (let callback of Object.values(lbry._balanceSubscribeCallbacks)) {
|
2017-03-26 14:30:18 -04:00
|
|
|
callback(balance);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-04-20 20:31:52 -04:00
|
|
|
if (!lbry._balanceUpdateInterval && Object.keys(lbry._balanceSubscribeCallbacks).length) {
|
|
|
|
lbry._balanceUpdateInterval = setInterval(() => {
|
2017-03-26 14:30:18 -04:00
|
|
|
lbry._updateBalanceSubscribers();
|
|
|
|
}, lbry._balanceSubscribeInterval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.balanceSubscribe = function(callback) {
|
|
|
|
const subscribeId = ++lbry._subscribeIdCount;
|
|
|
|
lbry._balanceSubscribeCallbacks[subscribeId] = callback;
|
|
|
|
lbry._updateBalanceSubscribers();
|
|
|
|
return subscribeId;
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.balanceUnsubscribe = function(subscribeId) {
|
|
|
|
delete lbry._balanceSubscribeCallbacks[subscribeId];
|
2017-04-20 20:31:52 -04:00
|
|
|
if (lbry._balanceUpdateInterval && !Object.keys(lbry._balanceSubscribeCallbacks).length) {
|
|
|
|
clearInterval(lbry._balanceUpdateInterval)
|
|
|
|
}
|
2017-01-12 21:03:34 -05:00
|
|
|
}
|
2016-05-16 04:16:40 -04:00
|
|
|
|
2017-03-08 03:56:34 -05:00
|
|
|
lbry.showMenuIfNeeded = function() {
|
2017-03-08 18:14:43 -05:00
|
|
|
const showingMenu = sessionStorage.getItem('menuShown') || null;
|
|
|
|
const chosenMenu = lbry.getClientSetting('showDeveloperMenu') ? 'developer' : 'normal';
|
|
|
|
if (chosenMenu != showingMenu) {
|
|
|
|
menu.showMenubar(chosenMenu == 'developer');
|
2017-03-08 03:56:34 -05:00
|
|
|
}
|
2017-03-08 18:14:43 -05:00
|
|
|
sessionStorage.setItem('menuShown', chosenMenu);
|
2017-03-08 03:56:34 -05:00
|
|
|
};
|
|
|
|
|
2017-04-27 02:52:14 -04:00
|
|
|
lbry.getVersionInfo = function() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
ipcRenderer.once('version-info-received', (event, versionInfo) => { resolve(versionInfo) });
|
|
|
|
ipcRenderer.send('version-info-requested');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-14 23:05:07 -04:00
|
|
|
/**
|
|
|
|
* Wrappers for API methods to simulate missing or future behavior. Unlike the old-style stubs,
|
|
|
|
* these are designed to be transparent wrappers around the corresponding API methods.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns results from the file_list API method, plus dummy entries for pending publishes.
|
|
|
|
* (If a real publish with the same name is found, the pending publish will be ignored and removed.)
|
|
|
|
*/
|
|
|
|
lbry.file_list = function(params={}) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2017-04-12 19:24:04 -04:00
|
|
|
const {name, channel_name, outpoint} = params;
|
2017-03-14 23:05:07 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* If we're searching by outpoint, check first to see if there's a matching pending publish.
|
|
|
|
* Pending publishes use their own faux outpoints that are always unique, so we don't need
|
|
|
|
* to check if there's a real file.
|
|
|
|
*/
|
2017-04-12 19:24:04 -04:00
|
|
|
if (outpoint) {
|
2017-03-14 23:05:07 -04:00
|
|
|
const pendingPublish = getPendingPublish({outpoint});
|
|
|
|
if (pendingPublish) {
|
|
|
|
resolve([pendingPublishToDummyFileInfo(pendingPublish)]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.call('file_list', params, (fileInfos) => {
|
2017-04-12 19:24:04 -04:00
|
|
|
removePendingPublishIfNeeded({name, channel_name, outpoint});
|
2017-03-14 23:05:07 -04:00
|
|
|
|
|
|
|
const dummyFileInfos = getPendingPublishes().map(pendingPublishToDummyFileInfo);
|
|
|
|
resolve([...fileInfos, ...dummyFileInfos]);
|
|
|
|
}, reject, reject);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
lbry.claim_list_mine = function(params={}) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
lbry.call('claim_list_mine', params, (claims) => {
|
2017-04-12 19:24:04 -04:00
|
|
|
for (let {name, channel_name, txid, nout} of claims) {
|
|
|
|
removePendingPublishIfNeeded({name, channel_name, outpoint: txid + ':' + nout});
|
|
|
|
}
|
2017-03-14 23:05:07 -04:00
|
|
|
|
2017-04-12 19:24:04 -04:00
|
|
|
const dummyClaims = getPendingPublishes().map(pendingPublishToDummyClaim);
|
2017-03-14 23:05:07 -04:00
|
|
|
resolve([...claims, ...dummyClaims]);
|
2017-04-12 19:24:04 -04:00
|
|
|
}, reject, reject)
|
2017-03-14 23:05:07 -04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-05-03 23:44:08 -04:00
|
|
|
const claimCacheKey = 'resolve_claim_cache';
|
|
|
|
lbry._claimCache = getLocal(claimCacheKey, {});
|
2017-04-13 14:52:26 -04:00
|
|
|
lbry.resolve = function(params={}) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
if (!params.uri) {
|
|
|
|
throw "Resolve has hacked cache on top of it that requires a URI"
|
|
|
|
}
|
2017-05-03 23:44:08 -04:00
|
|
|
if (params.uri && lbry._claimCache[params.uri] !== undefined) {
|
|
|
|
resolve(lbry._claimCache[params.uri]);
|
2017-04-13 14:52:26 -04:00
|
|
|
} else {
|
|
|
|
lbry.call('resolve', params, function(data) {
|
2017-05-03 23:44:08 -04:00
|
|
|
lbry._claimCache[params.uri] = data;
|
|
|
|
setLocal(claimCacheKey, lbry._claimCache)
|
2017-04-13 14:52:26 -04:00
|
|
|
resolve(data)
|
|
|
|
}, reject)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-04-17 12:18:33 -04:00
|
|
|
// Adds caching.
|
2017-05-03 23:44:08 -04:00
|
|
|
lbry._settingsPromise = null;
|
2017-04-17 12:18:33 -04:00
|
|
|
lbry.settings_get = function(params={}) {
|
2017-05-03 23:44:08 -04:00
|
|
|
if (params.allow_cached && lbry._settingsPromise) {
|
|
|
|
return lbry._settingsPromise;
|
|
|
|
}
|
|
|
|
lbry._settingsPromise = new Promise((resolve, reject) => {
|
2017-04-17 12:18:33 -04:00
|
|
|
lbry.call('settings_get', {}, (settings) => {
|
|
|
|
setSession('settings', settings);
|
|
|
|
resolve(settings);
|
2017-05-03 23:44:08 -04:00
|
|
|
}, reject);
|
2017-04-17 12:18:33 -04:00
|
|
|
});
|
2017-05-03 23:44:08 -04:00
|
|
|
return lbry._settingsPromise;
|
2017-04-17 12:18:33 -04:00
|
|
|
}
|
|
|
|
|
2017-04-12 16:23:20 -04:00
|
|
|
// lbry.get = function(params={}) {
|
|
|
|
// return function(params={}) {
|
|
|
|
// return new Promise((resolve, reject) => {
|
2017-05-03 13:54:36 -04:00
|
|
|
// jsonrpc.call(lbry.daemonConnectionString, "get", params, resolve, reject, reject);
|
2017-04-12 16:23:20 -04:00
|
|
|
// });
|
|
|
|
// };
|
|
|
|
// }
|
|
|
|
|
2017-03-08 01:23:07 -05:00
|
|
|
lbry = new Proxy(lbry, {
|
|
|
|
get: function(target, name) {
|
|
|
|
if (name in target) {
|
|
|
|
return target[name];
|
|
|
|
}
|
|
|
|
|
|
|
|
return function(params={}) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2017-05-03 13:54:36 -04:00
|
|
|
jsonrpc.call(lbry.daemonConnectionString, name, params, resolve, reject, reject);
|
2017-03-08 01:23:07 -05:00
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-01-17 05:06:39 -05:00
|
|
|
export default lbry;
|