logster 1.3.1 → 1.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (114) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +18 -18
  3. data/.travis.yml +15 -15
  4. data/CHANGELOG.md +140 -137
  5. data/Gemfile +4 -4
  6. data/Guardfile +8 -8
  7. data/LICENSE.txt +22 -22
  8. data/README.md +99 -99
  9. data/Rakefile +24 -24
  10. data/assets/fonts/FontAwesome.otf +0 -0
  11. data/assets/fonts/fontawesome-webfont.eot +0 -0
  12. data/assets/fonts/fontawesome-webfont.svg +639 -639
  13. data/assets/fonts/fontawesome-webfont.ttf +0 -0
  14. data/assets/fonts/fontawesome-webfont.woff +0 -0
  15. data/assets/fonts/fontawesome-webfont.woff2 +0 -0
  16. data/assets/images/Icon-144_rounded.png +0 -0
  17. data/assets/images/Icon-144_square.png +0 -0
  18. data/assets/images/icon_144x144.png +0 -0
  19. data/assets/images/icon_64x64.png +0 -0
  20. data/assets/javascript/client-app.js +92 -81
  21. data/assets/stylesheets/client-app.css +1 -1
  22. data/build_client_app.sh +0 -0
  23. data/client-app/.editorconfig +20 -20
  24. data/client-app/.ember-cli +9 -9
  25. data/client-app/.eslintignore +19 -19
  26. data/client-app/.eslintrc.js +46 -46
  27. data/client-app/.gitignore +23 -23
  28. data/client-app/.travis.yml +27 -27
  29. data/client-app/.watchmanconfig +3 -3
  30. data/client-app/README.md +57 -57
  31. data/client-app/app/app.js +14 -14
  32. data/client-app/app/components/actions-menu.js +37 -0
  33. data/client-app/app/components/message-info.js +65 -18
  34. data/client-app/app/components/message-row.js +45 -45
  35. data/client-app/app/components/panel-resizer.js +75 -75
  36. data/client-app/app/components/tab-contents.js +27 -27
  37. data/client-app/app/components/tab-link.js +5 -5
  38. data/client-app/app/components/tabbed-section.js +32 -32
  39. data/client-app/app/components/time-formatter.js +25 -25
  40. data/client-app/app/components/update-time.js +21 -21
  41. data/client-app/app/controllers/index.js +92 -83
  42. data/client-app/app/controllers/show.js +13 -13
  43. data/client-app/app/index.html +29 -29
  44. data/client-app/app/initializers/app-init.js +72 -55
  45. data/client-app/app/lib/preload.js +14 -14
  46. data/client-app/app/lib/utilities.js +140 -140
  47. data/client-app/app/models/message-collection.js +158 -158
  48. data/client-app/app/models/message.js +99 -99
  49. data/client-app/app/resolver.js +3 -3
  50. data/client-app/app/router.js +14 -14
  51. data/client-app/app/routes/index.js +57 -53
  52. data/client-app/app/routes/show.js +14 -14
  53. data/client-app/app/styles/app.css +497 -387
  54. data/client-app/app/templates/application.hbs +2 -2
  55. data/client-app/app/templates/components/actions-menu.hbs +12 -0
  56. data/client-app/app/templates/components/message-info.hbs +41 -44
  57. data/client-app/app/templates/components/message-row.hbs +17 -17
  58. data/client-app/app/templates/components/panel-resizer.hbs +3 -0
  59. data/client-app/app/templates/components/tabbed-section.hbs +10 -10
  60. data/client-app/app/templates/components/time-formatter.hbs +1 -1
  61. data/client-app/app/templates/index.hbs +65 -57
  62. data/client-app/app/templates/show.hbs +7 -4
  63. data/client-app/config/environment.js +51 -51
  64. data/client-app/config/optional-features.json +3 -3
  65. data/client-app/config/targets.js +18 -18
  66. data/client-app/ember-cli-build.js +29 -29
  67. data/client-app/package-lock.json +11365 -11365
  68. data/client-app/package.json +56 -56
  69. data/client-app/testem.js +25 -25
  70. data/client-app/tests/index.html +34 -34
  71. data/client-app/tests/integration/components/actions-menu-test.js +26 -0
  72. data/client-app/tests/integration/components/message-info-test.js +26 -26
  73. data/client-app/tests/integration/components/message-row-test.js +26 -26
  74. data/client-app/tests/integration/components/panel-resizer-test.js +26 -26
  75. data/client-app/tests/integration/components/tab-contents-test.js +26 -26
  76. data/client-app/tests/integration/components/tab-link-test.js +26 -26
  77. data/client-app/tests/integration/components/tabbed-section-test.js +26 -26
  78. data/client-app/tests/integration/components/time-formatter-test.js +26 -26
  79. data/client-app/tests/integration/components/update-time-test.js +26 -26
  80. data/client-app/tests/test-helper.js +8 -8
  81. data/client-app/tests/unit/controllers/index-test.js +12 -12
  82. data/client-app/tests/unit/controllers/show-test.js +12 -12
  83. data/client-app/tests/unit/initializers/app-init-test.js +31 -31
  84. data/client-app/tests/unit/routes/index-test.js +11 -11
  85. data/client-app/tests/unit/routes/show-test.js +11 -11
  86. data/lib/examples/sidekiq_logster_reporter.rb +21 -21
  87. data/lib/logster.rb +54 -54
  88. data/lib/logster/base_store.rb +130 -130
  89. data/lib/logster/configuration.rb +25 -25
  90. data/lib/logster/ignore_pattern.rb +65 -65
  91. data/lib/logster/logger.rb +108 -108
  92. data/lib/logster/message.rb +227 -227
  93. data/lib/logster/middleware/debug_exceptions.rb +26 -26
  94. data/lib/logster/middleware/reporter.rb +56 -56
  95. data/lib/logster/middleware/viewer.rb +221 -220
  96. data/lib/logster/rails/railtie.rb +58 -58
  97. data/lib/logster/redis_store.rb +481 -481
  98. data/lib/logster/version.rb +3 -3
  99. data/lib/logster/web.rb +14 -14
  100. data/logster.gemspec +34 -34
  101. data/test/examples/test_sidekiq_reporter_example.rb +46 -46
  102. data/test/fake_data/Gemfile +4 -4
  103. data/test/fake_data/generate.rb +10 -10
  104. data/test/logster/middleware/test_reporter.rb +21 -21
  105. data/test/logster/middleware/test_viewer.rb +96 -96
  106. data/test/logster/test_base_store.rb +147 -147
  107. data/test/logster/test_ignore_pattern.rb +41 -41
  108. data/test/logster/test_logger.rb +80 -80
  109. data/test/logster/test_message.rb +34 -34
  110. data/test/logster/test_redis_rate_limiter.rb +230 -230
  111. data/test/logster/test_redis_store.rb +427 -427
  112. data/test/test_helper.rb +38 -38
  113. data/vendor/assets/javascripts/logster.js.erb +39 -39
  114. metadata +6 -2
