syncthingController.js 105 KB

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