syncthingController.js 70 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911
  1. angular.module('syncthing.core')
  2. .config(function($locationProvider) {
  3. $locationProvider.html5Mode({enabled: true, requireBase: false}).hashPrefix('!');
  4. })
  5. .controller('SyncthingController', function ($scope, $http, $location, LocaleService, Events, $filter, $q) {
  6. 'use strict';
  7. // private/helper definitions
  8. var prevDate = 0;
  9. var navigatingAway = false;
  10. var online = false;
  11. var restarting = false;
  12. function initController() {
  13. LocaleService.autoConfigLocale();
  14. setInterval($scope.refresh, 10000);
  15. Events.start();
  16. }
  17. // public/scope definitions
  18. $scope.completion = {};
  19. $scope.config = {};
  20. $scope.configInSync = true;
  21. $scope.connections = {};
  22. $scope.errors = [];
  23. $scope.model = {};
  24. $scope.myID = '';
  25. $scope.devices = [];
  26. $scope.deviceRejections = {};
  27. $scope.discoveryCache = {};
  28. $scope.folderRejections = {};
  29. $scope.protocolChanged = false;
  30. $scope.reportData = {};
  31. $scope.reportDataPreview = '';
  32. $scope.reportDataPreviewVersion = '';
  33. $scope.reportDataPreviewDiff = false;
  34. $scope.reportPreview = false;
  35. $scope.folders = {};
  36. $scope.seenError = '';
  37. $scope.upgradeInfo = null;
  38. $scope.deviceStats = {};
  39. $scope.folderStats = {};
  40. $scope.progress = {};
  41. $scope.version = {};
  42. $scope.needed = [];
  43. $scope.neededTotal = 0;
  44. $scope.neededCurrentPage = 1;
  45. $scope.neededPageSize = 10;
  46. $scope.failed = {};
  47. $scope.failedCurrentPage = 1;
  48. $scope.failedPageSize = 10;
  49. $scope.scanProgress = {};
  50. $scope.themes = [];
  51. $scope.globalChangeEvents = {};
  52. $scope.metricRates = false;
  53. $scope.folderPathErrors = {};
  54. try {
  55. $scope.metricRates = (window.localStorage["metricRates"] == "true");
  56. } catch (exception) { }
  57. $scope.folderDefaults = {
  58. selectedDevices: {},
  59. type: "readwrite",
  60. rescanIntervalS: 60,
  61. fsWatcherDelayS: 10,
  62. minDiskFree: {value: 1, unit: "%"},
  63. maxConflicts: 10,
  64. fsync: true,
  65. order: "random",
  66. fileVersioningSelector: "none",
  67. trashcanClean: 0,
  68. simpleKeep: 5,
  69. staggeredMaxAge: 365,
  70. staggeredCleanInterval: 3600,
  71. staggeredVersionsPath: "",
  72. externalCommand: "",
  73. autoNormalize: true,
  74. path: ""
  75. };
  76. $scope.localStateTotal = {
  77. bytes: 0,
  78. directories: 0,
  79. files: 0
  80. };
  81. $(window).bind('beforeunload', function () {
  82. navigatingAway = true;
  83. });
  84. $scope.$on("$locationChangeSuccess", function () {
  85. LocaleService.useLocale($location.search().lang);
  86. });
  87. $scope.needActions = {
  88. 'rm': 'Del',
  89. 'rmdir': 'Del (dir)',
  90. 'sync': 'Sync',
  91. 'touch': 'Update'
  92. };
  93. $scope.needIcons = {
  94. 'rm': 'trash-o',
  95. 'rmdir': 'trash-o',
  96. 'sync': 'arrow-circle-o-down',
  97. 'touch': 'asterisk'
  98. };
  99. $scope.$on(Events.ONLINE, function () {
  100. if (online && !restarting) {
  101. return;
  102. }
  103. console.log('UIOnline');
  104. refreshSystem();
  105. refreshDiscoveryCache();
  106. refreshConfig();
  107. refreshConnectionStats();
  108. refreshDeviceStats();
  109. refreshFolderStats();
  110. refreshGlobalChanges();
  111. refreshThemes();
  112. $http.get(urlbase + '/system/version').success(function (data) {
  113. if ($scope.version.version && $scope.version.version !== data.version) {
  114. // We already have a version response, but it differs from
  115. // the new one. Reload the full GUI in case it's changed.
  116. document.location.reload(true);
  117. }
  118. $scope.version = data;
  119. $scope.version.isDevelopmentVersion = data.version.indexOf('-')>0;
  120. }).error($scope.emitHTTPError);
  121. $http.get(urlbase + '/svc/report').success(function (data) {
  122. $scope.reportData = data;
  123. if ($scope.system && $scope.config.options.urSeen < $scope.system.urVersionMax) {
  124. // Usage reporting format has changed, prompt the user to re-accept.
  125. $('#ur').modal();
  126. }
  127. }).error($scope.emitHTTPError);
  128. $http.get(urlbase + '/system/upgrade').success(function (data) {
  129. $scope.upgradeInfo = data;
  130. }).error(function () {
  131. $scope.upgradeInfo = null;
  132. });
  133. online = true;
  134. restarting = false;
  135. $('#networkError').modal('hide');
  136. $('#restarting').modal('hide');
  137. $('#shutdown').modal('hide');
  138. });
  139. $scope.$on(Events.OFFLINE, function () {
  140. if (navigatingAway || !online) {
  141. return;
  142. }
  143. console.log('UIOffline');
  144. online = false;
  145. if (!restarting) {
  146. $('#networkError').modal();
  147. }
  148. });
  149. $scope.$on('HTTPError', function (event, arg) {
  150. // Emitted when a HTTP call fails. We use the status code to try
  151. // to figure out what's wrong.
  152. if (navigatingAway || !online) {
  153. return;
  154. }
  155. console.log('HTTPError', arg);
  156. online = false;
  157. if (!restarting) {
  158. if (arg.status === 0) {
  159. // A network error, not an HTTP error
  160. $scope.$emit(Events.OFFLINE);
  161. } else if (arg.status >= 400 && arg.status <= 599) {
  162. // A genuine HTTP error
  163. $('#networkError').modal('hide');
  164. $('#restarting').modal('hide');
  165. $('#shutdown').modal('hide');
  166. $('#httpError').modal();
  167. }
  168. }
  169. });
  170. $scope.$on(Events.STATE_CHANGED, function (event, arg) {
  171. var data = arg.data;
  172. if ($scope.model[data.folder]) {
  173. $scope.model[data.folder].state = data.to;
  174. $scope.model[data.folder].error = data.error;
  175. // If a folder has started syncing, then any old list of
  176. // errors is obsolete. We may get a new list of errors very
  177. // shortly though.
  178. if (data.to === 'syncing') {
  179. $scope.failed[data.folder] = [];
  180. }
  181. // If a folder has started scanning, then any scan progress is
  182. // also obsolete.
  183. if (data.to === 'scanning') {
  184. delete $scope.scanProgress[data.folder];
  185. }
  186. // If a folder finished scanning, then refresh folder stats
  187. // to update last scan time.
  188. if(data.from === 'scanning' && data.to === 'idle') {
  189. refreshFolderStats();
  190. }
  191. }
  192. });
  193. $scope.$on(Events.LOCAL_INDEX_UPDATED, function (event, arg) {
  194. refreshFolderStats();
  195. refreshGlobalChanges();
  196. });
  197. $scope.$on(Events.DEVICE_DISCONNECTED, function (event, arg) {
  198. $scope.connections[arg.data.id].connected = false;
  199. refreshDeviceStats();
  200. });
  201. $scope.$on(Events.DEVICE_CONNECTED, function (event, arg) {
  202. if (!$scope.connections[arg.data.id]) {
  203. $scope.connections[arg.data.id] = {
  204. inbps: 0,
  205. outbps: 0,
  206. inBytesTotal: 0,
  207. outBytesTotal: 0,
  208. type: arg.data.type,
  209. address: arg.data.addr
  210. };
  211. $scope.completion[arg.data.id] = {
  212. _total: 100,
  213. _needBytes: 0
  214. };
  215. }
  216. });
  217. $scope.$on('ConfigLoaded', function () {
  218. if ($scope.config.options.urAccepted === 0) {
  219. // If usage reporting has been neither accepted nor declined,
  220. // we want to ask the user to make a choice. But we don't want
  221. // to bug them during initial setup, so we set a cookie with
  222. // the time of the first visit. When that cookie is present
  223. // and the time is more than four hours ago, we ask the
  224. // question.
  225. var firstVisit = document.cookie.replace(/(?:(?:^|.*;\s*)firstVisit\s*\=\s*([^;]*).*$)|^.*$/, "$1");
  226. if (!firstVisit) {
  227. document.cookie = "firstVisit=" + Date.now() + ";max-age=" + 30 * 24 * 3600;
  228. } else {
  229. if (+firstVisit < Date.now() - 4 * 3600 * 1000) {
  230. $('#ur').modal();
  231. }
  232. }
  233. }
  234. });
  235. $scope.$on(Events.DEVICE_REJECTED, function (event, arg) {
  236. $scope.deviceRejections[arg.data.device] = arg;
  237. });
  238. $scope.$on(Events.FOLDER_REJECTED, function (event, arg) {
  239. $scope.folderRejections[arg.data.folder + "-" + arg.data.device] = arg;
  240. });
  241. $scope.$on(Events.CONFIG_SAVED, function (event, arg) {
  242. updateLocalConfig(arg.data);
  243. $http.get(urlbase + '/system/config/insync').success(function (data) {
  244. $scope.configInSync = data.configInSync;
  245. }).error($scope.emitHTTPError);
  246. });
  247. $scope.$on(Events.DOWNLOAD_PROGRESS, function (event, arg) {
  248. var stats = arg.data;
  249. var progress = {};
  250. for (var folder in stats) {
  251. progress[folder] = {};
  252. for (var file in stats[folder]) {
  253. var s = stats[folder][file];
  254. var reused = 100 * s.reused / s.total;
  255. var copiedFromOrigin = 100 * s.copiedFromOrigin / s.total;
  256. var copiedFromElsewhere = 100 * s.copiedFromElsewhere / s.total;
  257. var pulled = 100 * s.pulled / s.total;
  258. var pulling = 100 * s.pulling / s.total;
  259. // We try to round up pulling to at least a percent so that it would be at least a bit visible.
  260. if (pulling < 1 && pulled + copiedFromElsewhere + copiedFromOrigin + reused <= 99) {
  261. pulling = 1;
  262. }
  263. progress[folder][file] = {
  264. reused: reused,
  265. copiedFromOrigin: copiedFromOrigin,
  266. copiedFromElsewhere: copiedFromElsewhere,
  267. pulled: pulled,
  268. pulling: pulling,
  269. bytesTotal: s.bytesTotal,
  270. bytesDone: s.bytesDone,
  271. };
  272. }
  273. }
  274. for (var folder in $scope.progress) {
  275. if (!(folder in progress)) {
  276. if ($scope.neededFolder === folder) {
  277. refreshNeed(folder);
  278. }
  279. } else if ($scope.neededFolder === folder) {
  280. for (file in $scope.progress[folder]) {
  281. if (!(file in progress[folder])) {
  282. refreshNeed(folder);
  283. break;
  284. }
  285. }
  286. }
  287. }
  288. $scope.progress = progress;
  289. console.log("DownloadProgress", $scope.progress);
  290. });
  291. $scope.$on(Events.FOLDER_SUMMARY, function (event, arg) {
  292. var data = arg.data;
  293. $scope.model[data.folder] = data.summary;
  294. recalcLocalStateTotal();
  295. });
  296. $scope.$on(Events.FOLDER_COMPLETION, function (event, arg) {
  297. var data = arg.data;
  298. if (!$scope.completion[data.device]) {
  299. $scope.completion[data.device] = {};
  300. }
  301. $scope.completion[data.device][data.folder] = data;
  302. recalcCompletion(data.device);
  303. });
  304. $scope.$on(Events.FOLDER_ERRORS, function (event, arg) {
  305. var data = arg.data;
  306. $scope.failed[data.folder] = data.errors;
  307. });
  308. $scope.$on(Events.FOLDER_SCAN_PROGRESS, function (event, arg) {
  309. var data = arg.data;
  310. $scope.scanProgress[data.folder] = {
  311. current: data.current,
  312. total: data.total,
  313. rate: data.rate
  314. };
  315. console.log("FolderScanProgress", data);
  316. });
  317. $scope.emitHTTPError = function (data, status, headers, config) {
  318. $scope.$emit('HTTPError', {data: data, status: status, headers: headers, config: config});
  319. };
  320. var debouncedFuncs = {};
  321. function refreshFolder(folder) {
  322. var key = "refreshFolder" + folder;
  323. if (!debouncedFuncs[key]) {
  324. debouncedFuncs[key] = debounce(function () {
  325. $http.get(urlbase + '/db/status?folder=' + encodeURIComponent(folder)).success(function (data) {
  326. $scope.model[folder] = data;
  327. recalcLocalStateTotal();
  328. console.log("refreshFolder", folder, data);
  329. }).error($scope.emitHTTPError);
  330. }, 1000, true);
  331. }
  332. debouncedFuncs[key]();
  333. }
  334. function updateLocalConfig(config) {
  335. var hasConfig = !isEmptyObject($scope.config);
  336. $scope.config = config;
  337. $scope.config.options._listenAddressesStr = $scope.config.options.listenAddresses.join(', ');
  338. $scope.config.options._globalAnnounceServersStr = $scope.config.options.globalAnnounceServers.join(', ');
  339. $scope.config.options._urAcceptedStr = "" + $scope.config.options.urAccepted;
  340. $scope.devices = $scope.config.devices;
  341. $scope.devices.forEach(function (deviceCfg) {
  342. $scope.completion[deviceCfg.deviceID] = {
  343. _total: 100,
  344. _needBytes: 0
  345. };
  346. });
  347. $scope.devices.sort(deviceCompare);
  348. $scope.folders = folderMap($scope.config.folders);
  349. Object.keys($scope.folders).forEach(function (folder) {
  350. refreshFolder(folder);
  351. $scope.folders[folder].devices.forEach(function (deviceCfg) {
  352. refreshCompletion(deviceCfg.deviceID, folder);
  353. });
  354. });
  355. // If we're not listening on localhost, and there is no
  356. // authentication configured, and the magic setting to silence the
  357. // warning isn't set, then yell at the user.
  358. var guiCfg = $scope.config.gui;
  359. $scope.openNoAuth = guiCfg.address.substr(0, 4) !== "127."
  360. && guiCfg.address.substr(0, 6) !== "[::1]:"
  361. && (!guiCfg.user || !guiCfg.password)
  362. && !guiCfg.insecureAdminAccess;
  363. if (!hasConfig) {
  364. $scope.$emit('ConfigLoaded');
  365. }
  366. }
  367. function refreshSystem() {
  368. $http.get(urlbase + '/system/status').success(function (data) {
  369. $scope.myID = data.myID;
  370. $scope.system = data;
  371. if ($scope.reportDataPreviewVersion === '') {
  372. $scope.reportDataPreviewVersion = $scope.system.urVersionMax;
  373. }
  374. var listenersFailed = [];
  375. for (var address in data.connectionServiceStatus) {
  376. if (data.connectionServiceStatus[address].error) {
  377. listenersFailed.push(address + ": " + data.connectionServiceStatus[address].error);
  378. }
  379. }
  380. $scope.listenersFailed = listenersFailed;
  381. $scope.listenersTotal = Object.keys(data.connectionServiceStatus).length;
  382. $scope.discoveryTotal = data.discoveryMethods;
  383. var discoveryFailed = [];
  384. for (var disco in data.discoveryErrors) {
  385. if (data.discoveryErrors[disco]) {
  386. discoveryFailed.push(disco + ": " + data.discoveryErrors[disco]);
  387. }
  388. }
  389. $scope.discoveryFailed = discoveryFailed;
  390. console.log("refreshSystem", data);
  391. }).error($scope.emitHTTPError);
  392. }
  393. function refreshDiscoveryCache() {
  394. $http.get(urlbase + '/system/discovery').success(function (data) {
  395. for (var device in data) {
  396. for (var i = 0; i < data[device].addresses.length; i++) {
  397. // Relay addresses are URLs with
  398. // .../?foo=barlongstuff that we strip away here. We
  399. // remove the final slash as well for symmetry with
  400. // tcp://192.0.2.42:1234 type addresses.
  401. data[device].addresses[i] = data[device].addresses[i].replace(/\/\?.*/, '');
  402. }
  403. }
  404. $scope.discoveryCache = data;
  405. console.log("refreshDiscoveryCache", data);
  406. }).error($scope.emitHTTPError);
  407. }
  408. function recalcLocalStateTotal () {
  409. $scope.localStateTotal = {
  410. bytes: 0,
  411. directories: 0,
  412. files: 0
  413. };
  414. for (var f in $scope.model) {
  415. $scope.localStateTotal.bytes += $scope.model[f].localBytes;
  416. $scope.localStateTotal.files += $scope.model[f].localFiles;
  417. $scope.localStateTotal.directories += $scope.model[f].localDirectories;
  418. }
  419. }
  420. function recalcCompletion(device) {
  421. var total = 0, needed = 0, deletes = 0;
  422. for (var folder in $scope.completion[device]) {
  423. if (folder === "_total" || folder === '_needBytes') {
  424. continue;
  425. }
  426. total += $scope.completion[device][folder].globalBytes;
  427. needed += $scope.completion[device][folder].needBytes;
  428. deletes += $scope.completion[device][folder].needDeletes;
  429. }
  430. if (total == 0) {
  431. $scope.completion[device]._total = 100;
  432. $scope.completion[device]._needBytes = 0;
  433. } else {
  434. $scope.completion[device]._total = Math.floor(100 * (1 - needed / total));
  435. $scope.completion[device]._needBytes = needed
  436. }
  437. if (needed == 0 && deletes > 0) {
  438. // We don't need any data, but we have deletes that we need
  439. // to do. Drop down the completion percentage to indicate
  440. // that we have stuff to do.
  441. $scope.completion[device]._total = 95;
  442. $scope.completion[device]._needBytes = 0;
  443. }
  444. console.log("recalcCompletion", device, $scope.completion[device]);
  445. }
  446. function refreshCompletion(device, folder) {
  447. if (device === $scope.myID) {
  448. return;
  449. }
  450. $http.get(urlbase + '/db/completion?device=' + device + '&folder=' + encodeURIComponent(folder)).success(function (data) {
  451. if (!$scope.completion[device]) {
  452. $scope.completion[device] = {};
  453. }
  454. $scope.completion[device][folder] = data;
  455. recalcCompletion(device);
  456. }).error($scope.emitHTTPError);
  457. }
  458. function refreshConnectionStats() {
  459. $http.get(urlbase + '/system/connections').success(function (data) {
  460. var now = Date.now(),
  461. td = (now - prevDate) / 1000,
  462. id;
  463. prevDate = now;
  464. try {
  465. data.total.inbps = Math.max(0, (data.total.inBytesTotal - $scope.connectionsTotal.inBytesTotal) / td);
  466. data.total.outbps = Math.max(0, (data.total.outBytesTotal - $scope.connectionsTotal.outBytesTotal) / td);
  467. } catch (e) {
  468. data.total.inbps = 0;
  469. data.total.outbps = 0;
  470. }
  471. $scope.connectionsTotal = data.total;
  472. data = data.connections;
  473. for (id in data) {
  474. if (!data.hasOwnProperty(id)) {
  475. continue;
  476. }
  477. try {
  478. data[id].inbps = Math.max(0, (data[id].inBytesTotal - $scope.connections[id].inBytesTotal) / td);
  479. data[id].outbps = Math.max(0, (data[id].outBytesTotal - $scope.connections[id].outBytesTotal) / td);
  480. } catch (e) {
  481. data[id].inbps = 0;
  482. data[id].outbps = 0;
  483. }
  484. }
  485. $scope.connections = data;
  486. console.log("refreshConnections", data);
  487. }).error($scope.emitHTTPError);
  488. }
  489. function refreshErrors() {
  490. $http.get(urlbase + '/system/error').success(function (data) {
  491. $scope.errors = data.errors;
  492. console.log("refreshErrors", data);
  493. }).error($scope.emitHTTPError);
  494. }
  495. function refreshConfig() {
  496. $http.get(urlbase + '/system/config').success(function (data) {
  497. updateLocalConfig(data);
  498. console.log("refreshConfig", data);
  499. }).error($scope.emitHTTPError);
  500. $http.get(urlbase + '/system/config/insync').success(function (data) {
  501. $scope.configInSync = data.configInSync;
  502. }).error($scope.emitHTTPError);
  503. }
  504. function refreshNeed(folder) {
  505. var url = urlbase + "/db/need?folder=" + encodeURIComponent(folder);
  506. url += "&page=" + $scope.neededCurrentPage;
  507. url += "&perpage=" + $scope.neededPageSize;
  508. $http.get(url).success(function (data) {
  509. if ($scope.neededFolder === folder) {
  510. console.log("refreshNeed", folder, data);
  511. parseNeeded(data);
  512. }
  513. }).error($scope.emitHTTPError);
  514. }
  515. function needAction(file) {
  516. var fDelete = 4096;
  517. var fDirectory = 16384;
  518. if ((file.flags & (fDelete + fDirectory)) === fDelete + fDirectory) {
  519. return 'rmdir';
  520. } else if ((file.flags & fDelete) === fDelete) {
  521. return 'rm';
  522. } else if ((file.flags & fDirectory) === fDirectory) {
  523. return 'touch';
  524. } else {
  525. return 'sync';
  526. }
  527. }
  528. function parseNeeded(data) {
  529. var merged = [];
  530. data.progress.forEach(function (item) {
  531. item.type = "progress";
  532. item.action = needAction(item);
  533. merged.push(item);
  534. });
  535. data.queued.forEach(function (item) {
  536. item.type = "queued";
  537. item.action = needAction(item);
  538. merged.push(item);
  539. });
  540. data.rest.forEach(function (item) {
  541. item.type = "rest";
  542. item.action = needAction(item);
  543. merged.push(item);
  544. });
  545. $scope.needed = merged;
  546. $scope.neededTotal = data.total;
  547. }
  548. function pathJoin(base, name) {
  549. base = expandTilde(base);
  550. if (base[base.length - 1] !== $scope.system.pathSeparator) {
  551. return base + $scope.system.pathSeparator + name;
  552. }
  553. return base + name;
  554. }
  555. function expandTilde(path) {
  556. if (path && path.trim().charAt(0) === '~') {
  557. return $scope.system.tilde + path.trim().substring(1);
  558. }
  559. return path;
  560. }
  561. function shouldSetDefaultFolderPath() {
  562. return $scope.config.options && $scope.config.options.defaultFolderPath && !$scope.editingExisting && $scope.folderEditor.folderPath.$pristine
  563. }
  564. $scope.neededPageChanged = function (page) {
  565. $scope.neededCurrentPage = page;
  566. refreshNeed($scope.neededFolder);
  567. };
  568. $scope.neededChangePageSize = function (perpage) {
  569. $scope.neededPageSize = perpage;
  570. refreshNeed($scope.neededFolder);
  571. };
  572. $scope.failedPageChanged = function (page) {
  573. $scope.failedCurrentPage = page;
  574. };
  575. $scope.failedChangePageSize = function (perpage) {
  576. $scope.failedPageSize = perpage;
  577. };
  578. var refreshDeviceStats = debounce(function () {
  579. $http.get(urlbase + "/stats/device").success(function (data) {
  580. $scope.deviceStats = data;
  581. for (var device in $scope.deviceStats) {
  582. $scope.deviceStats[device].lastSeen = new Date($scope.deviceStats[device].lastSeen);
  583. $scope.deviceStats[device].lastSeenDays = (new Date() - $scope.deviceStats[device].lastSeen) / 1000 / 86400;
  584. }
  585. console.log("refreshDeviceStats", data);
  586. }).error($scope.emitHTTPError);
  587. }, 2500);
  588. var refreshFolderStats = debounce(function () {
  589. $http.get(urlbase + "/stats/folder").success(function (data) {
  590. $scope.folderStats = data;
  591. for (var folder in $scope.folderStats) {
  592. if ($scope.folderStats[folder].lastFile) {
  593. $scope.folderStats[folder].lastFile.at = new Date($scope.folderStats[folder].lastFile.at);
  594. }
  595. $scope.folderStats[folder].lastScan = new Date($scope.folderStats[folder].lastScan);
  596. $scope.folderStats[folder].lastScanDays = (new Date() - $scope.folderStats[folder].lastScan) / 1000 / 86400;
  597. }
  598. console.log("refreshfolderStats", data);
  599. }).error($scope.emitHTTPError);
  600. }, 2500);
  601. var refreshThemes = debounce(function () {
  602. $http.get("themes.json").success(function (data) { // no urlbase here as this is served by the asset handler
  603. $scope.themes = data.themes;
  604. }).error($scope.emitHTTPError);
  605. }, 2500);
  606. var refreshGlobalChanges = debounce(function () {
  607. $http.get(urlbase + "/events/disk?limit=25").success(function (data) {
  608. data = data.reverse();
  609. $scope.globalChangeEvents = data;
  610. console.log("refreshGlobalChanges", data);
  611. }).error($scope.emitHTTPError);
  612. }, 2500);
  613. $scope.refresh = function () {
  614. refreshSystem();
  615. refreshDiscoveryCache();
  616. refreshConnectionStats();
  617. refreshErrors();
  618. };
  619. $scope.folderStatus = function (folderCfg) {
  620. if (typeof $scope.model[folderCfg.id] === 'undefined') {
  621. return 'unknown';
  622. }
  623. if (folderCfg.paused) {
  624. return 'paused';
  625. }
  626. // after restart syncthing process state may be empty
  627. if (!$scope.model[folderCfg.id].state) {
  628. return 'unknown';
  629. }
  630. if ($scope.model[folderCfg.id].invalid) {
  631. return 'stopped';
  632. }
  633. var state = '' + $scope.model[folderCfg.id].state;
  634. if (state === 'error') {
  635. return 'stopped'; // legacy, the state is called "stopped" in the GUI
  636. }
  637. if (state === 'idle' && $scope.neededItems(folderCfg.id) > 0) {
  638. return 'outofsync';
  639. }
  640. if (state === 'scanning') {
  641. return state;
  642. }
  643. if (folderCfg.devices.length <= 1) {
  644. return 'unshared';
  645. }
  646. return state;
  647. };
  648. $scope.folderClass = function (folderCfg) {
  649. var status = $scope.folderStatus(folderCfg);
  650. if (status === 'idle') {
  651. return 'success';
  652. }
  653. if (status == 'paused') {
  654. return 'default';
  655. }
  656. if (status === 'syncing' || status === 'scanning') {
  657. return 'primary';
  658. }
  659. if (status === 'unknown') {
  660. return 'info';
  661. }
  662. if (status === 'stopped' || status === 'outofsync' || status === 'error') {
  663. return 'danger';
  664. }
  665. if (status === 'unshared') {
  666. return 'warning';
  667. }
  668. return 'info';
  669. };
  670. $scope.neededItems = function (folderID) {
  671. if (!$scope.model[folderID]) {
  672. return 0
  673. }
  674. return $scope.model[folderID].needFiles + $scope.model[folderID].needDirectories +
  675. $scope.model[folderID].needSymlinks + $scope.model[folderID].needDeletes;
  676. };
  677. $scope.syncPercentage = function (folder) {
  678. if (typeof $scope.model[folder] === 'undefined') {
  679. return 100;
  680. }
  681. if ($scope.model[folder].globalBytes === 0) {
  682. return 100;
  683. }
  684. var pct = 100 * $scope.model[folder].inSyncBytes / $scope.model[folder].globalBytes;
  685. return Math.floor(pct);
  686. };
  687. $scope.syncRemaining = function (folder) {
  688. // Remaining sync bytes
  689. if (typeof $scope.model[folder] === 'undefined') {
  690. return 0;
  691. }
  692. if ($scope.model[folder].globalBytes === 0) {
  693. return 0;
  694. }
  695. var bytes = $scope.model[folder].globalBytes - $scope.model[folder].inSyncBytes;
  696. if (isNaN(bytes) || bytes < 0) {
  697. return 0;
  698. }
  699. return bytes;
  700. };
  701. $scope.scanPercentage = function (folder) {
  702. if (!$scope.scanProgress[folder]) {
  703. return undefined;
  704. }
  705. var pct = 100 * $scope.scanProgress[folder].current / $scope.scanProgress[folder].total;
  706. return Math.floor(pct);
  707. };
  708. $scope.scanRate = function (folder) {
  709. if (!$scope.scanProgress[folder]) {
  710. return 0;
  711. }
  712. return $scope.scanProgress[folder].rate;
  713. };
  714. $scope.scanRemaining = function (folder) {
  715. // Formats the remaining scan time as a string. Includes days and
  716. // hours only when relevant, resulting in time stamps like:
  717. // 00m 40s
  718. // 32m 40s
  719. // 2h 32m
  720. // 4d 2h
  721. if (!$scope.scanProgress[folder]) {
  722. return "";
  723. }
  724. // Calculate remaining bytes and seconds based on our current
  725. // rate.
  726. var remainingBytes = $scope.scanProgress[folder].total - $scope.scanProgress[folder].current;
  727. var seconds = remainingBytes / $scope.scanProgress[folder].rate;
  728. // Round up to closest ten seconds to avoid flapping too much to
  729. // and fro.
  730. seconds = Math.ceil(seconds / 10) * 10;
  731. // Separate out the number of days.
  732. var days = 0;
  733. var res = [];
  734. if (seconds >= 86400) {
  735. days = Math.floor(seconds / 86400);
  736. res.push('' + days + 'd')
  737. seconds = seconds % 86400;
  738. }
  739. // Separate out the number of hours.
  740. var hours = 0;
  741. if (seconds > 3600) {
  742. hours = Math.floor(seconds / 3600);
  743. res.push('' + hours + 'h')
  744. seconds = seconds % 3600;
  745. }
  746. var d = new Date(1970, 0, 1).setSeconds(seconds);
  747. if (days === 0) {
  748. // Format minutes only if we're within a day of completion.
  749. var f = $filter('date')(d, "m'm'");
  750. res.push(f);
  751. }
  752. if (days === 0 && hours === 0) {
  753. // Format seconds only when we're within an hour of completion.
  754. var f = $filter('date')(d, "ss's'");
  755. res.push(f);
  756. }
  757. return res.join(' ');
  758. };
  759. $scope.deviceStatus = function (deviceCfg) {
  760. if ($scope.deviceFolders(deviceCfg).length === 0) {
  761. return 'unused';
  762. }
  763. if (typeof $scope.connections[deviceCfg.deviceID] === 'undefined') {
  764. return 'unknown';
  765. }
  766. if (deviceCfg.paused) {
  767. return 'paused';
  768. }
  769. if ($scope.connections[deviceCfg.deviceID].connected) {
  770. if ($scope.completion[deviceCfg.deviceID] && $scope.completion[deviceCfg.deviceID]._total === 100) {
  771. return 'insync';
  772. } else {
  773. return 'syncing';
  774. }
  775. }
  776. // Disconnected
  777. return 'disconnected';
  778. };
  779. $scope.deviceClass = function (deviceCfg) {
  780. if ($scope.deviceFolders(deviceCfg).length === 0) {
  781. // Unused
  782. return 'warning';
  783. }
  784. if (typeof $scope.connections[deviceCfg.deviceID] === 'undefined') {
  785. return 'info';
  786. }
  787. if (deviceCfg.paused) {
  788. return 'default';
  789. }
  790. if ($scope.connections[deviceCfg.deviceID].connected) {
  791. if ($scope.completion[deviceCfg.deviceID] && $scope.completion[deviceCfg.deviceID]._total === 100) {
  792. return 'success';
  793. } else {
  794. return 'primary';
  795. }
  796. }
  797. // Disconnected
  798. return 'info';
  799. };
  800. $scope.syncthingStatus = function () {
  801. var syncCount = 0;
  802. var notifyCount = 0;
  803. var pauseCount = 0;
  804. // loop through all folders
  805. var folderListCache = $scope.folderList();
  806. for (var i = 0; i < folderListCache.length; i++) {
  807. var status = $scope.folderStatus(folderListCache[i]);
  808. switch (status) {
  809. case 'syncing':
  810. syncCount++;
  811. break;
  812. case 'stopped':
  813. case 'unknown':
  814. case 'outofsync':
  815. case 'error':
  816. notifyCount++;
  817. break;
  818. }
  819. }
  820. // loop through all devices
  821. var deviceCount = $scope.devices.length;
  822. for (var i = 0; i < $scope.devices.length; i++) {
  823. var status = $scope.deviceStatus({
  824. deviceID:$scope.devices[i].deviceID
  825. });
  826. switch (status) {
  827. case 'unknown':
  828. notifyCount++;
  829. break;
  830. case 'paused':
  831. pauseCount++;
  832. break;
  833. case 'unused':
  834. deviceCount--;
  835. break;
  836. }
  837. }
  838. // enumerate notifications
  839. if ($scope.openNoAuth || !$scope.configInSync || Object.keys($scope.deviceRejections).length > 0 || Object.keys($scope.folderRejections).length > 0 || $scope.errorList().length > 0 || !online) {
  840. notifyCount++;
  841. }
  842. // at least one folder is syncing
  843. if (syncCount > 0) {
  844. return 'sync';
  845. }
  846. // a device is unknown or a folder is stopped/unknown/outofsync/error or some other notification is open or gui offline
  847. if (notifyCount > 0) {
  848. return 'notify';
  849. }
  850. // all used devices are paused except (this) one
  851. if (pauseCount === deviceCount-1) {
  852. return 'pause';
  853. }
  854. return 'default';
  855. };
  856. $scope.deviceAddr = function (deviceCfg) {
  857. var conn = $scope.connections[deviceCfg.deviceID];
  858. if (conn && conn.connected) {
  859. return conn.address;
  860. }
  861. return '?';
  862. };
  863. $scope.deviceCompletion = function (deviceCfg) {
  864. var conn = $scope.connections[deviceCfg.deviceID];
  865. if (conn) {
  866. return conn.completion + '%';
  867. }
  868. return '';
  869. };
  870. $scope.friendlyNameFromShort = function (shortID) {
  871. var matches = $scope.devices.filter(function (n) {
  872. return n.deviceID.substr(0, 7) === shortID;
  873. });
  874. if (matches.length !== 1) {
  875. return shortID;
  876. }
  877. return matches[0].name;
  878. };
  879. $scope.findDevice = function (deviceID) {
  880. var matches = $scope.devices.filter(function (n) {
  881. return n.deviceID === deviceID;
  882. });
  883. if (matches.length !== 1) {
  884. return undefined;
  885. }
  886. return matches[0];
  887. };
  888. $scope.deviceName = function (deviceCfg) {
  889. if (typeof deviceCfg === 'undefined' || typeof deviceCfg.deviceID === 'undefined') {
  890. return "";
  891. }
  892. if (deviceCfg.name) {
  893. return deviceCfg.name;
  894. }
  895. return deviceCfg.deviceID.substr(0, 6);
  896. };
  897. $scope.thisDeviceName = function () {
  898. var device = $scope.thisDevice();
  899. if (typeof device === 'undefined') {
  900. return "(unknown device)";
  901. }
  902. if (device.name) {
  903. return device.name;
  904. }
  905. return device.deviceID.substr(0, 6);
  906. };
  907. $scope.setDevicePause = function (device, pause) {
  908. $scope.devices.forEach(function (cfg) {
  909. if (cfg.deviceID == device) {
  910. cfg.paused = pause;
  911. }
  912. });
  913. $scope.config.devices = $scope.devices;
  914. $scope.saveConfig();
  915. };
  916. $scope.setFolderPause = function (folder, pause) {
  917. var cfg = $scope.folders[folder];
  918. if (cfg) {
  919. cfg.paused = pause;
  920. $scope.config.folders = folderList($scope.folders);
  921. $scope.saveConfig();
  922. }
  923. };
  924. $scope.showDiscoveryFailures = function () {
  925. $('#discovery-failures').modal();
  926. };
  927. $scope.editSettings = function () {
  928. // Make a working copy
  929. $scope.tmpOptions = angular.copy($scope.config.options);
  930. $scope.tmpOptions.deviceName = $scope.thisDevice().name;
  931. $scope.tmpOptions.upgrades = "none";
  932. if ($scope.tmpOptions.autoUpgradeIntervalH > 0) {
  933. $scope.tmpOptions.upgrades = "stable";
  934. }
  935. if ($scope.tmpOptions.upgradeToPreReleases) {
  936. $scope.tmpOptions.upgrades = "candidate";
  937. }
  938. $scope.tmpGUI = angular.copy($scope.config.gui);
  939. $('#settings').modal();
  940. };
  941. $scope.saveConfig = function (cb) {
  942. var cfg = JSON.stringify($scope.config);
  943. var opts = {
  944. headers: {
  945. 'Content-Type': 'application/json'
  946. }
  947. };
  948. $http.post(urlbase + '/system/config', cfg, opts).success(function () {
  949. $http.get(urlbase + '/system/config/insync').success(function (data) {
  950. $scope.configInSync = data.configInSync;
  951. if (cb) {
  952. cb();
  953. }
  954. });
  955. }).error($scope.emitHTTPError);
  956. };
  957. $scope.urVersions = function() {
  958. var result = [];
  959. if ($scope.system) {
  960. for (var i = $scope.system.urVersionMax; i >= 2; i--) {
  961. result.push("" + i);
  962. }
  963. }
  964. return result;
  965. };
  966. $scope.saveSettings = function () {
  967. // Make sure something changed
  968. var changed = !angular.equals($scope.config.options, $scope.tmpOptions) || !angular.equals($scope.config.gui, $scope.tmpGUI);
  969. var themeChanged = $scope.config.gui.theme !== $scope.tmpGUI.theme;
  970. if (changed) {
  971. // Angular has issues with selects with numeric values, so we handle strings here.
  972. $scope.tmpOptions.urAccepted = parseInt($scope.tmpOptions._urAcceptedStr);
  973. // Check if auto-upgrade has been enabled or disabled. This
  974. // also has an effect on usage reporting, so do the check
  975. // for that later.
  976. if ($scope.tmpOptions.upgrades == "candidate") {
  977. $scope.tmpOptions.autoUpgradeIntervalH = $scope.tmpOptions.autoUpgradeIntervalH || 12;
  978. $scope.tmpOptions.upgradeToPreReleases = true;
  979. $scope.tmpOptions.urAccepted = $scope.system.urVersionMax;
  980. $scope.tmpOptions.urSeen = $scope.system.urVersionMax;
  981. } else if ($scope.tmpOptions.upgrades == "stable") {
  982. $scope.tmpOptions.autoUpgradeIntervalH = $scope.tmpOptions.autoUpgradeIntervalH || 12;
  983. $scope.tmpOptions.upgradeToPreReleases = false;
  984. } else {
  985. $scope.tmpOptions.autoUpgradeIntervalH = 0;
  986. }
  987. // Check if protocol will need to be changed on restart
  988. if ($scope.config.gui.useTLS !== $scope.tmpGUI.useTLS) {
  989. $scope.protocolChanged = true;
  990. }
  991. // Apply new settings locally
  992. $scope.thisDevice().name = $scope.tmpOptions.deviceName;
  993. $scope.config.options = angular.copy($scope.tmpOptions);
  994. $scope.config.gui = angular.copy($scope.tmpGUI);
  995. ['listenAddresses', 'globalAnnounceServers'].forEach(function (key) {
  996. $scope.config.options[key] = $scope.config.options["_" + key + "Str"].split(/[ ,]+/).map(function (x) {
  997. return x.trim();
  998. });
  999. });
  1000. $scope.saveConfig(function () {
  1001. if (themeChanged) {
  1002. document.location.reload(true);
  1003. }
  1004. });
  1005. }
  1006. $('#settings').modal("hide");
  1007. };
  1008. $scope.saveAdvanced = function () {
  1009. $scope.config = $scope.advancedConfig;
  1010. $scope.saveConfig();
  1011. $('#advanced').modal("hide");
  1012. };
  1013. $scope.restart = function () {
  1014. restarting = true;
  1015. $('#restarting').modal();
  1016. $http.post(urlbase + '/system/restart');
  1017. $scope.configInSync = true;
  1018. // Switch webpage protocol if needed
  1019. if ($scope.protocolChanged) {
  1020. var protocol = 'http';
  1021. if ($scope.config.gui.useTLS) {
  1022. protocol = 'https';
  1023. }
  1024. setTimeout(function () {
  1025. window.location.protocol = protocol;
  1026. }, 2500);
  1027. $scope.protocolChanged = false;
  1028. }
  1029. };
  1030. $scope.upgrade = function () {
  1031. restarting = true;
  1032. $('#majorUpgrade').modal('hide');
  1033. $('#upgrading').modal();
  1034. $http.post(urlbase + '/system/upgrade').success(function () {
  1035. $('#restarting').modal();
  1036. $('#upgrading').modal('hide');
  1037. }).error(function () {
  1038. $('#upgrading').modal('hide');
  1039. });
  1040. };
  1041. $scope.shutdown = function () {
  1042. restarting = true;
  1043. $http.post(urlbase + '/system/shutdown').success(function () {
  1044. $('#shutdown').modal();
  1045. }).error($scope.emitHTTPError);
  1046. $scope.configInSync = true;
  1047. };
  1048. $scope.editDevice = function (deviceCfg) {
  1049. $scope.currentDevice = $.extend({}, deviceCfg);
  1050. $scope.editingExisting = true;
  1051. $scope.willBeReintroducedBy = undefined;
  1052. if (deviceCfg.introducedBy) {
  1053. var introducerDevice = $scope.findDevice(deviceCfg.introducedBy);
  1054. if (introducerDevice && introducerDevice.introducer) {
  1055. $scope.willBeReintroducedBy = $scope.deviceName(introducerDevice);
  1056. }
  1057. }
  1058. $scope.currentDevice._addressesStr = deviceCfg.addresses.join(', ');
  1059. $scope.currentDevice.selectedFolders = {};
  1060. $scope.deviceFolders($scope.currentDevice).forEach(function (folder) {
  1061. $scope.currentDevice.selectedFolders[folder] = true;
  1062. });
  1063. $scope.deviceEditor.$setPristine();
  1064. $('#editDevice').modal();
  1065. };
  1066. $scope.addDevice = function (deviceID, name) {
  1067. return $http.get(urlbase + '/system/discovery')
  1068. .success(function (registry) {
  1069. $scope.discovery = [];
  1070. outer:
  1071. for (var id in registry) {
  1072. if ($scope.discovery.length === 5) {
  1073. break;
  1074. }
  1075. for (var i = 0; i < $scope.devices.length; i++) {
  1076. if ($scope.devices[i].deviceID === id) {
  1077. continue outer;
  1078. }
  1079. }
  1080. $scope.discovery.push(id);
  1081. }
  1082. })
  1083. .then(function () {
  1084. $scope.currentDevice = {
  1085. name: name,
  1086. deviceID: deviceID,
  1087. _addressesStr: 'dynamic',
  1088. compression: 'metadata',
  1089. introducer: false,
  1090. selectedFolders: {}
  1091. };
  1092. $scope.editingExisting = false;
  1093. $scope.deviceEditor.$setPristine();
  1094. $('#editDevice').modal();
  1095. });
  1096. };
  1097. $scope.deleteDevice = function () {
  1098. $('#editDevice').modal('hide');
  1099. if (!$scope.editingExisting) {
  1100. return;
  1101. }
  1102. $scope.devices = $scope.devices.filter(function (n) {
  1103. return n.deviceID !== $scope.currentDevice.deviceID;
  1104. });
  1105. $scope.config.devices = $scope.devices;
  1106. // In case we later added the device manually, remove the ignoral
  1107. // record.
  1108. $scope.config.ignoredDevices = $scope.config.ignoredDevices.filter(function (id) {
  1109. return id !== $scope.currentDevice.deviceID;
  1110. });
  1111. for (var id in $scope.folders) {
  1112. $scope.folders[id].devices = $scope.folders[id].devices.filter(function (n) {
  1113. return n.deviceID !== $scope.currentDevice.deviceID;
  1114. });
  1115. }
  1116. $scope.saveConfig();
  1117. };
  1118. $scope.saveDevice = function () {
  1119. $('#editDevice').modal('hide');
  1120. $scope.saveDeviceConfig($scope.currentDevice);
  1121. $scope.dismissDeviceRejection($scope.currentDevice.deviceID);
  1122. };
  1123. $scope.saveDeviceConfig = function (deviceCfg) {
  1124. deviceCfg.addresses = deviceCfg._addressesStr.split(',').map(function (x) {
  1125. return x.trim();
  1126. });
  1127. var done = false;
  1128. for (var i = 0; i < $scope.devices.length && !done; i++) {
  1129. if ($scope.devices[i].deviceID === deviceCfg.deviceID) {
  1130. $scope.devices[i] = deviceCfg;
  1131. done = true;
  1132. }
  1133. }
  1134. if (!done) {
  1135. $scope.devices.push(deviceCfg);
  1136. }
  1137. $scope.devices.sort(deviceCompare);
  1138. $scope.config.devices = $scope.devices;
  1139. // In case we are adding the device manually, remove the ignoral
  1140. // record.
  1141. $scope.config.ignoredDevices = $scope.config.ignoredDevices.filter(function (id) {
  1142. return id !== deviceCfg.deviceID;
  1143. });
  1144. for (var id in deviceCfg.selectedFolders) {
  1145. if (deviceCfg.selectedFolders[id]) {
  1146. var found = false;
  1147. for (i = 0; i < $scope.folders[id].devices.length; i++) {
  1148. if ($scope.folders[id].devices[i].deviceID === deviceCfg.deviceID) {
  1149. found = true;
  1150. break;
  1151. }
  1152. }
  1153. if (!found) {
  1154. $scope.folders[id].devices.push({
  1155. deviceID: deviceCfg.deviceID
  1156. });
  1157. }
  1158. } else {
  1159. $scope.folders[id].devices = $scope.folders[id].devices.filter(function (n) {
  1160. return n.deviceID !== deviceCfg.deviceID;
  1161. });
  1162. }
  1163. }
  1164. $scope.saveConfig();
  1165. };
  1166. $scope.dismissDeviceRejection = function (device) {
  1167. delete $scope.deviceRejections[device];
  1168. };
  1169. $scope.ignoreRejectedDevice = function (device) {
  1170. $scope.config.ignoredDevices.push(device);
  1171. $scope.saveConfig();
  1172. $scope.dismissDeviceRejection(device);
  1173. };
  1174. $scope.otherDevices = function () {
  1175. return $scope.devices.filter(function (n) {
  1176. return n.deviceID !== $scope.myID;
  1177. });
  1178. };
  1179. $scope.thisDevice = function () {
  1180. for (var i = 0; i < $scope.devices.length; i++) {
  1181. var n = $scope.devices[i];
  1182. if (n.deviceID === $scope.myID) {
  1183. return n;
  1184. }
  1185. }
  1186. };
  1187. $scope.allDevices = function () {
  1188. var devices = $scope.otherDevices();
  1189. devices.push($scope.thisDevice());
  1190. return devices;
  1191. };
  1192. $scope.errorList = function () {
  1193. if (!$scope.errors) {
  1194. return [];
  1195. }
  1196. return $scope.errors.filter(function (e) {
  1197. return e.when > $scope.seenError;
  1198. });
  1199. };
  1200. $scope.clearErrors = function () {
  1201. $scope.seenError = $scope.errors[$scope.errors.length - 1].when;
  1202. $http.post(urlbase + '/system/error/clear');
  1203. };
  1204. $scope.friendlyDevices = function (str) {
  1205. for (var i = 0; i < $scope.devices.length; i++) {
  1206. var cfg = $scope.devices[i];
  1207. str = str.replace(cfg.deviceID, $scope.deviceName(cfg));
  1208. }
  1209. return str;
  1210. };
  1211. $scope.folderList = function () {
  1212. return folderList($scope.folders);
  1213. };
  1214. $scope.directoryList = [];
  1215. $scope.$watch('currentFolder.path', function (newvalue) {
  1216. if (!newvalue) {
  1217. return;
  1218. }
  1219. $scope.currentFolder.path = expandTilde(newvalue);
  1220. $http.get(urlbase + '/system/browse', {
  1221. params: { current: newvalue }
  1222. }).success(function (data) {
  1223. $scope.directoryList = data;
  1224. }).error($scope.emitHTTPError);
  1225. });
  1226. $scope.$watch('currentFolder.label', function (newvalue) {
  1227. if (!newvalue || !shouldSetDefaultFolderPath()) {
  1228. return;
  1229. }
  1230. $scope.currentFolder.path = pathJoin($scope.config.options.defaultFolderPath, newvalue);
  1231. });
  1232. $scope.$watch('currentFolder.id', function (newvalue) {
  1233. if (!newvalue || !shouldSetDefaultFolderPath() || $scope.currentFolder.label) {
  1234. return;
  1235. }
  1236. $scope.currentFolder.path = pathJoin($scope.config.options.defaultFolderPath, newvalue);
  1237. });
  1238. $scope.loadFormIntoScope = function (form) {
  1239. console.log('loadFormIntoScope',form.$name);
  1240. switch (form.$name) {
  1241. case 'deviceEditor':
  1242. $scope.deviceEditor = form;
  1243. break;
  1244. case 'folderEditor':
  1245. $scope.folderEditor = form;
  1246. break;
  1247. }
  1248. };
  1249. $scope.globalChanges = function () {
  1250. $('#globalChanges').modal();
  1251. };
  1252. $scope.editFolderModal = function () {
  1253. $scope.folderPathErrors = {};
  1254. $scope.folderEditor.$setPristine();
  1255. $('#editIgnores textarea').val("");
  1256. $('#editFolder').modal();
  1257. };
  1258. $scope.editFolder = function (folderCfg) {
  1259. $scope.editingExisting = true;
  1260. $scope.currentFolder = angular.copy(folderCfg);
  1261. if ($scope.currentFolder.path.slice(-1) === $scope.system.pathSeparator) {
  1262. $scope.currentFolder.path = $scope.currentFolder.path.slice(0, -1);
  1263. }
  1264. $scope.currentFolder.selectedDevices = {};
  1265. $scope.currentFolder.devices.forEach(function (n) {
  1266. $scope.currentFolder.selectedDevices[n.deviceID] = true;
  1267. });
  1268. if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "trashcan") {
  1269. $scope.currentFolder.trashcanFileVersioning = true;
  1270. $scope.currentFolder.fileVersioningSelector = "trashcan";
  1271. $scope.currentFolder.trashcanClean = +$scope.currentFolder.versioning.params.cleanoutDays;
  1272. } else if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "simple") {
  1273. $scope.currentFolder.simpleFileVersioning = true;
  1274. $scope.currentFolder.fileVersioningSelector = "simple";
  1275. $scope.currentFolder.simpleKeep = +$scope.currentFolder.versioning.params.keep;
  1276. } else if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "staggered") {
  1277. $scope.currentFolder.staggeredFileVersioning = true;
  1278. $scope.currentFolder.fileVersioningSelector = "staggered";
  1279. $scope.currentFolder.staggeredMaxAge = Math.floor(+$scope.currentFolder.versioning.params.maxAge / 86400);
  1280. $scope.currentFolder.staggeredCleanInterval = +$scope.currentFolder.versioning.params.cleanInterval;
  1281. $scope.currentFolder.staggeredVersionsPath = $scope.currentFolder.versioning.params.versionsPath;
  1282. } else if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "external") {
  1283. $scope.currentFolder.externalFileVersioning = true;
  1284. $scope.currentFolder.fileVersioningSelector = "external";
  1285. $scope.currentFolder.externalCommand = $scope.currentFolder.versioning.params.command;
  1286. } else {
  1287. $scope.currentFolder.fileVersioningSelector = "none";
  1288. }
  1289. $scope.currentFolder.trashcanClean = $scope.currentFolder.trashcanClean || 0; // weeds out nulls and undefineds
  1290. $scope.currentFolder.simpleKeep = $scope.currentFolder.simpleKeep || 5;
  1291. $scope.currentFolder.staggeredCleanInterval = $scope.currentFolder.staggeredCleanInterval || 3600;
  1292. $scope.currentFolder.staggeredVersionsPath = $scope.currentFolder.staggeredVersionsPath || "";
  1293. // staggeredMaxAge can validly be zero, which we should not replace
  1294. // with the default value of 365. So only set the default if it's
  1295. // actually undefined.
  1296. if (typeof $scope.currentFolder.staggeredMaxAge === 'undefined') {
  1297. $scope.currentFolder.staggeredMaxAge = 365;
  1298. }
  1299. $scope.currentFolder.externalCommand = $scope.currentFolder.externalCommand || "";
  1300. $scope.editFolderModal();
  1301. };
  1302. $scope.addFolder = function () {
  1303. $http.get(urlbase + '/svc/random/string?length=10').success(function (data) {
  1304. $scope.editingExisting = false;
  1305. $scope.currentFolder = angular.copy($scope.folderDefaults);
  1306. $scope.currentFolder.id = (data.random.substr(0, 5) + '-' + data.random.substr(5, 5)).toLowerCase();
  1307. $scope.editFolderModal();
  1308. });
  1309. };
  1310. $scope.addFolderAndShare = function (folder, folderLabel, device) {
  1311. $scope.dismissFolderRejection(folder, device);
  1312. $scope.editingExisting = false;
  1313. $scope.currentFolder = angular.copy($scope.folderDefaults);
  1314. $scope.currentFolder.id = folder;
  1315. $scope.currentFolder.label = folderLabel;
  1316. $scope.currentFolder.viewFlags = {
  1317. importFromOtherDevice: true
  1318. };
  1319. $scope.currentFolder.selectedDevices[device] = true;
  1320. $scope.editFolderModal();
  1321. };
  1322. $scope.shareFolderWithDevice = function (folder, device) {
  1323. $scope.folders[folder].devices.push({
  1324. deviceID: device
  1325. });
  1326. $scope.config.folders = folderList($scope.folders);
  1327. $scope.saveConfig();
  1328. $scope.dismissFolderRejection(folder, device);
  1329. };
  1330. $scope.saveFolder = function () {
  1331. $('#editFolder').modal('hide');
  1332. var folderCfg = $scope.currentFolder;
  1333. folderCfg.devices = [];
  1334. folderCfg.selectedDevices[$scope.myID] = true;
  1335. for (var deviceID in folderCfg.selectedDevices) {
  1336. if (folderCfg.selectedDevices[deviceID] === true) {
  1337. folderCfg.devices.push({
  1338. deviceID: deviceID
  1339. });
  1340. }
  1341. }
  1342. delete folderCfg.selectedDevices;
  1343. if (folderCfg.fileVersioningSelector === "trashcan") {
  1344. folderCfg.versioning = {
  1345. 'Type': 'trashcan',
  1346. 'Params': {
  1347. 'cleanoutDays': '' + folderCfg.trashcanClean
  1348. }
  1349. };
  1350. delete folderCfg.trashcanFileVersioning;
  1351. delete folderCfg.trashcanClean;
  1352. } else if (folderCfg.fileVersioningSelector === "simple") {
  1353. folderCfg.versioning = {
  1354. 'Type': 'simple',
  1355. 'Params': {
  1356. 'keep': '' + folderCfg.simpleKeep
  1357. }
  1358. };
  1359. delete folderCfg.simpleFileVersioning;
  1360. delete folderCfg.simpleKeep;
  1361. } else if (folderCfg.fileVersioningSelector === "staggered") {
  1362. folderCfg.versioning = {
  1363. 'type': 'staggered',
  1364. 'params': {
  1365. 'maxAge': '' + (folderCfg.staggeredMaxAge * 86400),
  1366. 'cleanInterval': '' + folderCfg.staggeredCleanInterval,
  1367. 'versionsPath': '' + folderCfg.staggeredVersionsPath
  1368. }
  1369. };
  1370. delete folderCfg.staggeredFileVersioning;
  1371. delete folderCfg.staggeredMaxAge;
  1372. delete folderCfg.staggeredCleanInterval;
  1373. delete folderCfg.staggeredVersionsPath;
  1374. } else if (folderCfg.fileVersioningSelector === "external") {
  1375. folderCfg.versioning = {
  1376. 'Type': 'external',
  1377. 'Params': {
  1378. 'command': '' + folderCfg.externalCommand
  1379. }
  1380. };
  1381. delete folderCfg.externalFileVersioning;
  1382. delete folderCfg.externalCommand;
  1383. } else {
  1384. delete folderCfg.versioning;
  1385. }
  1386. var ignores = $('#editIgnores textarea').val().trim();
  1387. if (!$scope.editingExisting && ignores) {
  1388. folderCfg.paused = true;
  1389. };
  1390. $scope.folders[folderCfg.id] = folderCfg;
  1391. $scope.config.folders = folderList($scope.folders);
  1392. $scope.saveConfig(function () {
  1393. if (!$scope.editingExisting && ignores) {
  1394. $scope.saveIgnores(function () {
  1395. $scope.setFolderPause(folderCfg.id, false);
  1396. });
  1397. }
  1398. });
  1399. };
  1400. $scope.dismissFolderRejection = function (folder, device) {
  1401. delete $scope.folderRejections[folder + "-" + device];
  1402. };
  1403. $scope.ignoreRejectedFolder = function (folder, device) {
  1404. $scope.config.ignoredFolders.push(folder);
  1405. $scope.saveConfig();
  1406. $scope.dismissFolderRejection(folder, device);
  1407. };
  1408. $scope.sharesFolder = function (folderCfg) {
  1409. var names = [];
  1410. folderCfg.devices.forEach(function (device) {
  1411. if (device.deviceID !== $scope.myID) {
  1412. names.push($scope.deviceName($scope.findDevice(device.deviceID)));
  1413. }
  1414. });
  1415. names.sort();
  1416. return names.join(", ");
  1417. };
  1418. $scope.deviceFolders = function (deviceCfg) {
  1419. var folders = [];
  1420. for (var folderID in $scope.folders) {
  1421. var devices = $scope.folders[folderID].devices;
  1422. for (var i = 0; i < devices.length; i++) {
  1423. if (devices[i].deviceID === deviceCfg.deviceID) {
  1424. folders.push(folderID);
  1425. break;
  1426. }
  1427. }
  1428. }
  1429. folders.sort(folderCompare);
  1430. return folders;
  1431. };
  1432. $scope.folderLabel = function (folderID) {
  1433. var label = $scope.folders[folderID].label;
  1434. return label.length > 0 ? label : folderID;
  1435. }
  1436. $scope.deleteFolder = function (id) {
  1437. $('#editFolder').modal('hide');
  1438. if (!$scope.editingExisting) {
  1439. return;
  1440. }
  1441. delete $scope.folders[id];
  1442. delete $scope.model[id];
  1443. $scope.config.folders = folderList($scope.folders);
  1444. recalcLocalStateTotal();
  1445. $scope.saveConfig();
  1446. };
  1447. $scope.editIgnores = function () {
  1448. if (!$scope.editingExisting) {
  1449. return;
  1450. }
  1451. $('#editIgnoresButton').attr('disabled', 'disabled');
  1452. $http.get(urlbase + '/db/ignores?folder=' + encodeURIComponent($scope.currentFolder.id))
  1453. .success(function (data) {
  1454. data.ignore = data.ignore || [];
  1455. var textArea = $('#editIgnores textarea');
  1456. textArea.val(data.ignore.join('\n'));
  1457. $('#editIgnores').modal()
  1458. .one('shown.bs.modal', function () {
  1459. textArea.focus();
  1460. });
  1461. })
  1462. .then(function () {
  1463. $('#editIgnoresButton').removeAttr('disabled');
  1464. });
  1465. };
  1466. $scope.editIgnoresOnAddingFolder = function () {
  1467. if ($scope.editingExisting) {
  1468. return;
  1469. }
  1470. if ($scope.currentFolder.path.endsWith($scope.system.pathSeparator)) {
  1471. $scope.currentFolder.path = $scope.currentFolder.path.slice(0, -1);
  1472. };
  1473. $('#editIgnores').modal().one('shown.bs.modal', function () {
  1474. textArea.focus();
  1475. });
  1476. };
  1477. $scope.saveIgnores = function (cb) {
  1478. $http.post(urlbase + '/db/ignores?folder=' + encodeURIComponent($scope.currentFolder.id), {
  1479. ignore: $('#editIgnores textarea').val().split('\n')
  1480. }).success(function () {
  1481. if (cb) {
  1482. cb();
  1483. }
  1484. });
  1485. };
  1486. $scope.setAPIKey = function (cfg) {
  1487. $http.get(urlbase + '/svc/random/string?length=32').success(function (data) {
  1488. cfg.apiKey = data.random;
  1489. });
  1490. };
  1491. $scope.acceptUR = function () {
  1492. $scope.config.options.urAccepted = $scope.system.urVersionMax;
  1493. $scope.config.options.urSeen = $scope.system.urVersionMax;
  1494. $scope.saveConfig();
  1495. $('#ur').modal('hide');
  1496. };
  1497. $scope.declineUR = function () {
  1498. if ($scope.config.options.urAccepted === 0) {
  1499. $scope.config.options.urAccepted = -1;
  1500. }
  1501. $scope.config.options.urSeen = $scope.system.urVersionMax;
  1502. $scope.saveConfig();
  1503. $('#ur').modal('hide');
  1504. };
  1505. $scope.showNeed = function (folder) {
  1506. $scope.neededFolder = folder;
  1507. refreshNeed(folder);
  1508. $('#needed').modal().on('hidden.bs.modal', function () {
  1509. $scope.neededFolder = undefined;
  1510. $scope.needed = undefined;
  1511. $scope.neededTotal = 0;
  1512. $scope.neededCurrentPage = 1;
  1513. });
  1514. };
  1515. $scope.showFailed = function (folder) {
  1516. $scope.failedCurrent = $scope.failed[folder];
  1517. $scope.failedFolderPath = $scope.folders[folder].path;
  1518. if ($scope.failedFolderPath[$scope.failedFolderPath.length - 1] !== $scope.system.pathSeparator) {
  1519. $scope.failedFolderPath += $scope.system.pathSeparator;
  1520. }
  1521. $('#failed').modal().on('hidden.bs.modal', function () {
  1522. $scope.failedCurrent = undefined;
  1523. });
  1524. };
  1525. $scope.hasFailedFiles = function (folder) {
  1526. if (!$scope.failed[folder]) {
  1527. return false;
  1528. }
  1529. if ($scope.failed[folder].length === 0) {
  1530. return false;
  1531. }
  1532. return true;
  1533. };
  1534. $scope.override = function (folder) {
  1535. $http.post(urlbase + "/db/override?folder=" + encodeURIComponent(folder));
  1536. };
  1537. $scope.advanced = function () {
  1538. $scope.advancedConfig = angular.copy($scope.config);
  1539. $('#advanced').modal('show');
  1540. };
  1541. $scope.showReportPreview = function () {
  1542. $scope.reportPreview = true;
  1543. };
  1544. $scope.refreshReportDataPreview = function () {
  1545. $scope.reportDataPreview = '';
  1546. if (!$scope.reportDataPreviewVersion) {
  1547. return;
  1548. }
  1549. var version = parseInt($scope.reportDataPreviewVersion);
  1550. if ($scope.reportDataPreviewDiff && version > 2) {
  1551. $q.all([
  1552. $http.get(urlbase + '/svc/report?version=' + version),
  1553. $http.get(urlbase + '/svc/report?version=' + (version-1)),
  1554. ]).then(function (responses) {
  1555. var newReport = responses[0].data;
  1556. var oldReport = responses[1].data;
  1557. angular.forEach(oldReport, function(_, key) {
  1558. delete newReport[key];
  1559. });
  1560. $scope.reportDataPreview = newReport;
  1561. });
  1562. } else {
  1563. $http.get(urlbase + '/svc/report?version=' + version).success(function (data) {
  1564. $scope.reportDataPreview = data;
  1565. }).error($scope.emitHTTPError);
  1566. }
  1567. };
  1568. $scope.rescanAllFolders = function () {
  1569. $http.post(urlbase + "/db/scan");
  1570. };
  1571. $scope.rescanFolder = function (folder) {
  1572. $http.post(urlbase + "/db/scan?folder=" + encodeURIComponent(folder));
  1573. };
  1574. $scope.setAllFoldersPause = function(pause) {
  1575. var folderListCache = $scope.folderList();
  1576. for (var i = 0; i < folderListCache.length; i++) {
  1577. folderListCache[i].paused = pause;
  1578. }
  1579. $scope.config.folders = folderList(folderListCache);
  1580. $scope.saveConfig();
  1581. };
  1582. $scope.isAtleastOneFolderPausedStateSetTo = function(pause) {
  1583. var folderListCache = $scope.folderList();
  1584. for (var i = 0; i < folderListCache.length; i++) {
  1585. if (folderListCache[i].paused == pause) {
  1586. return true;
  1587. }
  1588. }
  1589. return false;
  1590. };
  1591. $scope.bumpFile = function (folder, file) {
  1592. var url = urlbase + "/db/prio?folder=" + encodeURIComponent(folder) + "&file=" + encodeURIComponent(file);
  1593. // In order to get the right view of data in the response.
  1594. url += "&page=" + $scope.neededCurrentPage;
  1595. url += "&perpage=" + $scope.neededPageSize;
  1596. $http.post(url).success(function (data) {
  1597. if ($scope.neededFolder === folder) {
  1598. console.log("bumpFile", folder, data);
  1599. parseNeeded(data);
  1600. }
  1601. }).error($scope.emitHTTPError);
  1602. };
  1603. $scope.versionString = function () {
  1604. if (!$scope.version.version) {
  1605. return '';
  1606. }
  1607. var os = {
  1608. 'darwin': 'Mac OS X',
  1609. 'dragonfly': 'DragonFly BSD',
  1610. 'freebsd': 'FreeBSD',
  1611. 'openbsd': 'OpenBSD',
  1612. 'netbsd': 'NetBSD',
  1613. 'linux': 'Linux',
  1614. 'windows': 'Windows',
  1615. 'solaris': 'Solaris'
  1616. }[$scope.version.os] || $scope.version.os;
  1617. var arch ={
  1618. '386': '32 bit',
  1619. 'amd64': '64 bit',
  1620. 'arm': 'ARM',
  1621. 'arm64': 'AArch64',
  1622. 'ppc64': 'PowerPC',
  1623. 'ppc64le': 'PowerPC (LE)'
  1624. }[$scope.version.arch] || $scope.version.arch;
  1625. return $scope.version.version + ', ' + os + ' (' + arch + ')';
  1626. };
  1627. $scope.inputTypeFor = function (key, value) {
  1628. if (key.substr(0, 1) === '_') {
  1629. return 'skip';
  1630. }
  1631. if (value === null) {
  1632. return 'null';
  1633. }
  1634. if (typeof value === 'number') {
  1635. return 'number';
  1636. }
  1637. if (typeof value === 'boolean') {
  1638. return 'checkbox';
  1639. }
  1640. if (value instanceof Array) {
  1641. return 'list';
  1642. }
  1643. if (typeof value === 'object') {
  1644. return 'skip';
  1645. }
  1646. return 'text';
  1647. };
  1648. $scope.themeName = function (theme) {
  1649. return theme.replace('-', ' ').replace(/(?:^|\s)\S/g, function (a) {
  1650. return a.toUpperCase();
  1651. });
  1652. };
  1653. $scope.modalLoaded = function () {
  1654. // once all modal elements have been processed
  1655. if ($('modal').length === 0) {
  1656. // pseudo main. called on all definitions assigned
  1657. initController();
  1658. }
  1659. }
  1660. $scope.toggleUnits = function () {
  1661. $scope.metricRates = !$scope.metricRates;
  1662. try {
  1663. window.localStorage["metricRates"] = $scope.metricRates;
  1664. } catch (exception) { }
  1665. }
  1666. });