1
0

syncthingController.js 100 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533
  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, $compile, $timeout, $rootScope, $translate) {
  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.discoveryCache = {};
  27. $scope.protocolChanged = false;
  28. $scope.reportData = {};
  29. $scope.reportDataPreview = '';
  30. $scope.reportPreview = false;
  31. $scope.folders = {};
  32. $scope.seenError = '';
  33. $scope.upgradeInfo = null;
  34. $scope.deviceStats = {};
  35. $scope.folderStats = {};
  36. $scope.progress = {};
  37. $scope.version = {};
  38. $scope.needed = {}
  39. $scope.neededFolder = '';
  40. $scope.failed = {};
  41. $scope.localChanged = {};
  42. $scope.scanProgress = {};
  43. $scope.themes = [];
  44. $scope.globalChangeEvents = {};
  45. $scope.metricRates = false;
  46. $scope.folderPathErrors = {};
  47. $scope.currentFolder = {};
  48. $scope.ignores = {
  49. text: '',
  50. error: null,
  51. disabled: false,
  52. };
  53. resetRemoteNeed();
  54. try {
  55. $scope.metricRates = (window.localStorage["metricRates"] == "true");
  56. } catch (exception) { }
  57. $scope.folderDefaults = {
  58. devices: [],
  59. sharedDevices: {},
  60. selectedDevices: {},
  61. unrelatedDevices: {},
  62. type: "sendreceive",
  63. rescanIntervalS: 3600,
  64. fsWatcherDelayS: 10,
  65. fsWatcherEnabled: true,
  66. minDiskFree: { value: 1, unit: "%" },
  67. maxConflicts: 10,
  68. fsync: true,
  69. order: "random",
  70. fileVersioningSelector: "none",
  71. trashcanClean: 0,
  72. versioningCleanupIntervalS: 3600,
  73. simpleKeep: 5,
  74. staggeredMaxAge: 365,
  75. staggeredCleanInterval: 3600,
  76. staggeredVersionsPath: "",
  77. externalCommand: "",
  78. autoNormalize: true,
  79. path: "",
  80. };
  81. $scope.localStateTotal = {
  82. bytes: 0,
  83. directories: 0,
  84. files: 0
  85. };
  86. $(window).bind('beforeunload', function () {
  87. navigatingAway = true;
  88. });
  89. $scope.$on("$locationChangeSuccess", function () {
  90. LocaleService.useLocale($location.search().lang);
  91. });
  92. $scope.needActions = {
  93. 'rm': 'Del',
  94. 'rmdir': 'Del (dir)',
  95. 'sync': 'Sync',
  96. 'touch': 'Update'
  97. };
  98. $scope.needIcons = {
  99. 'rm': 'far fa-fw fa-trash-alt',
  100. 'rmdir': 'far fa-fw fa-trash-alt',
  101. 'sync': 'far fa-fw fa-arrow-alt-circle-down',
  102. 'touch': 'fas fa-fw fa-asterisk'
  103. };
  104. $scope.$on(Events.ONLINE, function () {
  105. if (online && !restarting) {
  106. return;
  107. }
  108. console.log('UIOnline');
  109. refreshSystem();
  110. refreshDiscoveryCache();
  111. refreshConfig();
  112. refreshConnectionStats();
  113. refreshDeviceStats();
  114. refreshFolderStats();
  115. refreshGlobalChanges();
  116. refreshThemes();
  117. $http.get(urlbase + '/system/version').success(function (data) {
  118. console.log("version", data);
  119. if ($scope.version.version && $scope.version.version !== data.version) {
  120. // We already have a version response, but it differs from
  121. // the new one. Reload the full GUI in case it's changed.
  122. document.location.reload(true);
  123. }
  124. $scope.version = data;
  125. }).error($scope.emitHTTPError);
  126. $http.get(urlbase + '/svc/report').success(function (data) {
  127. $scope.reportData = data;
  128. if ($scope.system && $scope.config.options.urAccepted > -1 && $scope.config.options.urSeen < $scope.system.urVersionMax && $scope.config.options.urAccepted < $scope.system.urVersionMax) {
  129. // Usage reporting format has changed, prompt the user to re-accept.
  130. $('#ur').modal();
  131. }
  132. }).error($scope.emitHTTPError);
  133. $http.get(urlbase + '/system/upgrade').success(function (data) {
  134. $scope.upgradeInfo = data;
  135. }).error(function () {
  136. $scope.upgradeInfo = null;
  137. });
  138. online = true;
  139. restarting = false;
  140. $('#networkError').modal('hide');
  141. $('#restarting').modal('hide');
  142. $('#shutdown').modal('hide');
  143. });
  144. $scope.$on(Events.OFFLINE, function () {
  145. if (navigatingAway || !online) {
  146. return;
  147. }
  148. console.log('UIOffline');
  149. online = false;
  150. if (!restarting) {
  151. $('#networkError').modal();
  152. }
  153. });
  154. $scope.$on('HTTPError', function (event, arg) {
  155. // Emitted when a HTTP call fails. We use the status code to try
  156. // to figure out what's wrong.
  157. if (navigatingAway || !online) {
  158. return;
  159. }
  160. console.log('HTTPError', arg);
  161. online = false;
  162. if (!restarting) {
  163. if (arg.status === 0) {
  164. // A network error, not an HTTP error
  165. $scope.$emit(Events.OFFLINE);
  166. } else if (arg.status >= 400 && arg.status <= 599) {
  167. // A genuine HTTP error
  168. $('#networkError').modal('hide');
  169. $('#restarting').modal('hide');
  170. $('#shutdown').modal('hide');
  171. $('#httpError').modal();
  172. }
  173. }
  174. });
  175. $scope.$on(Events.STATE_CHANGED, function (event, arg) {
  176. var data = arg.data;
  177. if ($scope.model[data.folder]) {
  178. $scope.model[data.folder].state = data.to;
  179. $scope.model[data.folder].error = data.error;
  180. // If a folder has started scanning, then any scan progress is
  181. // also obsolete.
  182. if (data.to === 'scanning') {
  183. delete $scope.scanProgress[data.folder];
  184. }
  185. // If a folder finished scanning, then refresh folder stats
  186. // to update last scan time.
  187. if (data.from === 'scanning' && data.to === 'idle') {
  188. refreshFolderStats();
  189. }
  190. }
  191. });
  192. $scope.$on(Events.LOCAL_INDEX_UPDATED, function (event, arg) {
  193. refreshFolderStats();
  194. refreshGlobalChanges();
  195. });
  196. $scope.$on(Events.DEVICE_DISCONNECTED, function (event, arg) {
  197. $scope.connections[arg.data.id].connected = false;
  198. refreshDeviceStats();
  199. });
  200. $scope.$on(Events.DEVICE_CONNECTED, function (event, arg) {
  201. if (!$scope.connections[arg.data.id]) {
  202. $scope.connections[arg.data.id] = {
  203. inbps: 0,
  204. outbps: 0,
  205. inBytesTotal: 0,
  206. outBytesTotal: 0,
  207. type: arg.data.type,
  208. address: arg.data.addr
  209. };
  210. $scope.completion[arg.data.id] = {
  211. _total: 100,
  212. _needBytes: 0,
  213. _needItems: 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.CONFIG_SAVED, function (event, arg) {
  236. updateLocalConfig(arg.data);
  237. $http.get(urlbase + '/config/insync').success(function (data) {
  238. $scope.configInSync = data.configInSync;
  239. }).error($scope.emitHTTPError);
  240. });
  241. $scope.$on(Events.DOWNLOAD_PROGRESS, function (event, arg) {
  242. var stats = arg.data;
  243. var progress = {};
  244. for (var folder in stats) {
  245. progress[folder] = {};
  246. for (var file in stats[folder]) {
  247. var s = stats[folder][file];
  248. var reused = 100 * s.reused / s.total;
  249. var copiedFromOrigin = 100 * s.copiedFromOrigin / s.total;
  250. var copiedFromElsewhere = 100 * s.copiedFromElsewhere / s.total;
  251. var pulled = 100 * s.pulled / s.total;
  252. var pulling = 100 * s.pulling / s.total;
  253. // We try to round up pulling to at least a percent so that it would be at least a bit visible.
  254. if (pulling < 1 && pulled + copiedFromElsewhere + copiedFromOrigin + reused <= 99) {
  255. pulling = 1;
  256. }
  257. progress[folder][file] = {
  258. reused: reused,
  259. copiedFromOrigin: copiedFromOrigin,
  260. copiedFromElsewhere: copiedFromElsewhere,
  261. pulled: pulled,
  262. pulling: pulling,
  263. bytesTotal: s.bytesTotal,
  264. bytesDone: s.bytesDone,
  265. };
  266. }
  267. }
  268. for (var folder in $scope.progress) {
  269. if (!(folder in progress)) {
  270. if ($scope.neededFolder === folder) {
  271. $scope.refreshNeed($scope.needed.page, $scope.needed.perpage);
  272. }
  273. } else if ($scope.neededFolder === folder) {
  274. for (file in $scope.progress[folder]) {
  275. if (!(file in progress[folder])) {
  276. $scope.refreshNeed($scope.needed.page, $scope.needed.perpage);
  277. break;
  278. }
  279. }
  280. }
  281. }
  282. $scope.progress = progress;
  283. console.log("DownloadProgress", $scope.progress);
  284. });
  285. $scope.$on(Events.FOLDER_SUMMARY, function (event, arg) {
  286. var data = arg.data;
  287. $scope.model[data.folder] = data.summary;
  288. recalcLocalStateTotal();
  289. });
  290. $scope.$on(Events.FOLDER_COMPLETION, function (event, arg) {
  291. var data = arg.data;
  292. if (!$scope.completion[data.device]) {
  293. $scope.completion[data.device] = {};
  294. }
  295. $scope.completion[data.device][data.folder] = data;
  296. recalcCompletion(data.device);
  297. });
  298. $scope.$on(Events.FOLDER_ERRORS, function (event, arg) {
  299. $scope.model[arg.data.folder].errors = arg.data.errors.length;
  300. });
  301. $scope.$on(Events.FOLDER_SCAN_PROGRESS, function (event, arg) {
  302. var data = arg.data;
  303. $scope.scanProgress[data.folder] = {
  304. current: data.current,
  305. total: data.total,
  306. rate: data.rate
  307. };
  308. console.log("FolderScanProgress", data);
  309. });
  310. $scope.emitHTTPError = function (data, status, headers, config) {
  311. $scope.$emit('HTTPError', { data: data, status: status, headers: headers, config: config });
  312. };
  313. var debouncedFuncs = {};
  314. function refreshFolder(folder) {
  315. if ($scope.folders[folder].paused) {
  316. return;
  317. }
  318. var key = "refreshFolder" + folder;
  319. if (!debouncedFuncs[key]) {
  320. debouncedFuncs[key] = debounce(function () {
  321. $http.get(urlbase + '/db/status?folder=' + encodeURIComponent(folder)).success(function (data) {
  322. $scope.model[folder] = data;
  323. recalcLocalStateTotal();
  324. console.log("refreshFolder", folder, data);
  325. }).error($scope.emitHTTPError);
  326. }, 1000);
  327. }
  328. debouncedFuncs[key]();
  329. }
  330. function updateLocalConfig(config) {
  331. var hasConfig = !isEmptyObject($scope.config);
  332. $scope.config = config;
  333. $scope.config.options._listenAddressesStr = $scope.config.options.listenAddresses.join(', ');
  334. $scope.config.options._globalAnnounceServersStr = $scope.config.options.globalAnnounceServers.join(', ');
  335. $scope.config.options._urAcceptedStr = "" + $scope.config.options.urAccepted;
  336. $scope.devices = $scope.config.devices;
  337. $scope.devices.forEach(function (deviceCfg) {
  338. $scope.completion[deviceCfg.deviceID] = {
  339. _total: 100,
  340. _needBytes: 0,
  341. _needItems: 0
  342. };
  343. });
  344. $scope.devices.sort(deviceCompare);
  345. $scope.folders = folderMap($scope.config.folders);
  346. Object.keys($scope.folders).forEach(function (folder) {
  347. refreshFolder(folder);
  348. $scope.folders[folder].devices.forEach(function (deviceCfg) {
  349. refreshCompletion(deviceCfg.deviceID, folder);
  350. });
  351. });
  352. refreshNoAuthWarning();
  353. setDefaultTheme();
  354. if (!hasConfig) {
  355. $scope.$emit('ConfigLoaded');
  356. }
  357. }
  358. function refreshSystem() {
  359. $http.get(urlbase + '/system/status').success(function (data) {
  360. $scope.myID = data.myID;
  361. $scope.system = data;
  362. if ($scope.reportDataPreviewVersion === '') {
  363. $scope.reportDataPreviewVersion = $scope.system.urVersionMax;
  364. }
  365. var listenersFailed = [];
  366. for (var address in data.connectionServiceStatus) {
  367. if (data.connectionServiceStatus[address].error) {
  368. listenersFailed.push(address + ": " + data.connectionServiceStatus[address].error);
  369. }
  370. }
  371. $scope.listenersFailed = listenersFailed;
  372. $scope.listenersTotal = $scope.sizeOf(data.connectionServiceStatus);
  373. $scope.discoveryTotal = data.discoveryMethods;
  374. var discoveryFailed = [];
  375. for (var disco in data.discoveryErrors) {
  376. if (data.discoveryErrors[disco]) {
  377. discoveryFailed.push(disco + ": " + data.discoveryErrors[disco]);
  378. }
  379. }
  380. $scope.discoveryFailed = discoveryFailed;
  381. refreshNoAuthWarning();
  382. console.log("refreshSystem", data);
  383. }).error($scope.emitHTTPError);
  384. }
  385. function refreshNoAuthWarning() {
  386. if (!$scope.system || !$scope.config || !$scope.config.gui) {
  387. // We need all to be able to determine the state.
  388. return
  389. }
  390. // If we're not listening on localhost, and there is no
  391. // authentication configured, and the magic setting to silence the
  392. // warning isn't set, then yell at the user.
  393. var addr = $scope.system.guiAddressUsed;
  394. var guiCfg = $scope.config.gui;
  395. $scope.openNoAuth = addr.substr(0, 4) !== "127."
  396. && addr.substr(0, 6) !== "[::1]:"
  397. && addr.substr(0, 1) !== "/"
  398. && (!guiCfg.user || !guiCfg.password)
  399. && guiCfg.authMode !== 'ldap'
  400. && !guiCfg.insecureAdminAccess;
  401. if (guiCfg.user && guiCfg.password) {
  402. $scope.dismissNotification('authenticationUserAndPassword');
  403. }
  404. }
  405. function refreshDiscoveryCache() {
  406. $http.get(urlbase + '/system/discovery').success(function (data) {
  407. for (var device in data) {
  408. for (var i = 0; i < data[device].addresses.length; i++) {
  409. // Relay addresses are URLs with
  410. // .../?foo=barlongstuff that we strip away here. We
  411. // remove the final slash as well for symmetry with
  412. // tcp://192.0.2.42:1234 type addresses.
  413. data[device].addresses[i] = data[device].addresses[i].replace(/\/\?.*/, '');
  414. }
  415. }
  416. $scope.discoveryCache = data;
  417. console.log("refreshDiscoveryCache", data);
  418. }).error($scope.emitHTTPError);
  419. }
  420. function recalcLocalStateTotal() {
  421. $scope.localStateTotal = {
  422. bytes: 0,
  423. directories: 0,
  424. files: 0
  425. };
  426. for (var f in $scope.model) {
  427. $scope.localStateTotal.bytes += $scope.model[f].localBytes;
  428. $scope.localStateTotal.files += $scope.model[f].localFiles;
  429. $scope.localStateTotal.directories += $scope.model[f].localDirectories;
  430. }
  431. }
  432. function recalcCompletion(device) {
  433. var total = 0, needed = 0, deletes = 0, items = 0;
  434. for (var folder in $scope.completion[device]) {
  435. if (folder === "_total" || folder === '_needBytes' || folder === '_needItems') {
  436. continue;
  437. }
  438. total += $scope.completion[device][folder].globalBytes;
  439. needed += $scope.completion[device][folder].needBytes;
  440. items += $scope.completion[device][folder].needItems;
  441. deletes += $scope.completion[device][folder].needDeletes;
  442. }
  443. if (total == 0) {
  444. $scope.completion[device]._total = 100;
  445. $scope.completion[device]._needBytes = 0;
  446. $scope.completion[device]._needItems = 0;
  447. } else {
  448. $scope.completion[device]._total = Math.floor(100 * (1 - needed / total));
  449. $scope.completion[device]._needBytes = needed;
  450. $scope.completion[device]._needItems = items + deletes;
  451. }
  452. if (needed == 0 && deletes > 0) {
  453. // We don't need any data, but we have deletes that we need
  454. // to do. Drop down the completion percentage to indicate
  455. // that we have stuff to do.
  456. $scope.completion[device]._total = 95;
  457. }
  458. console.log("recalcCompletion", device, $scope.completion[device]);
  459. }
  460. function refreshCompletion(device, folder) {
  461. if (device === $scope.myID) {
  462. return;
  463. }
  464. $http.get(urlbase + '/db/completion?device=' + device + '&folder=' + encodeURIComponent(folder)).success(function (data) {
  465. if (!$scope.completion[device]) {
  466. $scope.completion[device] = {};
  467. }
  468. $scope.completion[device][folder] = data;
  469. recalcCompletion(device);
  470. }).error($scope.emitHTTPError);
  471. }
  472. function refreshConnectionStats() {
  473. $http.get(urlbase + '/system/connections').success(function (data) {
  474. var now = Date.now(),
  475. td = (now - prevDate) / 1000,
  476. id;
  477. prevDate = now;
  478. try {
  479. data.total.inbps = Math.max(0, (data.total.inBytesTotal - $scope.connectionsTotal.inBytesTotal) / td);
  480. data.total.outbps = Math.max(0, (data.total.outBytesTotal - $scope.connectionsTotal.outBytesTotal) / td);
  481. } catch (e) {
  482. data.total.inbps = 0;
  483. data.total.outbps = 0;
  484. }
  485. $scope.connectionsTotal = data.total;
  486. data = data.connections;
  487. for (id in data) {
  488. if (!data.hasOwnProperty(id)) {
  489. continue;
  490. }
  491. try {
  492. data[id].inbps = Math.max(0, (data[id].inBytesTotal - $scope.connections[id].inBytesTotal) / td);
  493. data[id].outbps = Math.max(0, (data[id].outBytesTotal - $scope.connections[id].outBytesTotal) / td);
  494. } catch (e) {
  495. data[id].inbps = 0;
  496. data[id].outbps = 0;
  497. }
  498. }
  499. $scope.connections = data;
  500. console.log("refreshConnections", data);
  501. }).error($scope.emitHTTPError);
  502. }
  503. function refreshErrors() {
  504. $http.get(urlbase + '/system/error').success(function (data) {
  505. $scope.errors = data.errors;
  506. console.log("refreshErrors", data);
  507. }).error($scope.emitHTTPError);
  508. }
  509. function refreshConfig() {
  510. $http.get(urlbase + '/config').success(function (data) {
  511. updateLocalConfig(data);
  512. console.log("refreshConfig", data);
  513. }).error($scope.emitHTTPError);
  514. $http.get(urlbase + '/config/insync').success(function (data) {
  515. $scope.configInSync = data.configInSync;
  516. }).error($scope.emitHTTPError);
  517. }
  518. $scope.refreshNeed = function (page, perpage) {
  519. if (!$scope.neededFolder) {
  520. return;
  521. }
  522. var url = urlbase + "/db/need?folder=" + encodeURIComponent($scope.neededFolder);
  523. url += "&page=" + page;
  524. url += "&perpage=" + perpage;
  525. $http.get(url).success(function (data) {
  526. console.log("refreshNeed", $scope.neededFolder, data);
  527. parseNeeded(data);
  528. }).error($scope.emitHTTPError);
  529. }
  530. function needAction(file) {
  531. var fDelete = 4096;
  532. var fDirectory = 16384;
  533. if ((file.flags & (fDelete + fDirectory)) === fDelete + fDirectory) {
  534. return 'rmdir';
  535. } else if ((file.flags & fDelete) === fDelete) {
  536. return 'rm';
  537. } else if ((file.flags & fDirectory) === fDirectory) {
  538. return 'touch';
  539. } else {
  540. return 'sync';
  541. }
  542. }
  543. function parseNeeded(data) {
  544. $scope.needed = data;
  545. var merged = [];
  546. data.progress.forEach(function (item) {
  547. item.type = "progress";
  548. item.action = needAction(item);
  549. merged.push(item);
  550. });
  551. data.queued.forEach(function (item) {
  552. item.type = "queued";
  553. item.action = needAction(item);
  554. merged.push(item);
  555. });
  556. data.rest.forEach(function (item) {
  557. item.type = "rest";
  558. item.action = needAction(item);
  559. merged.push(item);
  560. });
  561. $scope.needed.items = merged;
  562. }
  563. function pathJoin(base, name) {
  564. base = expandTilde(base);
  565. if (base[base.length - 1] !== $scope.system.pathSeparator) {
  566. return base + $scope.system.pathSeparator + name;
  567. }
  568. return base + name;
  569. }
  570. function expandTilde(path) {
  571. if (path && path.trim().charAt(0) === '~') {
  572. return $scope.system.tilde + path.trim().substring(1);
  573. }
  574. return path;
  575. }
  576. function shouldSetDefaultFolderPath() {
  577. return $scope.config.options && $scope.config.options.defaultFolderPath && !$scope.editingExisting && $scope.folderEditor.folderPath.$pristine
  578. }
  579. function resetRemoteNeed() {
  580. $scope.remoteNeed = {};
  581. $scope.remoteNeedFolders = [];
  582. $scope.remoteNeedDevice = undefined;
  583. }
  584. function setDefaultTheme() {
  585. if (!document.getElementById("fallback-theme-css")) {
  586. // check if no support for prefers-color-scheme
  587. var colorSchemeNotSupported = typeof window.matchMedia === "undefined" || window.matchMedia('(prefers-color-scheme: dark)').media === 'not all';
  588. if ($scope.config.gui.theme === "default" && colorSchemeNotSupported) {
  589. document.documentElement.style.display = 'none';
  590. document.head.insertAdjacentHTML(
  591. 'beforeend',
  592. '<link id="fallback-theme-css" rel="stylesheet" href="theme-assets/light/assets/css/theme.css" onload="document.documentElement.style.display = \'\'">'
  593. );
  594. }
  595. }
  596. }
  597. function saveIgnores(ignores, cb) {
  598. $http.post(urlbase + '/db/ignores?folder=' + encodeURIComponent($scope.currentFolder.id), {
  599. ignore: ignores
  600. }).success(function () {
  601. if (cb) {
  602. cb();
  603. }
  604. });
  605. };
  606. $scope.refreshFailed = function (page, perpage) {
  607. if (!$scope.failed || !$scope.failed.folder) {
  608. return;
  609. }
  610. var url = urlbase + '/folder/errors?folder=' + encodeURIComponent($scope.failed.folder);
  611. url += "&page=" + page + "&perpage=" + perpage;
  612. $http.get(url).success(function (data) {
  613. $scope.failed = data;
  614. }).error($scope.emitHTTPError);
  615. };
  616. $scope.refreshRemoteNeed = function (folder, page, perpage) {
  617. if (!$scope.remoteNeedDevice) {
  618. return;
  619. }
  620. var url = urlbase + '/db/remoteneed?device=' + $scope.remoteNeedDevice.deviceID;
  621. url += '&folder=' + encodeURIComponent(folder);
  622. url += "&page=" + page + "&perpage=" + perpage;
  623. $http.get(url).success(function (data) {
  624. $scope.remoteNeed[folder] = data;
  625. }).error(function (err) {
  626. $scope.remoteNeed[folder] = undefined;
  627. $scope.emitHTTPError(err);
  628. });
  629. };
  630. $scope.refreshLocalChanged = function (page, perpage) {
  631. if (!$scope.localChangedFolder) {
  632. return;
  633. }
  634. var url = urlbase + '/db/localchanged?folder=';
  635. url += encodeURIComponent($scope.localChangedFolder);
  636. url += "&page=" + page + "&perpage=" + perpage;
  637. $http.get(url).success(function (data) {
  638. $scope.localChanged = data;
  639. }).error($scope.emitHTTPError);
  640. };
  641. var refreshDeviceStats = debounce(function () {
  642. $http.get(urlbase + "/stats/device").success(function (data) {
  643. $scope.deviceStats = data;
  644. for (var device in $scope.deviceStats) {
  645. $scope.deviceStats[device].lastSeen = new Date($scope.deviceStats[device].lastSeen);
  646. $scope.deviceStats[device].lastSeenDays = (new Date() - $scope.deviceStats[device].lastSeen) / 1000 / 86400;
  647. }
  648. console.log("refreshDeviceStats", data);
  649. }).error($scope.emitHTTPError);
  650. }, 2500);
  651. var refreshFolderStats = debounce(function () {
  652. $http.get(urlbase + "/stats/folder").success(function (data) {
  653. $scope.folderStats = data;
  654. for (var folder in $scope.folderStats) {
  655. if ($scope.folderStats[folder].lastFile) {
  656. $scope.folderStats[folder].lastFile.at = new Date($scope.folderStats[folder].lastFile.at);
  657. }
  658. $scope.folderStats[folder].lastScan = new Date($scope.folderStats[folder].lastScan);
  659. $scope.folderStats[folder].lastScanDays = (new Date() - $scope.folderStats[folder].lastScan) / 1000 / 86400;
  660. }
  661. console.log("refreshfolderStats", data);
  662. }).error($scope.emitHTTPError);
  663. }, 2500);
  664. var refreshThemes = debounce(function () {
  665. $http.get("themes.json").success(function (data) { // no urlbase here as this is served by the asset handler
  666. $scope.themes = data.themes;
  667. }).error($scope.emitHTTPError);
  668. }, 2500);
  669. var refreshGlobalChanges = debounce(function () {
  670. $http.get(urlbase + "/events/disk?limit=25").success(function (data) {
  671. if (!data) {
  672. // For reasons unknown this is called with data being the empty
  673. // string on shutdown, causing an error on .reverse().
  674. return;
  675. }
  676. data = data.reverse();
  677. $scope.globalChangeEvents = data;
  678. console.log("refreshGlobalChanges", data);
  679. }).error($scope.emitHTTPError);
  680. }, 2500);
  681. $scope.refresh = function () {
  682. refreshSystem();
  683. refreshDiscoveryCache();
  684. refreshConnectionStats();
  685. refreshErrors();
  686. };
  687. $scope.folderStatus = function (folderCfg) {
  688. if (folderCfg.paused) {
  689. return 'paused';
  690. }
  691. var folderInfo = $scope.model[folderCfg.id];
  692. // after restart syncthing process state may be empty
  693. if (typeof folderInfo === 'undefined' || !folderInfo.state) {
  694. return 'unknown';
  695. }
  696. var state = '' + folderInfo.state;
  697. if (state === 'error') {
  698. return 'stopped'; // legacy, the state is called "stopped" in the GUI
  699. }
  700. if (state !== 'idle') {
  701. return state;
  702. }
  703. if (folderInfo.needTotalItems > 0) {
  704. return 'outofsync';
  705. }
  706. if ($scope.hasFailedFiles(folderCfg.id)) {
  707. return 'faileditems';
  708. }
  709. if (folderInfo.receiveOnlyTotalItems) {
  710. return 'localadditions';
  711. }
  712. if (folderCfg.devices.length <= 1) {
  713. return 'unshared';
  714. }
  715. return state;
  716. };
  717. $scope.folderClass = function (folderCfg) {
  718. var status = $scope.folderStatus(folderCfg);
  719. if (status === 'idle' || status === 'localadditions') {
  720. return 'success';
  721. }
  722. if (status == 'paused') {
  723. return 'default';
  724. }
  725. if (status === 'syncing' || status === 'sync-preparing' || status === 'scanning' || status === 'cleaning') {
  726. return 'primary';
  727. }
  728. if (status === 'unknown') {
  729. return 'info';
  730. }
  731. if (status === 'stopped' || status === 'outofsync' || status === 'error' || status === 'faileditems') {
  732. return 'danger';
  733. }
  734. if (status === 'unshared' || status === 'scan-waiting' || status === 'sync-waiting' || status === 'clean-waiting') {
  735. return 'warning';
  736. }
  737. return 'info';
  738. };
  739. $scope.syncPercentage = function (folder) {
  740. if (typeof $scope.model[folder] === 'undefined') {
  741. return 100;
  742. }
  743. if ($scope.model[folder].needTotalItems === 0) {
  744. return 100;
  745. }
  746. if (($scope.model[folder].needBytes == 0 && $scope.model[folder].needDeletes > 0) || $scope.model[folder].globalBytes == 0) {
  747. // We don't need any data, but we have deletes that we need
  748. // to do. Drop down the completion percentage to indicate
  749. // that we have stuff to do.
  750. // Do the same thing in case we only have zero byte files to sync.
  751. return 95;
  752. }
  753. var pct = 100 * $scope.model[folder].inSyncBytes / $scope.model[folder].globalBytes;
  754. return Math.floor(pct);
  755. };
  756. $scope.scanPercentage = function (folder) {
  757. if (!$scope.scanProgress[folder]) {
  758. return undefined;
  759. }
  760. var pct = 100 * $scope.scanProgress[folder].current / $scope.scanProgress[folder].total;
  761. return Math.floor(pct);
  762. };
  763. $scope.scanRate = function (folder) {
  764. if (!$scope.scanProgress[folder]) {
  765. return 0;
  766. }
  767. return $scope.scanProgress[folder].rate;
  768. };
  769. $scope.scanRemaining = function (folder) {
  770. // Formats the remaining scan time as a string. Includes days and
  771. // hours only when relevant, resulting in time stamps like:
  772. // 00m 40s
  773. // 32m 40s
  774. // 2h 32m
  775. // 4d 2h
  776. // In case remaining scan time appears to be >31d, omit the
  777. // details, i.e.:
  778. // > 1 month
  779. if (!$scope.scanProgress[folder]) {
  780. return "";
  781. }
  782. // Calculate remaining bytes and seconds based on our current
  783. // rate.
  784. var remainingBytes = $scope.scanProgress[folder].total - $scope.scanProgress[folder].current;
  785. var seconds = remainingBytes / $scope.scanProgress[folder].rate;
  786. // Round up to closest ten seconds to avoid flapping too much to
  787. // and fro.
  788. seconds = Math.ceil(seconds / 10) * 10;
  789. // Separate out the number of days.
  790. var days = 0;
  791. var res = [];
  792. if (seconds >= 86400) {
  793. days = Math.floor(seconds / 86400);
  794. if (days > 31) {
  795. return '> 1 month';
  796. }
  797. res.push('' + days + 'd')
  798. seconds = seconds % 86400;
  799. }
  800. // Separate out the number of hours.
  801. var hours = 0;
  802. if (seconds > 3600) {
  803. hours = Math.floor(seconds / 3600);
  804. res.push('' + hours + 'h')
  805. seconds = seconds % 3600;
  806. }
  807. var d = new Date(1970, 0, 1).setSeconds(seconds);
  808. if (days === 0) {
  809. // Format minutes only if we're within a day of completion.
  810. var f = $filter('date')(d, "m'm'");
  811. res.push(f);
  812. }
  813. if (days === 0 && hours === 0) {
  814. // Format seconds only when we're within an hour of completion.
  815. var f = $filter('date')(d, "ss's'");
  816. res.push(f);
  817. }
  818. return res.join(' ');
  819. };
  820. $scope.deviceStatus = function (deviceCfg) {
  821. var status = '';
  822. if ($scope.deviceFolders(deviceCfg).length === 0) {
  823. status = 'unused-';
  824. }
  825. if (typeof $scope.connections[deviceCfg.deviceID] === 'undefined') {
  826. return 'unknown';
  827. }
  828. if (deviceCfg.paused) {
  829. return status + 'paused';
  830. }
  831. if ($scope.connections[deviceCfg.deviceID].connected) {
  832. if ($scope.completion[deviceCfg.deviceID] && $scope.completion[deviceCfg.deviceID]._total === 100) {
  833. return status + 'insync';
  834. } else {
  835. return 'syncing';
  836. }
  837. }
  838. // Disconnected
  839. return status + 'disconnected';
  840. };
  841. $scope.deviceClass = function (deviceCfg) {
  842. if (typeof $scope.connections[deviceCfg.deviceID] === 'undefined') {
  843. return 'info';
  844. }
  845. if (deviceCfg.paused) {
  846. return 'default';
  847. }
  848. if ($scope.connections[deviceCfg.deviceID].connected) {
  849. if ($scope.completion[deviceCfg.deviceID] && $scope.completion[deviceCfg.deviceID]._total === 100) {
  850. return 'success';
  851. } else {
  852. return 'primary';
  853. }
  854. }
  855. // Disconnected
  856. return 'info';
  857. };
  858. $scope.syncthingStatus = function () {
  859. var syncCount = 0;
  860. var notifyCount = 0;
  861. var pauseCount = 0;
  862. // loop through all folders
  863. var folderListCache = $scope.folderList();
  864. for (var i = 0; i < folderListCache.length; i++) {
  865. var status = $scope.folderStatus(folderListCache[i]);
  866. switch (status) {
  867. case 'sync-preparing':
  868. case 'syncing':
  869. syncCount++;
  870. break;
  871. case 'stopped':
  872. case 'unknown':
  873. case 'outofsync':
  874. case 'error':
  875. notifyCount++;
  876. break;
  877. }
  878. }
  879. // loop through all devices
  880. var deviceCount = $scope.devices.length;
  881. var pendingFolders = 0;
  882. for (var i = 0; i < $scope.devices.length; i++) {
  883. var status = $scope.deviceStatus({
  884. deviceID: $scope.devices[i].deviceID
  885. });
  886. switch (status) {
  887. case 'unknown':
  888. notifyCount++;
  889. break;
  890. case 'paused':
  891. pauseCount++;
  892. break;
  893. case 'unused':
  894. deviceCount--;
  895. break;
  896. }
  897. pendingFolders += $scope.devices[i].pendingFolders.length;
  898. }
  899. // enumerate notifications
  900. if ($scope.openNoAuth || !$scope.configInSync || $scope.errorList().length > 0 || !online || (
  901. !isEmptyObject($scope.config) && ($scope.config.pendingDevices.length > 0 || pendingFolders > 0)
  902. )) {
  903. notifyCount++;
  904. }
  905. // at least one folder is syncing
  906. if (syncCount > 0) {
  907. return 'sync';
  908. }
  909. // a device is unknown or a folder is stopped/unknown/outofsync/error or some other notification is open or gui offline
  910. if (notifyCount > 0) {
  911. return 'notify';
  912. }
  913. // all used devices are paused except (this) one
  914. if (pauseCount === deviceCount - 1) {
  915. return 'pause';
  916. }
  917. return 'default';
  918. };
  919. $scope.deviceAddr = function (deviceCfg) {
  920. var conn = $scope.connections[deviceCfg.deviceID];
  921. if (conn && conn.connected) {
  922. return conn.address;
  923. }
  924. return '?';
  925. };
  926. $scope.friendlyNameFromShort = function (shortID) {
  927. var matches = $scope.devices.filter(function (n) {
  928. return n.deviceID.substr(0, 7) === shortID;
  929. });
  930. if (matches.length !== 1) {
  931. return shortID;
  932. }
  933. return matches[0].name;
  934. };
  935. $scope.friendlyNameFromID = function (deviceID) {
  936. var match = $scope.findDevice(deviceID);
  937. if (match) {
  938. return $scope.deviceName(match);
  939. }
  940. return deviceID.substr(0, 6);
  941. };
  942. $scope.findDevice = function (deviceID) {
  943. var matches = $scope.devices.filter(function (n) {
  944. return n.deviceID === deviceID;
  945. });
  946. if (matches.length !== 1) {
  947. return undefined;
  948. }
  949. return matches[0];
  950. };
  951. $scope.deviceName = function (deviceCfg) {
  952. if (typeof deviceCfg === 'undefined' || typeof deviceCfg.deviceID === 'undefined') {
  953. return "";
  954. }
  955. if (deviceCfg.name) {
  956. return deviceCfg.name;
  957. }
  958. return deviceCfg.deviceID.substr(0, 6);
  959. };
  960. $scope.thisDeviceName = function () {
  961. var device = $scope.thisDevice();
  962. if (typeof device === 'undefined') {
  963. return "(unknown device)";
  964. }
  965. if (device.name) {
  966. return device.name;
  967. }
  968. return device.deviceID.substr(0, 6);
  969. };
  970. $scope.setDevicePause = function (device, pause) {
  971. $scope.devices.forEach(function (cfg) {
  972. if (cfg.deviceID == device) {
  973. cfg.paused = pause;
  974. }
  975. });
  976. $scope.config.devices = $scope.devices;
  977. $scope.saveConfig();
  978. };
  979. $scope.setFolderPause = function (folder, pause) {
  980. var cfg = $scope.folders[folder];
  981. if (cfg) {
  982. cfg.paused = pause;
  983. $scope.config.folders = folderList($scope.folders);
  984. $scope.saveConfig();
  985. }
  986. };
  987. $scope.showDiscoveryFailures = function () {
  988. $('#discovery-failures').modal();
  989. };
  990. $scope.logging = {
  991. facilities: {},
  992. refreshFacilities: function () {
  993. $http.get(urlbase + '/system/debug').success(function (data) {
  994. var facilities = {};
  995. data.enabled = data.enabled || [];
  996. $.each(data.facilities, function (key, value) {
  997. facilities[key] = {
  998. description: value,
  999. enabled: data.enabled.indexOf(key) > -1
  1000. }
  1001. })
  1002. $scope.logging.facilities = facilities;
  1003. }).error($scope.emitHTTPError);
  1004. },
  1005. show: function () {
  1006. $scope.logging.refreshFacilities();
  1007. $scope.logging.timer = $timeout($scope.logging.fetch);
  1008. var textArea = $('#logViewerText');
  1009. textArea.on("scroll", $scope.logging.onScroll);
  1010. $('#logViewer').modal().one('shown.bs.modal', function () {
  1011. // Scroll to bottom.
  1012. textArea.scrollTop(textArea[0].scrollHeight);
  1013. }).one('hidden.bs.modal', function () {
  1014. $timeout.cancel($scope.logging.timer);
  1015. textArea.off("scroll", $scope.logging.onScroll);
  1016. $scope.logging.timer = null;
  1017. $scope.logging.entries = [];
  1018. });
  1019. },
  1020. onFacilityChange: function (facility) {
  1021. var enabled = $scope.logging.facilities[facility].enabled;
  1022. // Disable checkboxes while we're in flight.
  1023. $.each($scope.logging.facilities, function (key) {
  1024. $scope.logging.facilities[key].enabled = null;
  1025. })
  1026. $http.post(urlbase + '/system/debug?' + (enabled ? 'enable=' : 'disable=') + facility)
  1027. .success($scope.logging.refreshFacilities)
  1028. .error($scope.emitHTTPError);
  1029. },
  1030. onScroll: function () {
  1031. var textArea = $('#logViewerText');
  1032. var scrollTop = textArea.prop('scrollTop');
  1033. var scrollHeight = textArea.prop('scrollHeight');
  1034. $scope.logging.paused = scrollHeight > (scrollTop + textArea.outerHeight());
  1035. // Browser events do not cause redraw, trigger manually.
  1036. $scope.$apply();
  1037. },
  1038. timer: null,
  1039. entries: [],
  1040. paused: false,
  1041. content: function () {
  1042. var content = "";
  1043. $.each($scope.logging.entries, function (idx, entry) {
  1044. content += entry.when.split('.')[0].replace('T', ' ') + ' ' + entry.message + "\n";
  1045. });
  1046. return content;
  1047. },
  1048. fetch: function () {
  1049. var textArea = $('#logViewerText');
  1050. if ($scope.logging.paused) {
  1051. if (!$scope.logging.timer) return;
  1052. $scope.logging.timer = $timeout($scope.logging.fetch, 500);
  1053. return;
  1054. }
  1055. var last = null;
  1056. if ($scope.logging.entries.length > 0) {
  1057. last = $scope.logging.entries[$scope.logging.entries.length - 1].when;
  1058. }
  1059. $http.get(urlbase + '/system/log' + (last ? '?since=' + encodeURIComponent(last) : '')).success(function (data) {
  1060. if (!$scope.logging.timer) return;
  1061. $scope.logging.timer = $timeout($scope.logging.fetch, 2000);
  1062. if (!$scope.logging.paused) {
  1063. if (data.messages) {
  1064. $scope.logging.entries.push.apply($scope.logging.entries, data.messages);
  1065. // Wait for the text area to be redrawn, adding new lines, and then scroll to bottom.
  1066. $timeout(function () {
  1067. textArea.scrollTop(textArea[0].scrollHeight);
  1068. });
  1069. }
  1070. }
  1071. });
  1072. }
  1073. };
  1074. $scope.discardChangedSettings = function () {
  1075. $("#discard-changes-confirmation").modal("hide");
  1076. $("#settings").off("hide.bs.modal").modal("hide");
  1077. };
  1078. $scope.showSettings = function () {
  1079. // Make a working copy
  1080. $scope.tmpOptions = angular.copy($scope.config.options);
  1081. $scope.tmpOptions.deviceName = $scope.thisDevice().name;
  1082. $scope.tmpOptions.upgrades = "none";
  1083. if ($scope.tmpOptions.autoUpgradeIntervalH > 0) {
  1084. $scope.tmpOptions.upgrades = "stable";
  1085. }
  1086. if ($scope.tmpOptions.upgradeToPreReleases) {
  1087. $scope.tmpOptions.upgrades = "candidate";
  1088. }
  1089. $scope.tmpGUI = angular.copy($scope.config.gui);
  1090. $scope.tmpRemoteIgnoredDevices = angular.copy($scope.config.remoteIgnoredDevices);
  1091. $scope.tmpDevices = angular.copy($scope.config.devices);
  1092. $('#settings').modal("show");
  1093. $("#settings a[href='#settings-general']").tab("show");
  1094. $("#settings").on('hide.bs.modal', function (event) {
  1095. if ($scope.settingsModified()) {
  1096. event.preventDefault();
  1097. $("#discard-changes-confirmation").modal("show");
  1098. } else {
  1099. $("#settings").off("hide.bs.modal");
  1100. }
  1101. });
  1102. };
  1103. $scope.saveConfig = function (callback) {
  1104. var cfg = JSON.stringify($scope.config);
  1105. var opts = {
  1106. headers: {
  1107. 'Content-Type': 'application/json'
  1108. }
  1109. };
  1110. $http.put(urlbase + '/config', cfg, opts).success(function () {
  1111. refreshConfig();
  1112. if (callback) {
  1113. callback();
  1114. }
  1115. }).error(function (data, status, headers, config) {
  1116. refreshConfig();
  1117. $scope.emitHTTPError(data, status, headers, config);
  1118. });
  1119. };
  1120. $scope.urVersions = function () {
  1121. var result = [];
  1122. if ($scope.system) {
  1123. for (var i = $scope.system.urVersionMax; i >= 2; i--) {
  1124. result.push("" + i);
  1125. }
  1126. }
  1127. return result;
  1128. };
  1129. $scope.settingsModified = function () {
  1130. // Options has artificial properties injected into the temp config.
  1131. // Need to recompute them before we can check equality
  1132. var options = angular.copy($scope.config.options);
  1133. options.deviceName = $scope.thisDevice().name;
  1134. options.upgrades = "none";
  1135. if (options.autoUpgradeIntervalH > 0) {
  1136. options.upgrades = "stable";
  1137. }
  1138. if (options.upgradeToPreReleases) {
  1139. options.upgrades = "candidate";
  1140. }
  1141. var optionsEqual = angular.equals(options, $scope.tmpOptions);
  1142. var guiEquals = angular.equals($scope.config.gui, $scope.tmpGUI);
  1143. var ignoredDevicesEquals = angular.equals($scope.config.remoteIgnoredDevices, $scope.tmpRemoteIgnoredDevices);
  1144. var ignoredFoldersEquals = angular.equals($scope.config.devices, $scope.tmpDevices);
  1145. console.log("settings equals - options: " + optionsEqual + " gui: " + guiEquals + " ignDev: " + ignoredDevicesEquals + " ignFol: " + ignoredFoldersEquals);
  1146. return !optionsEqual || !guiEquals || !ignoredDevicesEquals || !ignoredFoldersEquals;
  1147. };
  1148. $scope.saveSettings = function () {
  1149. // Make sure something changed
  1150. if ($scope.settingsModified()) {
  1151. var themeChanged = $scope.config.gui.theme !== $scope.tmpGUI.theme;
  1152. // Angular has issues with selects with numeric values, so we handle strings here.
  1153. $scope.tmpOptions.urAccepted = parseInt($scope.tmpOptions._urAcceptedStr);
  1154. // Check if auto-upgrade has been enabled or disabled. This
  1155. // also has an effect on usage reporting, so do the check
  1156. // for that later.
  1157. if ($scope.tmpOptions.upgrades == "candidate") {
  1158. $scope.tmpOptions.autoUpgradeIntervalH = $scope.tmpOptions.autoUpgradeIntervalH || 12;
  1159. $scope.tmpOptions.upgradeToPreReleases = true;
  1160. $scope.tmpOptions.urAccepted = $scope.system.urVersionMax;
  1161. $scope.tmpOptions.urSeen = $scope.system.urVersionMax;
  1162. } else if ($scope.tmpOptions.upgrades == "stable") {
  1163. $scope.tmpOptions.autoUpgradeIntervalH = $scope.tmpOptions.autoUpgradeIntervalH || 12;
  1164. $scope.tmpOptions.upgradeToPreReleases = false;
  1165. } else {
  1166. $scope.tmpOptions.autoUpgradeIntervalH = 0;
  1167. $scope.tmpOptions.upgradeToPreReleases = false;
  1168. }
  1169. // Check if protocol will need to be changed on restart
  1170. if ($scope.config.gui.useTLS !== $scope.tmpGUI.useTLS) {
  1171. $scope.protocolChanged = true;
  1172. }
  1173. // Parse strings to arrays before copying over
  1174. ['listenAddresses', 'globalAnnounceServers'].forEach(function (key) {
  1175. $scope.tmpOptions[key] = $scope.tmpOptions["_" + key + "Str"].split(/[ ,]+/).map(function (x) {
  1176. return x.trim();
  1177. });
  1178. });
  1179. // Apply new settings locally
  1180. $scope.thisDeviceIn($scope.tmpDevices).name = $scope.tmpOptions.deviceName;
  1181. $scope.config.options = angular.copy($scope.tmpOptions);
  1182. $scope.config.gui = angular.copy($scope.tmpGUI);
  1183. $scope.config.remoteIgnoredDevices = angular.copy($scope.tmpRemoteIgnoredDevices);
  1184. $scope.config.devices = angular.copy($scope.tmpDevices);
  1185. // $scope.devices is updated by updateLocalConfig based on
  1186. // the config changed event, but settingsModified will look
  1187. // at it before that and conclude that the settings are
  1188. // modified (even though we just saved) unless we update
  1189. // here as well...
  1190. $scope.devices = $scope.config.devices;
  1191. $scope.saveConfig(function () {
  1192. if (themeChanged) {
  1193. document.location.reload(true);
  1194. }
  1195. });
  1196. }
  1197. $("#settings").off("hide.bs.modal").modal("hide");
  1198. };
  1199. $scope.saveAdvanced = function () {
  1200. $scope.config = $scope.advancedConfig;
  1201. $scope.saveConfig();
  1202. $('#advanced').modal("hide");
  1203. };
  1204. $scope.restart = function () {
  1205. restarting = true;
  1206. $('#restarting').modal();
  1207. $http.post(urlbase + '/system/restart');
  1208. $scope.configInSync = true;
  1209. // Switch webpage protocol if needed
  1210. if ($scope.protocolChanged) {
  1211. var protocol = 'http';
  1212. if ($scope.config.gui.useTLS) {
  1213. protocol = 'https';
  1214. }
  1215. setTimeout(function () {
  1216. window.location.protocol = protocol;
  1217. }, 2500);
  1218. $scope.protocolChanged = false;
  1219. }
  1220. };
  1221. $scope.upgrade = function () {
  1222. restarting = true;
  1223. $('#upgrade').modal('hide');
  1224. $('#majorUpgrade').modal('hide');
  1225. $('#upgrading').modal();
  1226. $http.post(urlbase + '/system/upgrade').success(function () {
  1227. $('#restarting').modal();
  1228. $('#upgrading').modal('hide');
  1229. }).error(function () {
  1230. $('#upgrading').modal('hide');
  1231. });
  1232. };
  1233. $scope.shutdown = function () {
  1234. restarting = true;
  1235. $http.post(urlbase + '/system/shutdown').success(function () {
  1236. $('#shutdown').modal();
  1237. }).error($scope.emitHTTPError);
  1238. $scope.configInSync = true;
  1239. };
  1240. $scope.editDevice = function (deviceCfg) {
  1241. $scope.currentDevice = $.extend({}, deviceCfg);
  1242. $scope.editingExisting = true;
  1243. $scope.willBeReintroducedBy = undefined;
  1244. if (deviceCfg.introducedBy) {
  1245. var introducerDevice = $scope.findDevice(deviceCfg.introducedBy);
  1246. if (introducerDevice && introducerDevice.introducer) {
  1247. $scope.willBeReintroducedBy = $scope.deviceName(introducerDevice);
  1248. }
  1249. }
  1250. $scope.currentDevice._addressesStr = deviceCfg.addresses.join(', ');
  1251. $scope.currentDevice.selectedFolders = {};
  1252. $scope.deviceFolders($scope.currentDevice).forEach(function (folder) {
  1253. $scope.currentDevice.selectedFolders[folder] = true;
  1254. });
  1255. $scope.deviceEditor.$setPristine();
  1256. $('#editDevice').modal();
  1257. };
  1258. $scope.selectAllFolders = function () {
  1259. angular.forEach($scope.folders, function (_, id) {
  1260. $scope.currentDevice.selectedFolders[id] = true;
  1261. });
  1262. };
  1263. $scope.deSelectAllFolders = function () {
  1264. angular.forEach($scope.folders, function (_, id) {
  1265. $scope.currentDevice.selectedFolders[id] = false;
  1266. });
  1267. };
  1268. $scope.addDevice = function (deviceID, name) {
  1269. return $http.get(urlbase + '/system/discovery')
  1270. .success(function (registry) {
  1271. $scope.discovery = [];
  1272. outer:
  1273. for (var id in registry) {
  1274. if ($scope.discovery.length === 5) {
  1275. break;
  1276. }
  1277. for (var i = 0; i < $scope.devices.length; i++) {
  1278. if ($scope.devices[i].deviceID === id) {
  1279. continue outer;
  1280. }
  1281. }
  1282. $scope.discovery.push(id);
  1283. }
  1284. })
  1285. .then(function () {
  1286. $scope.currentDevice = {
  1287. name: name,
  1288. deviceID: deviceID,
  1289. _addressesStr: 'dynamic',
  1290. compression: 'metadata',
  1291. introducer: false,
  1292. selectedFolders: {},
  1293. pendingFolders: [],
  1294. ignoredFolders: []
  1295. };
  1296. $scope.editingExisting = false;
  1297. $scope.deviceEditor.$setPristine();
  1298. $('#editDevice').modal();
  1299. });
  1300. };
  1301. $scope.deleteDevice = function () {
  1302. $('#editDevice').modal('hide');
  1303. if (!$scope.editingExisting) {
  1304. return;
  1305. }
  1306. $scope.devices = $scope.devices.filter(function (n) {
  1307. return n.deviceID !== $scope.currentDevice.deviceID;
  1308. });
  1309. $scope.config.devices = $scope.devices;
  1310. for (var id in $scope.folders) {
  1311. $scope.folders[id].devices = $scope.folders[id].devices.filter(function (n) {
  1312. return n.deviceID !== $scope.currentDevice.deviceID;
  1313. });
  1314. }
  1315. $scope.saveConfig();
  1316. };
  1317. $scope.saveDevice = function () {
  1318. $('#editDevice').modal('hide');
  1319. $scope.saveDeviceConfig($scope.currentDevice);
  1320. };
  1321. $scope.saveDeviceConfig = function (deviceCfg) {
  1322. deviceCfg.addresses = deviceCfg._addressesStr.split(',').map(function (x) {
  1323. return x.trim();
  1324. });
  1325. var done = false;
  1326. for (var i = 0; i < $scope.devices.length && !done; i++) {
  1327. if ($scope.devices[i].deviceID === deviceCfg.deviceID) {
  1328. $scope.devices[i] = deviceCfg;
  1329. done = true;
  1330. }
  1331. }
  1332. if (!done) {
  1333. $scope.devices.push(deviceCfg);
  1334. }
  1335. $scope.devices.sort(deviceCompare);
  1336. $scope.config.devices = $scope.devices;
  1337. for (var id in deviceCfg.selectedFolders) {
  1338. if (deviceCfg.selectedFolders[id]) {
  1339. var found = false;
  1340. for (i = 0; i < $scope.folders[id].devices.length; i++) {
  1341. if ($scope.folders[id].devices[i].deviceID === deviceCfg.deviceID) {
  1342. found = true;
  1343. break;
  1344. }
  1345. }
  1346. if (!found) {
  1347. $scope.folders[id].devices.push({
  1348. deviceID: deviceCfg.deviceID
  1349. });
  1350. }
  1351. } else {
  1352. $scope.folders[id].devices = $scope.folders[id].devices.filter(function (n) {
  1353. return n.deviceID !== deviceCfg.deviceID;
  1354. });
  1355. }
  1356. }
  1357. $scope.saveConfig();
  1358. };
  1359. $scope.ignoreDevice = function (pendingDevice) {
  1360. pendingDevice = angular.copy(pendingDevice);
  1361. // Bump time
  1362. pendingDevice.time = (new Date()).toISOString();
  1363. $scope.config.remoteIgnoredDevices.push(pendingDevice);
  1364. $scope.saveConfig();
  1365. };
  1366. $scope.unignoreDeviceFromTemporaryConfig = function (ignoredDevice) {
  1367. $scope.tmpRemoteIgnoredDevices = $scope.tmpRemoteIgnoredDevices.filter(function (existingIgnoredDevice) {
  1368. return ignoredDevice.deviceID !== existingIgnoredDevice.deviceID;
  1369. });
  1370. };
  1371. $scope.ignoredFoldersCountTmpConfig = function () {
  1372. var count = 0;
  1373. ($scope.tmpDevices || []).forEach(function (deviceCfg) {
  1374. count += deviceCfg.ignoredFolders.length;
  1375. });
  1376. return count;
  1377. };
  1378. $scope.unignoreFolderFromTemporaryConfig = function (device, ignoredFolderID) {
  1379. for (var i = 0; i < $scope.tmpDevices.length; i++) {
  1380. if ($scope.tmpDevices[i].deviceID == device) {
  1381. $scope.tmpDevices[i].ignoredFolders = $scope.tmpDevices[i].ignoredFolders.filter(function (existingIgnoredFolder) {
  1382. return existingIgnoredFolder.id !== ignoredFolderID;
  1383. });
  1384. return;
  1385. }
  1386. }
  1387. };
  1388. $scope.otherDevices = function () {
  1389. return $scope.devices.filter(function (n) {
  1390. return n.deviceID !== $scope.myID;
  1391. });
  1392. };
  1393. $scope.thisDevice = function () {
  1394. return $scope.thisDeviceIn($scope.devices);
  1395. };
  1396. $scope.thisDeviceIn = function (l) {
  1397. for (var i = 0; i < l.length; i++) {
  1398. var n = l[i];
  1399. if (n.deviceID === $scope.myID) {
  1400. return n;
  1401. }
  1402. }
  1403. };
  1404. $scope.allDevices = function () {
  1405. var devices = $scope.otherDevices();
  1406. devices.push($scope.thisDevice());
  1407. return devices;
  1408. };
  1409. $scope.setAllDevicesPause = function (pause) {
  1410. $scope.devices.forEach(function (cfg) {
  1411. cfg.paused = pause;
  1412. });
  1413. $scope.config.devices = $scope.devices;
  1414. $scope.saveConfig();
  1415. }
  1416. $scope.isAtleastOneDevicePausedStateSetTo = function (pause) {
  1417. for (var i = 0; i < $scope.devices.length; i++) {
  1418. if ($scope.devices[i].paused == pause) {
  1419. return true;
  1420. }
  1421. }
  1422. return false
  1423. }
  1424. $scope.errorList = function () {
  1425. if (!$scope.errors) {
  1426. return [];
  1427. }
  1428. return $scope.errors.filter(function (e) {
  1429. return e.when > $scope.seenError;
  1430. });
  1431. };
  1432. $scope.clearErrors = function () {
  1433. $scope.seenError = $scope.errors[$scope.errors.length - 1].when;
  1434. $http.post(urlbase + '/system/error/clear');
  1435. };
  1436. $scope.fsWatcherErrorMap = function () {
  1437. var errs = {}
  1438. $.each($scope.folders, function (id, cfg) {
  1439. if (cfg.fsWatcherEnabled && $scope.model[cfg.id] && $scope.model[id].watchError && !cfg.paused && $scope.folderStatus(cfg) !== 'stopped') {
  1440. errs[id] = $scope.model[id].watchError;
  1441. }
  1442. });
  1443. return errs;
  1444. };
  1445. $scope.friendlyDevices = function (str) {
  1446. for (var i = 0; i < $scope.devices.length; i++) {
  1447. var cfg = $scope.devices[i];
  1448. str = str.replace(cfg.deviceID, $scope.deviceName(cfg));
  1449. }
  1450. return str;
  1451. };
  1452. $scope.folderList = function () {
  1453. return folderList($scope.folders);
  1454. };
  1455. $scope.directoryList = [];
  1456. $scope.$watch('currentFolder.path', function (newvalue) {
  1457. if (!newvalue) {
  1458. return;
  1459. }
  1460. $scope.currentFolder.path = expandTilde(newvalue);
  1461. $http.get(urlbase + '/system/browse', {
  1462. params: { current: newvalue }
  1463. }).success(function (data) {
  1464. $scope.directoryList = data;
  1465. }).error($scope.emitHTTPError);
  1466. });
  1467. $scope.$watch('currentFolder.label', function (newvalue) {
  1468. if (!newvalue || !shouldSetDefaultFolderPath()) {
  1469. return;
  1470. }
  1471. $scope.currentFolder.path = pathJoin($scope.config.options.defaultFolderPath, newvalue);
  1472. });
  1473. $scope.$watch('currentFolder.id', function (newvalue) {
  1474. if (!newvalue || !shouldSetDefaultFolderPath() || $scope.currentFolder.label) {
  1475. return;
  1476. }
  1477. $scope.currentFolder.path = pathJoin($scope.config.options.defaultFolderPath, newvalue);
  1478. });
  1479. $scope.fsWatcherToggled = function () {
  1480. if ($scope.currentFolder.fsWatcherEnabled) {
  1481. $scope.currentFolder.rescanIntervalS = 3600;
  1482. } else {
  1483. $scope.currentFolder.rescanIntervalS = 60;
  1484. }
  1485. };
  1486. $scope.loadFormIntoScope = function (form) {
  1487. console.log('loadFormIntoScope', form.$name);
  1488. switch (form.$name) {
  1489. case 'deviceEditor':
  1490. $scope.deviceEditor = form;
  1491. break;
  1492. case 'folderEditor':
  1493. $scope.folderEditor = form;
  1494. break;
  1495. }
  1496. };
  1497. $scope.globalChanges = function () {
  1498. $('#globalChanges').modal();
  1499. };
  1500. $scope.editFolderModal = function () {
  1501. $scope.folderPathErrors = {};
  1502. $scope.folderEditor.$setPristine();
  1503. $('#editFolder').modal().one('shown.bs.tab', function (e) {
  1504. if (e.target.attributes.href.value === "#folder-ignores") {
  1505. $('#folder-ignores textarea').focus();
  1506. }
  1507. }).one('hidden.bs.modal', function () {
  1508. $('.nav-tabs a[href="#folder-general"]').tab('show');
  1509. window.location.hash = "";
  1510. });
  1511. };
  1512. $scope.editFolder = function (folderCfg) {
  1513. $scope.editingExisting = true;
  1514. $scope.currentFolder = angular.copy(folderCfg);
  1515. if ($scope.currentFolder.path.length > 1 && $scope.currentFolder.path.slice(-1) === $scope.system.pathSeparator) {
  1516. $scope.currentFolder.path = $scope.currentFolder.path.slice(0, -1);
  1517. }
  1518. // Cache complete device objects indexed by ID for lookups
  1519. var devMap = deviceMap($scope.devices)
  1520. $scope.currentFolder.sharedDevices = [];
  1521. $scope.currentFolder.selectedDevices = {};
  1522. $scope.currentFolder.devices.forEach(function (n) {
  1523. if (n.deviceID !== $scope.myID) {
  1524. $scope.currentFolder.sharedDevices.push(devMap[n.deviceID]);
  1525. }
  1526. $scope.currentFolder.selectedDevices[n.deviceID] = true;
  1527. });
  1528. $scope.currentFolder.unrelatedDevices = $scope.devices.filter(function (n) {
  1529. return n.deviceID !== $scope.myID
  1530. && !$scope.currentFolder.selectedDevices[n.deviceID]
  1531. });
  1532. if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "trashcan") {
  1533. $scope.currentFolder.trashcanFileVersioning = true;
  1534. $scope.currentFolder.fileVersioningSelector = "trashcan";
  1535. $scope.currentFolder.trashcanClean = +$scope.currentFolder.versioning.params.cleanoutDays;
  1536. $scope.currentFolder.versioningCleanupIntervalS = +$scope.currentFolder.versioning.cleanupIntervalS;
  1537. } else if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "simple") {
  1538. $scope.currentFolder.simpleFileVersioning = true;
  1539. $scope.currentFolder.fileVersioningSelector = "simple";
  1540. $scope.currentFolder.simpleKeep = +$scope.currentFolder.versioning.params.keep;
  1541. $scope.currentFolder.versioningCleanupIntervalS = +$scope.currentFolder.versioning.cleanupIntervalS;
  1542. $scope.currentFolder.trashcanClean = +$scope.currentFolder.versioning.params.cleanoutDays;
  1543. } else if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "staggered") {
  1544. $scope.currentFolder.staggeredFileVersioning = true;
  1545. $scope.currentFolder.fileVersioningSelector = "staggered";
  1546. $scope.currentFolder.staggeredMaxAge = Math.floor(+$scope.currentFolder.versioning.params.maxAge / 86400);
  1547. $scope.currentFolder.staggeredCleanInterval = +$scope.currentFolder.versioning.params.cleanInterval;
  1548. $scope.currentFolder.staggeredVersionsPath = $scope.currentFolder.versioning.params.versionsPath;
  1549. $scope.currentFolder.versioningCleanupIntervalS = +$scope.currentFolder.versioning.cleanupIntervalS;
  1550. } else if ($scope.currentFolder.versioning && $scope.currentFolder.versioning.type === "external") {
  1551. $scope.currentFolder.externalFileVersioning = true;
  1552. $scope.currentFolder.fileVersioningSelector = "external";
  1553. $scope.currentFolder.externalCommand = $scope.currentFolder.versioning.params.command;
  1554. } else {
  1555. $scope.currentFolder.fileVersioningSelector = "none";
  1556. }
  1557. $scope.currentFolder.trashcanClean = $scope.currentFolder.trashcanClean || 0; // weeds out nulls and undefineds
  1558. $scope.currentFolder.simpleKeep = $scope.currentFolder.simpleKeep || 5;
  1559. $scope.currentFolder.staggeredCleanInterval = $scope.currentFolder.staggeredCleanInterval || 3600;
  1560. $scope.currentFolder.staggeredVersionsPath = $scope.currentFolder.staggeredVersionsPath || "";
  1561. $scope.currentFolder.versioningCleanupIntervalS = $scope.currentFolder.versioningCleanupIntervalS || 3600;
  1562. // staggeredMaxAge can validly be zero, which we should not replace
  1563. // with the default value of 365. So only set the default if it's
  1564. // actually undefined.
  1565. if (typeof $scope.currentFolder.staggeredMaxAge === 'undefined') {
  1566. $scope.currentFolder.staggeredMaxAge = 365;
  1567. }
  1568. $scope.currentFolder.externalCommand = $scope.currentFolder.externalCommand || "";
  1569. $scope.ignores.text = 'Loading...';
  1570. $scope.ignores.error = null;
  1571. $scope.ignores.disabled = true;
  1572. $http.get(urlbase + '/db/ignores?folder=' + encodeURIComponent($scope.currentFolder.id))
  1573. .success(function (data) {
  1574. $scope.currentFolder.ignores = data.ignore || [];
  1575. $scope.ignores.text = $scope.currentFolder.ignores.join('\n');
  1576. $scope.ignores.error = data.error;
  1577. $scope.ignores.disabled = false;
  1578. })
  1579. .error(function (err) {
  1580. $scope.ignores.text = $translate.instant("Failed to load ignore patterns.");
  1581. $scope.emitHTTPError(err);
  1582. });
  1583. $scope.editFolderModal();
  1584. };
  1585. $scope.selectAllSharedDevices = function (state) {
  1586. var devices = $scope.currentFolder.sharedDevices;
  1587. for (var i = 0; i < devices.length; i++) {
  1588. $scope.currentFolder.selectedDevices[devices[i].deviceID] = !!state;
  1589. }
  1590. };
  1591. $scope.selectAllUnrelatedDevices = function (state) {
  1592. var devices = $scope.currentFolder.unrelatedDevices;
  1593. for (var i = 0; i < devices.length; i++) {
  1594. $scope.currentFolder.selectedDevices[devices[i].deviceID] = !!state;
  1595. }
  1596. };
  1597. $scope.addFolder = function () {
  1598. $http.get(urlbase + '/svc/random/string?length=10').success(function (data) {
  1599. $scope.editingExisting = false;
  1600. $scope.currentFolder = angular.copy($scope.folderDefaults);
  1601. $scope.currentFolder.id = (data.random.substr(0, 5) + '-' + data.random.substr(5, 5)).toLowerCase();
  1602. $scope.currentFolder.unrelatedDevices = $scope.otherDevices();
  1603. $scope.ignores.text = '';
  1604. $scope.ignores.error = null;
  1605. $scope.ignores.disabled = false;
  1606. $scope.editFolderModal();
  1607. });
  1608. };
  1609. $scope.addFolderAndShare = function (folder, folderLabel, device) {
  1610. $scope.editingExisting = false;
  1611. $scope.currentFolder = angular.copy($scope.folderDefaults);
  1612. $scope.currentFolder.id = folder;
  1613. $scope.currentFolder.label = folderLabel;
  1614. $scope.currentFolder.viewFlags = {
  1615. importFromOtherDevice: true
  1616. };
  1617. $scope.currentFolder.selectedDevices[device] = true;
  1618. $scope.currentFolder.unrelatedDevices = $scope.otherDevices();
  1619. $scope.ignores.text = '';
  1620. $scope.ignores.error = null;
  1621. $scope.ignores.disabled = false;
  1622. $scope.editFolderModal();
  1623. };
  1624. $scope.shareFolderWithDevice = function (folder, device) {
  1625. $scope.folders[folder].devices.push({
  1626. deviceID: device
  1627. });
  1628. $scope.config.folders = folderList($scope.folders);
  1629. $scope.saveConfig();
  1630. };
  1631. $scope.saveFolder = function () {
  1632. $('#editFolder').modal('hide');
  1633. var folderCfg = angular.copy($scope.currentFolder);
  1634. folderCfg.selectedDevices[$scope.myID] = true;
  1635. var newDevices = [];
  1636. folderCfg.devices.forEach(function (dev) {
  1637. if (folderCfg.selectedDevices[dev.deviceID] === true) {
  1638. newDevices.push(dev);
  1639. delete folderCfg.selectedDevices[dev.deviceID];
  1640. };
  1641. });
  1642. for (var deviceID in folderCfg.selectedDevices) {
  1643. if (folderCfg.selectedDevices[deviceID] === true) {
  1644. newDevices.push({
  1645. deviceID: deviceID
  1646. });
  1647. }
  1648. }
  1649. folderCfg.devices = newDevices;
  1650. delete folderCfg.sharedDevices;
  1651. delete folderCfg.selectedDevices;
  1652. delete folderCfg.unrelatedDevices;
  1653. if (folderCfg.fileVersioningSelector === "trashcan") {
  1654. folderCfg.versioning = {
  1655. 'type': 'trashcan',
  1656. 'params': {
  1657. 'cleanoutDays': '' + folderCfg.trashcanClean
  1658. },
  1659. 'cleanupIntervalS': folderCfg.versioningCleanupIntervalS
  1660. };
  1661. delete folderCfg.trashcanFileVersioning;
  1662. delete folderCfg.trashcanClean;
  1663. } else if (folderCfg.fileVersioningSelector === "simple") {
  1664. folderCfg.versioning = {
  1665. 'type': 'simple',
  1666. 'params': {
  1667. 'keep': '' + folderCfg.simpleKeep,
  1668. 'cleanoutDays': '' + folderCfg.trashcanClean
  1669. },
  1670. 'cleanupIntervalS': folderCfg.versioningCleanupIntervalS
  1671. };
  1672. delete folderCfg.simpleFileVersioning;
  1673. delete folderCfg.simpleKeep;
  1674. } else if (folderCfg.fileVersioningSelector === "staggered") {
  1675. folderCfg.versioning = {
  1676. 'type': 'staggered',
  1677. 'params': {
  1678. 'maxAge': '' + (folderCfg.staggeredMaxAge * 86400),
  1679. 'cleanInterval': '' + folderCfg.staggeredCleanInterval,
  1680. 'versionsPath': '' + folderCfg.staggeredVersionsPath
  1681. },
  1682. 'cleanupIntervalS': folderCfg.versioningCleanupIntervalS
  1683. };
  1684. delete folderCfg.staggeredFileVersioning;
  1685. delete folderCfg.staggeredMaxAge;
  1686. delete folderCfg.staggeredCleanInterval;
  1687. delete folderCfg.staggeredVersionsPath;
  1688. } else if (folderCfg.fileVersioningSelector === "external") {
  1689. folderCfg.versioning = {
  1690. 'type': 'external',
  1691. 'params': {
  1692. 'command': '' + folderCfg.externalCommand
  1693. },
  1694. 'cleanupIntervalS': folderCfg.versioningCleanupIntervalS
  1695. };
  1696. delete folderCfg.externalFileVersioning;
  1697. delete folderCfg.externalCommand;
  1698. } else {
  1699. delete folderCfg.versioning;
  1700. }
  1701. var ignoresLoaded = !$scope.ignores.disabled;
  1702. var ignores = $scope.ignores.text.split('\n');
  1703. // Split always returns a minimum 1-length array even for no patterns
  1704. if (ignores.length === 1 && ignores[0] === "") {
  1705. ignores = [];
  1706. }
  1707. if (!$scope.editingExisting && ignores.length) {
  1708. folderCfg.paused = true;
  1709. };
  1710. $scope.folders[folderCfg.id] = folderCfg;
  1711. $scope.config.folders = folderList($scope.folders);
  1712. if (ignoresLoaded && $scope.editingExisting && ignores !== folderCfg.ignores) {
  1713. saveIgnores(ignores);
  1714. };
  1715. $scope.saveConfig(function () {
  1716. if (!$scope.editingExisting && ignores.length) {
  1717. saveIgnores(ignores, function () {
  1718. $scope.setFolderPause(folderCfg.id, false);
  1719. });
  1720. }
  1721. });
  1722. };
  1723. $scope.ignoreFolder = function (device, pendingFolder) {
  1724. pendingFolder = angular.copy(pendingFolder);
  1725. // Bump time
  1726. pendingFolder.time = (new Date()).toISOString();
  1727. for (var i = 0; i < $scope.devices.length; i++) {
  1728. if ($scope.devices[i].deviceID == device) {
  1729. $scope.devices[i].ignoredFolders.push(pendingFolder);
  1730. $scope.saveConfig();
  1731. return;
  1732. }
  1733. }
  1734. };
  1735. $scope.sharesFolder = function (folderCfg) {
  1736. var names = [];
  1737. folderCfg.devices.forEach(function (device) {
  1738. if (device.deviceID !== $scope.myID) {
  1739. names.push($scope.deviceName($scope.findDevice(device.deviceID)));
  1740. }
  1741. });
  1742. names.sort();
  1743. return names.join(", ");
  1744. };
  1745. $scope.deviceFolders = function (deviceCfg) {
  1746. var folders = [];
  1747. $scope.folderList().forEach(function (folder) {
  1748. for (var i = 0; i < folder.devices.length; i++) {
  1749. if (folder.devices[i].deviceID === deviceCfg.deviceID) {
  1750. folders.push(folder.id);
  1751. break;
  1752. }
  1753. }
  1754. });
  1755. return folders;
  1756. };
  1757. $scope.folderLabel = function (folderID) {
  1758. if (!$scope.folders[folderID]) {
  1759. return folderID;
  1760. }
  1761. var label = $scope.folders[folderID].label;
  1762. return label && label.length > 0 ? label : folderID;
  1763. };
  1764. $scope.deleteFolder = function (id) {
  1765. $('#editFolder').modal('hide');
  1766. if (!$scope.editingExisting) {
  1767. return;
  1768. }
  1769. delete $scope.folders[id];
  1770. delete $scope.model[id];
  1771. $scope.config.folders = folderList($scope.folders);
  1772. recalcLocalStateTotal();
  1773. $scope.saveConfig();
  1774. };
  1775. function resetRestoreVersions() {
  1776. $scope.restoreVersions = {
  1777. folder: null,
  1778. selections: {},
  1779. versions: null,
  1780. tree: null,
  1781. errors: null,
  1782. filters: {},
  1783. massAction: function (name, action) {
  1784. $.each($scope.restoreVersions.versions, function (key) {
  1785. if (key.indexOf(name + '/') == 0 && (!$scope.restoreVersions.filters.text || key.indexOf($scope.restoreVersions.filters.text) > -1)) {
  1786. if (action == 'unset') {
  1787. delete $scope.restoreVersions.selections[key];
  1788. return;
  1789. }
  1790. var availableVersions = [];
  1791. $.each($scope.restoreVersions.filterVersions($scope.restoreVersions.versions[key]), function (idx, version) {
  1792. availableVersions.push(version.versionTime);
  1793. })
  1794. if (availableVersions.length) {
  1795. availableVersions.sort(function (a, b) { return a - b; });
  1796. if (action == 'latest') {
  1797. $scope.restoreVersions.selections[key] = availableVersions.pop();
  1798. } else if (action == 'oldest') {
  1799. $scope.restoreVersions.selections[key] = availableVersions.shift();
  1800. }
  1801. }
  1802. }
  1803. });
  1804. },
  1805. filterVersions: function (versions) {
  1806. var filteredVersions = [];
  1807. $.each(versions, function (idx, version) {
  1808. if (moment(version.versionTime).isBetween($scope.restoreVersions.filters['start'], $scope.restoreVersions.filters['end'], null, '[]')) {
  1809. filteredVersions.push(version);
  1810. }
  1811. });
  1812. return filteredVersions;
  1813. },
  1814. selectionCount: function () {
  1815. var count = 0;
  1816. $.each($scope.restoreVersions.selections, function (key, value) {
  1817. if (value) {
  1818. count++;
  1819. }
  1820. });
  1821. return count;
  1822. },
  1823. restore: function () {
  1824. $scope.restoreVersions.tree.clear();
  1825. $scope.restoreVersions.tree = null;
  1826. $scope.restoreVersions.versions = null;
  1827. var selections = {};
  1828. $.each($scope.restoreVersions.selections, function (key, value) {
  1829. if (value) {
  1830. selections[key] = value;
  1831. }
  1832. });
  1833. $scope.restoreVersions.selections = {};
  1834. $http.post(urlbase + '/folder/versions?folder=' + encodeURIComponent($scope.restoreVersions.folder), selections).success(function (data) {
  1835. if (Object.keys(data).length == 0) {
  1836. $('#restoreVersions').modal('hide');
  1837. } else {
  1838. $scope.restoreVersions.errors = data;
  1839. }
  1840. });
  1841. },
  1842. show: function (folder) {
  1843. $scope.restoreVersions.folder = folder;
  1844. var closed = false;
  1845. var modalShown = $q.defer();
  1846. $('#restoreVersions').modal().one('hidden.bs.modal', function () {
  1847. closed = true;
  1848. resetRestoreVersions();
  1849. }).one('shown.bs.modal', function () {
  1850. modalShown.resolve();
  1851. });
  1852. var dataReceived = $http.get(urlbase + '/folder/versions?folder=' + encodeURIComponent($scope.restoreVersions.folder))
  1853. .success(function (data) {
  1854. $.each(data, function (key, values) {
  1855. $.each(values, function (idx, value) {
  1856. value.modTime = new Date(value.modTime);
  1857. value.versionTime = new Date(value.versionTime);
  1858. });
  1859. values.sort(function (a, b) {
  1860. return b.versionTime - a.versionTime;
  1861. });
  1862. });
  1863. if (closed) return;
  1864. $scope.restoreVersions.versions = data;
  1865. });
  1866. $q.all([dataReceived, modalShown.promise]).then(function () {
  1867. $timeout(function () {
  1868. if (closed) {
  1869. resetRestoreVersions();
  1870. return;
  1871. }
  1872. $scope.restoreVersions.tree = $("#restoreTree").fancytree({
  1873. extensions: ["table", "filter"],
  1874. quicksearch: true,
  1875. filter: {
  1876. autoApply: true,
  1877. counter: true,
  1878. hideExpandedCounter: true,
  1879. hideExpanders: true,
  1880. highlight: true,
  1881. leavesOnly: false,
  1882. nodata: true,
  1883. mode: "hide"
  1884. },
  1885. table: {
  1886. indentation: 20,
  1887. nodeColumnIdx: 0,
  1888. },
  1889. debugLevel: 2,
  1890. source: buildTree($scope.restoreVersions.versions),
  1891. renderColumns: function (event, data) {
  1892. var node = data.node,
  1893. $tdList = $(node.tr).find(">td"),
  1894. template;
  1895. if (node.folder) {
  1896. template = '<div ng-include="\'syncthing/folder/restoreVersionsMassActions.html\'" class="pull-right"/>';
  1897. } else {
  1898. template = '<div ng-include="\'syncthing/folder/restoreVersionsVersionSelector.html\'" class="pull-right"/>';
  1899. }
  1900. var scope = $rootScope.$new(true);
  1901. scope.key = node.key;
  1902. scope.restoreVersions = $scope.restoreVersions;
  1903. $tdList.eq(1).html(
  1904. $compile(template)(scope)
  1905. );
  1906. // Force angular to redraw.
  1907. $timeout(function () {
  1908. $scope.$apply();
  1909. });
  1910. }
  1911. }).fancytree("getTree");
  1912. var minDate = moment(),
  1913. maxDate = moment(0, 'X'),
  1914. date;
  1915. // Find version window.
  1916. $.each($scope.restoreVersions.versions, function (key) {
  1917. $.each($scope.restoreVersions.versions[key], function (idx, version) {
  1918. date = moment(version.versionTime);
  1919. if (date.isBefore(minDate)) {
  1920. minDate = date;
  1921. }
  1922. if (date.isAfter(maxDate)) {
  1923. maxDate = date;
  1924. }
  1925. });
  1926. });
  1927. $scope.restoreVersions.filters['start'] = minDate;
  1928. $scope.restoreVersions.filters['end'] = maxDate;
  1929. var ranges = {
  1930. 'All time': [minDate, maxDate],
  1931. 'Today': [moment(), moment()],
  1932. 'Yesterday': [moment().subtract(1, 'days'), moment().subtract(1, 'days')],
  1933. 'Last 7 Days': [moment().subtract(6, 'days'), moment()],
  1934. 'Last 30 Days': [moment().subtract(29, 'days'), moment()],
  1935. 'This Month': [moment().startOf('month'), moment().endOf('month')],
  1936. 'Last Month': [moment().subtract(1, 'month').startOf('month'), moment().subtract(1, 'month').endOf('month')]
  1937. };
  1938. // Filter out invalid ranges.
  1939. $.each(ranges, function (key, range) {
  1940. if (!range[0].isBetween(minDate, maxDate, null, '[]') && !range[1].isBetween(minDate, maxDate, null, '[]')) {
  1941. delete ranges[key];
  1942. }
  1943. });
  1944. $("#restoreVersionDateRange").daterangepicker({
  1945. timePicker: true,
  1946. timePicker24Hour: true,
  1947. timePickerSeconds: true,
  1948. autoUpdateInput: true,
  1949. opens: "left",
  1950. drops: "up",
  1951. startDate: minDate,
  1952. endDate: maxDate,
  1953. minDate: minDate,
  1954. maxDate: maxDate,
  1955. ranges: ranges,
  1956. locale: {
  1957. format: 'YYYY/MM/DD HH:mm:ss',
  1958. }
  1959. }).on('apply.daterangepicker', function (ev, picker) {
  1960. $scope.restoreVersions.filters['start'] = picker.startDate;
  1961. $scope.restoreVersions.filters['end'] = picker.endDate;
  1962. // Events for this UI element are not managed by angular.
  1963. // Force angular to wake up.
  1964. $timeout(function () {
  1965. $scope.$apply();
  1966. });
  1967. });
  1968. });
  1969. });
  1970. }
  1971. };
  1972. }
  1973. resetRestoreVersions();
  1974. $scope.$watchCollection('restoreVersions.filters', function () {
  1975. if (!$scope.restoreVersions.tree) return;
  1976. $scope.restoreVersions.tree.filterNodes(function (node) {
  1977. if (node.folder) return false;
  1978. if ($scope.restoreVersions.filters.text && node.key.indexOf($scope.restoreVersions.filters.text) < 0) {
  1979. return false;
  1980. }
  1981. if ($scope.restoreVersions.filterVersions(node.data.versions).length == 0) {
  1982. return false;
  1983. }
  1984. return true;
  1985. });
  1986. });
  1987. $scope.setAPIKey = function (cfg) {
  1988. $http.get(urlbase + '/svc/random/string?length=32').success(function (data) {
  1989. cfg.apiKey = data.random;
  1990. });
  1991. };
  1992. $scope.acceptUR = function () {
  1993. $scope.config.options.urAccepted = $scope.system.urVersionMax;
  1994. $scope.config.options.urSeen = $scope.system.urVersionMax;
  1995. $scope.saveConfig();
  1996. $('#ur').modal('hide');
  1997. };
  1998. $scope.declineUR = function () {
  1999. if ($scope.config.options.urAccepted === 0) {
  2000. $scope.config.options.urAccepted = -1;
  2001. }
  2002. $scope.config.options.urSeen = $scope.system.urVersionMax;
  2003. $scope.saveConfig();
  2004. $('#ur').modal('hide');
  2005. };
  2006. $scope.showNeed = function (folder) {
  2007. $scope.neededFolder = folder;
  2008. $scope.refreshNeed(1, 10);
  2009. $('#needed').modal().one('hidden.bs.modal', function () {
  2010. $scope.needed = undefined;
  2011. $scope.neededFolder = '';
  2012. });
  2013. };
  2014. $scope.showRemoteNeed = function (device) {
  2015. resetRemoteNeed();
  2016. $scope.remoteNeedDevice = device;
  2017. $scope.deviceFolders(device).forEach(function (folder) {
  2018. var comp = $scope.completion[device.deviceID][folder];
  2019. if (comp !== undefined && comp.needItems + comp.needDeletes === 0) {
  2020. return;
  2021. }
  2022. $scope.remoteNeedFolders.push(folder);
  2023. $scope.refreshRemoteNeed(folder, 1, 10);
  2024. });
  2025. $('#remoteNeed').modal().one('hidden.bs.modal', function () {
  2026. resetRemoteNeed();
  2027. });
  2028. };
  2029. $scope.showFailed = function (folder) {
  2030. $scope.failed.folder = folder;
  2031. $scope.failed = $scope.refreshFailed(1, 10);
  2032. $('#failed').modal().one('hidden.bs.modal', function () {
  2033. $scope.failed = {};
  2034. });
  2035. };
  2036. $scope.hasFailedFiles = function (folder) {
  2037. if (!$scope.model[folder]) {
  2038. return false;
  2039. }
  2040. return $scope.model[folder].errors !== 0;
  2041. };
  2042. $scope.override = function (folder) {
  2043. $http.post(urlbase + "/db/override?folder=" + encodeURIComponent(folder));
  2044. };
  2045. $scope.showLocalChanged = function (folder) {
  2046. $scope.localChangedFolder = folder;
  2047. $scope.localChanged = $scope.refreshLocalChanged(1, 10);
  2048. $('#localChanged').modal().one('hidden.bs.modal', function () {
  2049. $scope.localChanged = {};
  2050. $scope.localChangedFolder = undefined;
  2051. });
  2052. };
  2053. $scope.revert = function (folder) {
  2054. $http.post(urlbase + "/db/revert?folder=" + encodeURIComponent(folder));
  2055. };
  2056. $scope.canRevert = function (folder) {
  2057. var f = $scope.model[folder];
  2058. if (!f) {
  2059. return false;
  2060. }
  2061. return $scope.model[folder].receiveOnlyTotalItems > 0;
  2062. };
  2063. $scope.advanced = function () {
  2064. $scope.advancedConfig = angular.copy($scope.config);
  2065. $('#advanced').modal('show');
  2066. };
  2067. $scope.showReportPreview = function () {
  2068. $scope.reportPreview = true;
  2069. };
  2070. $scope.refreshReportDataPreview = function (ver, diff) {
  2071. $scope.reportDataPreview = '';
  2072. if (!ver) {
  2073. return;
  2074. }
  2075. var version = parseInt(ver);
  2076. if (diff && version > 2) {
  2077. $q.all([
  2078. $http.get(urlbase + '/svc/report?version=' + version),
  2079. $http.get(urlbase + '/svc/report?version=' + (version - 1)),
  2080. ]).then(function (responses) {
  2081. var newReport = responses[0].data;
  2082. var oldReport = responses[1].data;
  2083. angular.forEach(oldReport, function (_, key) {
  2084. delete newReport[key];
  2085. });
  2086. $scope.reportDataPreview = newReport;
  2087. });
  2088. } else {
  2089. $http.get(urlbase + '/svc/report?version=' + version).success(function (data) {
  2090. $scope.reportDataPreview = data;
  2091. }).error($scope.emitHTTPError);
  2092. }
  2093. };
  2094. $scope.rescanAllFolders = function () {
  2095. $http.post(urlbase + "/db/scan");
  2096. };
  2097. $scope.rescanFolder = function (folder) {
  2098. $http.post(urlbase + "/db/scan?folder=" + encodeURIComponent(folder));
  2099. };
  2100. $scope.setAllFoldersPause = function (pause) {
  2101. var folderListCache = $scope.folderList();
  2102. for (var i = 0; i < folderListCache.length; i++) {
  2103. folderListCache[i].paused = pause;
  2104. }
  2105. $scope.config.folders = folderList(folderListCache);
  2106. $scope.saveConfig();
  2107. };
  2108. $scope.isAtleastOneFolderPausedStateSetTo = function (pause) {
  2109. var folderListCache = $scope.folderList();
  2110. for (var i = 0; i < folderListCache.length; i++) {
  2111. if (folderListCache[i].paused == pause) {
  2112. return true;
  2113. }
  2114. }
  2115. return false;
  2116. };
  2117. $scope.activateAllFsWatchers = function () {
  2118. var folders = $scope.folderList();
  2119. $.each(folders, function (i) {
  2120. if (folders[i].fsWatcherEnabled) {
  2121. return;
  2122. }
  2123. folders[i].fsWatcherEnabled = true;
  2124. if (folders[i].rescanIntervalS === 0) {
  2125. return;
  2126. }
  2127. // Delay full scans, but scan at least once per day
  2128. folders[i].rescanIntervalS *= 60;
  2129. if (folders[i].rescanIntervalS > 86400) {
  2130. folders[i].rescanIntervalS = 86400;
  2131. }
  2132. });
  2133. $scope.config.folders = folders;
  2134. $scope.saveConfig();
  2135. };
  2136. $scope.bumpFile = function (folder, file) {
  2137. var url = urlbase + "/db/prio?folder=" + encodeURIComponent(folder) + "&file=" + encodeURIComponent(file);
  2138. // In order to get the right view of data in the response.
  2139. url += "&page=" + $scope.needed.page;
  2140. url += "&perpage=" + $scope.needed.perpage;
  2141. $http.post(url).success(function (data) {
  2142. if ($scope.neededFolder === folder) {
  2143. console.log("bumpFile", folder, data);
  2144. parseNeeded(data);
  2145. }
  2146. }).error($scope.emitHTTPError);
  2147. };
  2148. $scope.versionString = function () {
  2149. if (!$scope.version.version) {
  2150. return '';
  2151. }
  2152. var os = {
  2153. 'darwin': 'macOS',
  2154. 'dragonfly': 'DragonFly BSD',
  2155. 'freebsd': 'FreeBSD',
  2156. 'openbsd': 'OpenBSD',
  2157. 'netbsd': 'NetBSD',
  2158. 'linux': 'Linux',
  2159. 'windows': 'Windows',
  2160. 'solaris': 'Solaris'
  2161. }[$scope.version.os] || $scope.version.os;
  2162. var arch = {
  2163. '386': '32 bit',
  2164. 'amd64': '64 bit',
  2165. 'arm': 'ARM',
  2166. 'arm64': 'AArch64',
  2167. 'ppc64': 'PowerPC',
  2168. 'ppc64le': 'PowerPC (LE)'
  2169. }[$scope.version.arch] || $scope.version.arch;
  2170. return $scope.version.version + ', ' + os + ' (' + arch + ')';
  2171. };
  2172. $scope.inputTypeFor = function (key, value) {
  2173. if (key.substr(0, 1) === '_') {
  2174. return 'skip';
  2175. }
  2176. if (value === null) {
  2177. return 'null';
  2178. }
  2179. if (typeof value === 'number') {
  2180. return 'number';
  2181. }
  2182. if (typeof value === 'boolean') {
  2183. return 'checkbox';
  2184. }
  2185. if (value instanceof Array) {
  2186. return 'list';
  2187. }
  2188. if (typeof value === 'object') {
  2189. return 'skip';
  2190. }
  2191. return 'text';
  2192. };
  2193. $scope.themeName = function (theme) {
  2194. return theme.replace('-', ' ').replace(/(?:^|\s)\S/g, function (a) {
  2195. return a.toUpperCase();
  2196. });
  2197. };
  2198. $scope.modalLoaded = function () {
  2199. // once all modal elements have been processed
  2200. if ($('modal').length === 0) {
  2201. // pseudo main. called on all definitions assigned
  2202. initController();
  2203. }
  2204. };
  2205. $scope.toggleUnits = function () {
  2206. $scope.metricRates = !$scope.metricRates;
  2207. try {
  2208. window.localStorage["metricRates"] = $scope.metricRates;
  2209. } catch (exception) { }
  2210. };
  2211. $scope.sizeOf = function (dict) {
  2212. if (dict === undefined) {
  2213. return 0;
  2214. }
  2215. return Object.keys(dict).length;
  2216. };
  2217. $scope.dismissNotification = function (id) {
  2218. var idx = $scope.config.options.unackedNotificationIDs.indexOf(id);
  2219. if (idx > -1) {
  2220. $scope.config.options.unackedNotificationIDs.splice(idx, 1);
  2221. $scope.saveConfig();
  2222. }
  2223. };
  2224. $scope.abbreviatedError = function (addr) {
  2225. var status = $scope.system.lastDialStatus[addr];
  2226. if (!status || !status.error) {
  2227. return null;
  2228. }
  2229. var time = $filter('date')(status.when, "HH:mm:ss")
  2230. var err = status.error.replace(/.+: /, '');
  2231. return err + " (" + time + ")";
  2232. }
  2233. $scope.setCrashReportingEnabled = function (enabled) {
  2234. $scope.config.options.crashReportingEnabled = enabled;
  2235. $scope.saveConfig();
  2236. };
  2237. $scope.isUnixAddress = function (address) {
  2238. return address != null &&
  2239. (address.indexOf('/') == 0 ||
  2240. address.indexOf('unix://') == 0 ||
  2241. address.indexOf('unixs://') == 0);
  2242. }
  2243. });