@@ -1,55 +1,72 @@
1
- import {
2
- updateHiddenProperty,
3
- resetTitleCount
4
- } from "client-app/lib/utilities";
5
- import { init } from "client-app/lib/preload";
6
-
7
- export function initialize() {
8
- // config for moment.js
9
- moment.updateLocale("en", {
10
- relativeTime: {
11
- future: "in %s",
12
- past: "%s ago",
13
- s: "secs",
14
- m: "a min",
15
- mm: "%d mins",
16
- h: "an hr",
17
- hh: "%d hrs",
18
- d: "a day",
19
- dd: "%d days",
20
- M: "a mth",
21
- MM: "%d mths",
22
- y: "a yr",
23
- yy: "%d yrs"
24
- }
25
- });
26
-
27
- // parse preloaded json
28
- const dataset = document.getElementById("preloaded-data").dataset;
29
- init(dataset);
30
-
31
- // setup event for updating document title and title count
32
- let hiddenProperty;
33
- let visibilitychange;
34
-
35
- ["", "webkit", "ms", "moz", "ms"].forEach(prefix => {
36
- const check = prefix + (prefix === "" ? "hidden" : "Hidden");
37
- if (document[check] !== undefined && !hiddenProperty) {
38
- hiddenProperty = check;
39
- visibilitychange = prefix + "visibilitychange";
40
- }
41
- });
42
-
43
- updateHiddenProperty(hiddenProperty);
44
- document.addEventListener(
45
- visibilitychange,
46
- () => {
47
- resetTitleCount();
48
- },
49
- false
50
- );
51
- }
52
-
53
- export default {
54
- initialize
55
- };
1
+ import {
2
+ updateHiddenProperty,
3
+ resetTitleCount
4
+ } from "client-app/lib/utilities";
5
+ import { init } from "client-app/lib/preload";
6
+ import Evented from "@ember/object/evented";
7
+ import EmberObject from "@ember/object";
8
+
9
+ const TARGETS = ["component", "route"];
10
+
11
+ export function initialize(app) {
12
+ // config for moment.js
13
+ moment.updateLocale("en", {
14
+ relativeTime: {
15
+ future: "in %s",
16
+ past: "%s ago",
17
+ s: "secs",
18
+ m: "a min",
19
+ mm: "%d mins",
20
+ h: "an hr",
21
+ hh: "%d hrs",
22
+ d: "a day",
23
+ dd: "%d days",
24
+ M: "a mth",
25
+ MM: "%d mths",
26
+ y: "a yr",
27
+ yy: "%d yrs"
28
+ }
29
+ });
30
+
31
+ // parse preloaded json
32
+ const dataset = document.getElementById("preloaded-data").dataset;
33
+ init(dataset);
34
+
35
+ // setup event for updating document title and title count
36
+ let hiddenProperty;
37
+ let visibilitychange;
38
+
39
+ ["", "webkit", "ms", "moz", "ms"].forEach(prefix => {
40
+ const check = prefix + (prefix === "" ? "hidden" : "Hidden");
41
+ if (document[check] !== undefined && !hiddenProperty) {
42
+ hiddenProperty = check;
43
+ visibilitychange = prefix + "visibilitychange";
44
+ }
45
+ });
46
+
47
+ updateHiddenProperty(hiddenProperty);
48
+ document.addEventListener(
49
+ visibilitychange,
50
+ () => {
51
+ resetTitleCount();
52
+ },
53
+ false
54
+ );
55
+
56
+ app.register("events:main", EmberObject.extend(Evented).create(), {
57
+ instantiate: false
58
+ });
59
+ TARGETS.forEach(t => app.inject(t, "events", "events:main"));
60
+
61
+ const isMobile =
62
+ /mobile/i.test(navigator.userAgent) && !/iPad/.test(navigator.userAgent);
63
+ if (isMobile) {
64
+ Em.$("body").addClass("mobile");
65
+ }
66
+ app.register("site:main", { isMobile }, { instantiate: false });
67
+ app.inject("controller", "site", "site:main");
68
+ }
69
+
70
+ export default {
71
+ initialize
72
+ };
@@ -1,14 +1,14 @@
1
- let CONTAINER;
2
-
3
- export function init(dataset) {
4
- CONTAINER = {
5
- rootPath: dataset.rootPath,
6
- preload: JSON.parse(dataset.preloaded)
7
- };
8
- }
9
-
10
- export default {
11
- get(key) {
12
- return Em.get(CONTAINER, key);
13
- }
14
- };
1
+ let CONTAINER;
2
+
3
+ export function init(dataset) {
4
+ CONTAINER = {
5
+ rootPath: dataset.rootPath,
6
+ preload: JSON.parse(dataset.preloaded)
7
+ };
8
+ }
9
+
10
+ export default {
11
+ get(key) {
12
+ return Em.get(CONTAINER, key);
13
+ }
14
+ };
@@ -1,140 +1,140 @@
1
- import Preload from "client-app/lib/preload";
2
-
3
- const entityMap = {
4
- "&": "&",
5
- "<": "&lt;",
6
- ">": "&gt;",
7
- '"': "&quot;",
8
- "'": "&#39;",
9
- "/": "&#x2F;"
10
- };
11
-
12
- export function escapeHtml(string) {
13
- return String(string).replace(/[&<>"'/]/g, s => entityMap[s]);
14
- }
15
-
16
- export function ajax(url, settings) {
17
- settings = settings || {};
18
- settings.headers = settings.headers || {};
19
- settings.headers["X-SILENCE-LOGGER"] = true;
20
- return Em.$.ajax(Preload.get("rootPath") + url, settings);
21
- }
22
-
23
- export function preloadOrAjax(url, settings) {
24
- const preloaded = Preload.get(`preload.${url.replace(".json", "")}`);
25
- if (preloaded) {
26
- return Em.RSVP.resolve(preloaded);
27
- } else {
28
- return ajax(url, settings);
29
- }
30
- }
31
-
32
- let HIDDEN_PROPERTY;
33
- let TITLE;
34
- let TITLE_COUNT;
35
-
36
- export function updateHiddenProperty(property) {
37
- HIDDEN_PROPERTY = property;
38
- }
39
-
40
- export function isHidden() {
41
- if (HIDDEN_PROPERTY !== undefined) {
42
- return document[HIDDEN_PROPERTY];
43
- } else {
44
- return !document.hasFocus;
45
- }
46
- }
47
-
48
- export function increaseTitleCount(increment) {
49
- if (!isHidden()) {
50
- return;
51
- }
52
- TITLE = TITLE || document.title;
53
- TITLE_COUNT = TITLE_COUNT || 0;
54
- TITLE_COUNT += increment;
55
- document.title = `${TITLE} (${TITLE_COUNT})`;
56
- }
57
-
58
- export function resetTitleCount() {
59
- TITLE_COUNT = 0;
60
- document.title = TITLE || document.title;
61
- }
62
-
63
- export function formatTime(timestamp) {
64
- let formatted;
65
- const time = moment(timestamp);
66
- const now = moment();
67
-
68
- if (time.diff(now.startOf("day")) > 0) {
69
- formatted = time.format("h:mm a");
70
- } else {
71
- if (time.diff(now.startOf("week")) > 0) {
72
- formatted = time.format("dd h:mm a");
73
- } else {
74
- if (time.diff(now.startOf("year")) > 0) {
75
- formatted = time.format("D MMM h:mm a");
76
- } else {
77
- formatted = time.format("D MMM YY");
78
- }
79
- }
80
- }
81
-
82
- return formatted;
83
- }
84
-
85
- export function buildArrayString(array) {
86
- const buffer = [];
87
- array.forEach(v => {
88
- if (v === null) {
89
- buffer.push("null");
90
- } else if (Object.prototype.toString.call(v) === "[object Array]") {
91
- buffer.push(buildArrayString(v));
92
- } else {
93
- buffer.push(escapeHtml(v.toString()));
94
- }
95
- });
96
- return "[" + buffer.join(", ") + "]";
97
- }
98
-
99
- export function buildHashString(hash, recurse) {
100
- if (!hash) return "";
101
-
102
- const buffer = [];
103
- const hashes = [];
104
- _.each(hash, (v, k) => {
105
- if (v === null) {
106
- buffer.push("null");
107
- } else if (Object.prototype.toString.call(v) === "[object Array]") {
108
- buffer.push(
109
- "<tr><td>" +
110
- escapeHtml(k) +
111
- "</td><td>" +
112
- buildArrayString(v) +
113
- "</td></tr>"
114
- );
115
- } else if (typeof v === "object") {
116
- hashes.push(k);
117
- } else {
118
- buffer.push(
119
- "<tr><td>" + escapeHtml(k) + "</td><td>" + escapeHtml(v) + "</td></tr>"
120
- );
121
- }
122
- });
123
-
124
- if (_.size(hashes) > 0) {
125
- _.each(hashes, function(k1) {
126
- const v = hash[k1];
127
- buffer.push("<tr><td></td><td><table>");
128
- buffer.push(
129
- "<td>" +
130
- escapeHtml(k1) +
131
- "</td><td>" +
132
- buildHashString(v, true) +
133
- "</td>"
134
- );
135
- buffer.push("</table></td></tr>");
136
- });
137
- }
138
- const className = recurse ? "" : "env-table";
139
- return "<table class='" + className + "'>" + buffer.join("\n") + "</table>";
140
- }
1
+ import Preload from "client-app/lib/preload";
2
+
3
+ const entityMap = {
4
+ "&": "&amp;",
5
+ "<": "&lt;",
6
+ ">": "&gt;",
7
+ '"': "&quot;",
8
+ "'": "&#39;",
9
+ "/": "&#x2F;"
10
+ };
11
+
12
+ export function escapeHtml(string) {
13
+ return String(string).replace(/[&<>"'/]/g, s => entityMap[s]);
14
+ }
15
+
16
+ export function ajax(url, settings) {
17
+ settings = settings || {};
18
+ settings.headers = settings.headers || {};
19
+ settings.headers["X-SILENCE-LOGGER"] = true;
20
+ return Em.$.ajax(Preload.get("rootPath") + url, settings);
21
+ }
22
+
23
+ export function preloadOrAjax(url, settings) {
24
+ const preloaded = Preload.get(`preload.${url.replace(".json", "")}`);
25
+ if (preloaded) {
26
+ return Em.RSVP.resolve(preloaded);
27
+ } else {
28
+ return ajax(url, settings);
29
+ }
30
+ }
31
+
32
+ let HIDDEN_PROPERTY;
33
+ let TITLE;
34
+ let TITLE_COUNT;
35
+
36
+ export function updateHiddenProperty(property) {
37
+ HIDDEN_PROPERTY = property;
38
+ }
39
+
40
+ export function isHidden() {
41
+ if (HIDDEN_PROPERTY !== undefined) {
42
+ return document[HIDDEN_PROPERTY];
43
+ } else {
44
+ return !document.hasFocus;
45
+ }
46
+ }
47
+
48
+ export function increaseTitleCount(increment) {
49
+ if (!isHidden()) {
50
+ return;
51
+ }
52
+ TITLE = TITLE || document.title;
53
+ TITLE_COUNT = TITLE_COUNT || 0;
54
+ TITLE_COUNT += increment;
55
+ document.title = `${TITLE} (${TITLE_COUNT})`;
56
+ }
57
+
58
+ export function resetTitleCount() {
59
+ TITLE_COUNT = 0;
60
+ document.title = TITLE || document.title;
61
+ }
62
+
63
+ export function formatTime(timestamp) {
64
+ let formatted;
65
+ const time = moment(timestamp);
66
+ const now = moment();
67
+
68
+ if (time.diff(now.startOf("day")) > 0) {
69
+ formatted = time.format("h:mm a");
70
+ } else {
71
+ if (time.diff(now.startOf("week")) > 0) {
72
+ formatted = time.format("dd h:mm a");
73
+ } else {
74
+ if (time.diff(now.startOf("year")) > 0) {
75
+ formatted = time.format("D MMM h:mm a");
76
+ } else {
77
+ formatted = time.format("D MMM YY");
78
+ }
79
+ }
80
+ }
81
+
82
+ return formatted;
83
+ }
84
+
85
+ export function buildArrayString(array) {
86
+ const buffer = [];
87
+ array.forEach(v => {
88
+ if (v === null) {
89
+ buffer.push("null");
90
+ } else if (Object.prototype.toString.call(v) === "[object Array]") {
91
+ buffer.push(buildArrayString(v));
92
+ } else {
93
+ buffer.push(escapeHtml(v.toString()));
94
+ }
95
+ });
96
+ return "[" + buffer.join(", ") + "]";
97
+ }
98
+
99
+ export function buildHashString(hash, recurse) {
100
+ if (!hash) return "";
101
+
102
+ const buffer = [];
103
+ const hashes = [];
104
+ _.each(hash, (v, k) => {
105
+ if (v === null) {
106
+ buffer.push("null");
107
+ } else if (Object.prototype.toString.call(v) === "[object Array]") {
108
+ buffer.push(
109
+ "<tr><td>" +
110
+ escapeHtml(k) +
111
+ "</td><td>" +
112
+ buildArrayString(v) +
113
+ "</td></tr>"
114
+ );
115
+ } else if (typeof v === "object") {
116
+ hashes.push(k);
117
+ } else {
118
+ buffer.push(
119
+ "<tr><td>" + escapeHtml(k) + "</td><td>" + escapeHtml(v) + "</td></tr>"
120
+ );
121
+ }
122
+ });
123
+
124
+ if (_.size(hashes) > 0) {
125
+ _.each(hashes, function(k1) {
126
+ const v = hash[k1];
127
+ buffer.push("<tr><td></td><td><table>");
128
+ buffer.push(
129
+ "<td>" +
130
+ escapeHtml(k1) +
131
+ "</td><td>" +
132
+ buildHashString(v, true) +
133
+ "</td>"
134
+ );
135
+ buffer.push("</table></td></tr>");
136
+ });
137
+ }
138
+ const className = recurse ? "" : "env-table";
139
+ return "<table class='" + className + "'>" + buffer.join("\n") + "</table>";
140
+ }
@@ -1,158 +1,158 @@
1
- import { ajax, increaseTitleCount } from "client-app/lib/utilities";
2
- import Message from "client-app/models/message";
3
- import { compare } from "@ember/utils";
4
- import { computed } from "@ember/object";
5
-
6
- export default Em.Object.extend({
7
- messages: Em.A(),
8
- currentMessage: null,
9
- total: 0,
10
-
11
- solve(message) {
12
- message.solve().then(() => {
13
- this.reload();
14
- });
15
- },
16
-
17
- destroy(message) {
18
- const messages = this.get("messages");
19
- const idx = messages.indexOf(message);
20
- message.destroy();
21
- message.set("selected", false);
22
- this.set("total", this.get("total") - 1);
23
- this.get("messages").removeObject(message);
24
-
25
- if (idx > 0) {
26
- message = messages[idx - 1];
27
- message.set("selected", true);
28
- this.set("currentMessage", message);
29
- } else {
30
- if (this.get("total") > 0) {
31
- message = messages[0];
32
- message.set("selected", true);
33
- this.set("currentMessage", message);
34
- } else {
35
- this.reload();
36
- }
37
- }
38
- },
39
-
40
- load(opts) {
41
- opts = opts || {};
42
-
43
- const data = {
44
- filter: this.get("filter").join("_")
45
- };
46
-
47
- const search = this.get("search");
48
- if (!_.isEmpty(search)) {
49
- data.search = search;
50
- const regexSearch = this.get("regexSearch");
51
- if (regexSearch) {
52
- data.regex_search = "true";
53
- }
54
- }
55
-
56
- if (opts.before) {
57
- data.before = opts.before;
58
- }
59
-
60
- if (opts.after) {
61
- data.after = opts.after;
62
- }
63
-
64
- ajax("/messages.json", {
65
- data: data
66
- }).then(data => {
67
- // guard against race: ensure the results we're trying to apply
68
- // match the current search terms
69
- if (compare(data.filter, this.get("filter")) != 0) {
70
- return;
71
- }
72
- if (compare(data.search, this.get("search")) != 0) {
73
- return;
74
- }
75
-
76
- if (data.messages.length > 0) {
77
- const newRows = this.toMessages(data.messages);
78
- const messages = this.get("messages");
79
- if (opts.before) {
80
- messages.unshiftObjects(newRows);
81
- } else {
82
- newRows.forEach(nmsg => {
83
- messages.forEach(emsg => {
84
- if (emsg.key == nmsg.key) {
85
- messages.removeObject(emsg);
86
- if (this.get("currentMessage") === emsg) {
87
- // TODO would updateFromJson() work here?
88
- this.set("currentMessage", nmsg);
89
- nmsg.set("selected", emsg.get("selected"));
90
- }
91
- }
92
- });
93
- });
94
- messages.addObjects(newRows);
95
- if (newRows.length > 0) {
96
- increaseTitleCount(newRows.length);
97
- }
98
- }
99
- }
100
- this.set("total", data.total);
101
- });
102
- },
103
-
104
- reload() {
105
- this.set("total", 0);
106
- this.get("messages").clear();
107
-
108
- this.load();
109
- },
110
-
111
- loadMore() {
112
- const messages = this.get("messages");
113
- if (messages.length === 0) {
114
- this.load({});
115
- return;
116
- }
117
-
118
- const lastKey = messages[messages.length - 1].get("key");
119
- this.load({
120
- after: lastKey
121
- });
122
- },
123
-
124
- moreBefore: computed("totalBefore", function() {
125
- return this.get("totalBefore") > 0;
126
- }),
127
-
128
- totalBefore: computed("total", "messages.length", function() {
129
- return this.get("total") - this.get("messages").length;
130
- }),
131
-
132
- showMoreBefore: function() {
133
- const messages = this.get("messages");
134
- const firstKey = messages[0].get("key");
135
-
136
- this.load({
137
- before: firstKey
138
- });
139
- },
140
-
141
- regexSearch: computed("search", function() {
142
- const search = this.get("search");
143
- if (search && search.length > 2 && search[0] === "/") {
144
- const match = search.match(/\/(.*)\/(.*)/);
145
- if (match && match.length === 3) {
146
- try {
147
- return new RegExp(match[1], match[2]);
148
- } catch (err) {
149
- // don't care
150
- }
151
- }
152
- }
153
- }),
154
-
155
- toMessages(messages) {
156
- return messages.map(m => Message.create(m));
157
- }
158
- });
1
+ import { ajax, increaseTitleCount } from "client-app/lib/utilities";
2
+ import Message from "client-app/models/message";
3
+ import { compare } from "@ember/utils";
4
+ import { computed } from "@ember/object";
5
+
6
+ export default Em.Object.extend({
7
+ messages: Em.A(),
8
+ currentMessage: null,
9
+ total: 0,
10
+
11
+ solve(message) {
12
+ message.solve().then(() => {
13
+ this.reload();
14
+ });
15
+ },
16
+
17
+ destroy(message) {
18
+ const messages = this.get("messages");
19
+ const idx = messages.indexOf(message);
20
+ message.destroy();
21
+ message.set("selected", false);
22
+ this.set("total", this.get("total") - 1);
23
+ this.get("messages").removeObject(message);
24
+
25
+ if (idx > 0) {
26
+ message = messages[idx - 1];
27
+ message.set("selected", true);
28
+ this.set("currentMessage", message);
29
+ } else {
30
+ if (this.get("total") > 0) {
31
+ message = messages[0];
32
+ message.set("selected", true);
33
+ this.set("currentMessage", message);
34
+ } else {
35
+ this.reload();
36
+ }
37
+ }
38
+ },
39
+
40
+ load(opts) {
41
+ opts = opts || {};
42
+
43
+ const data = {
44
+ filter: this.get("filter").join("_")
45
+ };
46
+
47
+ const search = this.get("search");
48
+ if (!_.isEmpty(search)) {
49
+ data.search = search;
50
+ const regexSearch = this.get("regexSearch");
51
+ if (regexSearch) {
52
+ data.regex_search = "true";
53
+ }
54
+ }
55
+
56
+ if (opts.before) {
57
+ data.before = opts.before;
58
+ }
59
+
60
+ if (opts.after) {
61
+ data.after = opts.after;
62
+ }
63
+
64
+ ajax("/messages.json", {
65
+ data: data
66
+ }).then(data => {
67
+ // guard against race: ensure the results we're trying to apply
68
+ // match the current search terms
69
+ if (compare(data.filter, this.get("filter")) != 0) {
70
+ return;
71
+ }
72
+ if (compare(data.search, this.get("search")) != 0) {
73
+ return;
74
+ }
75
+
76
+ if (data.messages.length > 0) {
77
+ const newRows = this.toMessages(data.messages);
78
+ const messages = this.get("messages");
79
+ if (opts.before) {
80
+ messages.unshiftObjects(newRows);
81
+ } else {
82
+ newRows.forEach(nmsg => {
83
+ messages.forEach(emsg => {
84
+ if (emsg.key == nmsg.key) {
85
+ messages.removeObject(emsg);
86
+ if (this.get("currentMessage") === emsg) {
87
+ // TODO would updateFromJson() work here?
88
+ this.set("currentMessage", nmsg);
89
+ nmsg.set("selected", emsg.get("selected"));
90
+ }
91
+ }
92
+ });
93
+ });
94
+ messages.addObjects(newRows);
95
+ if (newRows.length > 0) {
96
+ increaseTitleCount(newRows.length);
97
+ }
98
+ }
99
+ }
100
+ this.set("total", data.total);
101
+ });
102
+ },
103
+
104
+ reload() {
105
+ this.set("total", 0);
106
+ this.get("messages").clear();
107
+
108
+ this.load();
109
+ },
110
+
111
+ loadMore() {
112
+ const messages = this.get("messages");
113
+ if (messages.length === 0) {
114
+ this.load({});
115
+ return;
116
+ }
117
+
118
+ const lastKey = messages[messages.length - 1].get("key");
119
+ this.load({
120
+ after: lastKey
121
+ });
122
+ },
123
+
124
+ moreBefore: computed("totalBefore", function() {
125
+ return this.get("totalBefore") > 0;
126
+ }),
127
+
128
+ totalBefore: computed("total", "messages.length", function() {
129
+ return this.get("total") - this.get("messages").length;
130
+ }),
131
+
132
+ showMoreBefore: function() {
133
+ const messages = this.get("messages");
134
+ const firstKey = messages[0].get("key");
135
+
136
+ this.load({
137
+ before: firstKey
138
+ });
139
+ },
140
+
141
+ regexSearch: computed("search", function() {
142
+ const search = this.get("search");
143
+ if (search && search.length > 2 && search[0] === "/") {
144
+ const match = search.match(/\/(.*)\/(.*)/);
145
+ if (match && match.length === 3) {
146
+ try {
147
+ return new RegExp(match[1], match[2]);
148
+ } catch (err) {
149
+ // don't care
150
+ }
151
+ }
152
+ }
153
+ }),
154
+
155
+ toMessages(messages) {
156
+ return messages.map(m => Message.create(m));
157
+ }
158
+ });