diff --git a/Dockerfile b/Dockerfile index 4d20316..88c79cd 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,5 +1,5 @@ # syntax=docker/dockerfile:1 - +# Dockerfile for running test FROM eclipse-temurin:17-jdk-jammy as base WORKDIR /build COPY --chmod=0755 mvnw mvnw diff --git a/README.md b/README.md index e89f098..64c3cb2 100644 --- a/README.md +++ b/README.md @@ -86,6 +86,17 @@ Use Cloud logging to inspect the logs and health of the machine. ### Tests 🧪 #### Unit test #### Integration test +The goal is to achieve clos to production behaviour as possible. Database and Repositories are powered by +an [Embedded DB](https://github.com/flapdoodle-oss/de.flapdoodle.embed.mongo) that only runs in `test` profile. + +1. `Mockito` is used for Mocking external services +2. `MockMvc` is used for the Integration test + +While ``mvn clean test`` is good for running the tests during development, we advise you use the following command to run the `Dockerfile` to verify that the test can run in an +isolated environment without any preconfiguration on your local machine. +```shell +docker build -t java-docker-image-test --progress=plain --no-cache --target=test +``` --- diff --git a/htmlReport/css/coverage.css b/htmlReport/css/coverage.css new file mode 100644 index 0000000..cef7765 --- /dev/null +++ b/htmlReport/css/coverage.css @@ -0,0 +1,154 @@ +/* + * Copyright 2000-2021 JetBrains s.r.o. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +* { + margin: 0; + padding: 0; +} + +body { + background-color: #fff; + font-family: helvetica neue, tahoma, arial, sans-serif; + font-size: 82%; + color: #151515; +} + +h1 { + margin: 0.5em 0; + color: #010101; + font-weight: normal; + font-size: 18px; +} + +h2 { + margin: 0.5em 0; + color: #010101; + font-weight: normal; + font-size: 16px; +} + +a { + color: #1564C2; + text-decoration: none; +} + +a:hover { + text-decoration: underline; +} + +span.separator { + color: #9BA9BA; + padding-left: 5px; + padding-right: 5px; +} + +div.content { + width: 99%; +} + +table.coverageStats { + width: 100%; + border-collapse: collapse; +} + +table.overallStats { + width: 20%; +} + +table.coverageStats td, table.coverageStats th { + padding: 4px 2px; + border-bottom: 1px solid #ccc; +} + +table.coverageStats th { + background-color: #959BA4; + border: none; + font-weight: bold; + text-align: left; + color: #FFF; +} + +table.coverageStats th.coverageStat { + width: 15%; +} + +table.coverageStats th a { + color: #FFF; +} + +table.coverageStats th a:hover { + text-decoration: none; +} + +table.coverageStats th.sortedDesc a { + background: url(../img/arrowDown.gif) no-repeat 100% 2px; + padding-right: 20px; +} + +table.coverageStats th.sortedAsc a { + background: url(../img/arrowUp.gif) no-repeat 100% 2px; + padding-right: 20px; +} + +div.footer { + margin: 2em .5em; + font-size: 85%; + text-align: left; + line-height: 140%; +} + +code.sourceCode { + width: 100%; + border: 1px solid #ccc; + font: normal 12px 'Menlo', 'Bitstream Vera Sans Mono', 'Courier New', 'Courier', monospace; + white-space: pre; +} + +code.sourceCode b { + font-weight: normal; +} + +code.sourceCode span.number { + color: #151515; +} + +code.sourceCode .fc { + background-color: #cfc; +} + +code.sourceCode .pc { + background-color: #ffc; +} + +code.sourceCode .nc { + background-color: #fcc; +} + +.percent, .absValue { + font-size: 90%; +} + +.percent .green, .absValue .green { + color: #32cc32; +} + +.percent .red, .absValue .red { + color: #f00; +} + +.percent .totalDiff { + color: #3f3f3f; +} diff --git a/htmlReport/css/idea.min.css b/htmlReport/css/idea.min.css new file mode 100644 index 0000000..a8d5292 --- /dev/null +++ b/htmlReport/css/idea.min.css @@ -0,0 +1,118 @@ +/* + * Copyright 2000-2021 JetBrains s.r.o. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/* +Intellij Idea-like styling (c) Vasily Polovnyov +*/ + +.hljs { + color: #000; + background: #fff; +} + +.hljs-subst, +.hljs-title { + font-weight: normal; + color: #000; +} + +.hljs-comment, +.hljs-quote { + color: #808080; + font-style: italic; +} + +.hljs-meta { + color: #808000; +} + +.hljs-tag { + background: #efefef; +} + +.hljs-section, +.hljs-name, +.hljs-literal, +.hljs-keyword, +.hljs-selector-tag, +.hljs-type, +.hljs-selector-id, +.hljs-selector-class { + font-weight: bold; + color: #000080; +} + +.hljs-attribute, +.hljs-number, +.hljs-regexp, +.hljs-link { + font-weight: bold; + color: #0000ff; +} + +.hljs-number, +.hljs-regexp, +.hljs-link { + font-weight: normal; +} + +.hljs-string { + color: #008000; + font-weight: bold; +} + +.hljs-symbol, +.hljs-bullet, +.hljs-formula { + color: #000; + background: #d0eded; + font-style: italic; +} + +.hljs-doctag { + text-decoration: underline; +} + +.hljs-variable, +.hljs-template-variable { + color: #660e7a; +} + +.hljs-addition { + background: #baeeba; +} + +.hljs-deletion { + background: #ffc8bd; +} + +.hljs-emphasis { + font-style: italic; +} + +.hljs-strong { + font-weight: bold; +} + +.hljs-ln-numbers { + display: block; + float: left; + width: 3em; + border-right: 1px solid #ccc; + font-style: normal; + text-align: right; + background-color: #eee; +} diff --git a/htmlReport/img/arrowDown.gif b/htmlReport/img/arrowDown.gif new file mode 100644 index 0000000..a4ac9b4 Binary files /dev/null and b/htmlReport/img/arrowDown.gif differ diff --git a/htmlReport/img/arrowUp.gif b/htmlReport/img/arrowUp.gif new file mode 100644 index 0000000..d488db0 Binary files /dev/null and b/htmlReport/img/arrowUp.gif differ diff --git a/htmlReport/index.html b/htmlReport/index.html new file mode 100644 index 0000000..267de40 --- /dev/null +++ b/htmlReport/index.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_BLOCK.html b/htmlReport/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..434c59f --- /dev/null +++ b/htmlReport/index_SORT_BY_BLOCK.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_BLOCK_DESC.html b/htmlReport/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..c6f38cf --- /dev/null +++ b/htmlReport/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_CLASS.html b/htmlReport/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..c9e275b --- /dev/null +++ b/htmlReport/index_SORT_BY_CLASS.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_CLASS_DESC.html b/htmlReport/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..ab3765d --- /dev/null +++ b/htmlReport/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_LINE.html b/htmlReport/index_SORT_BY_LINE.html new file mode 100644 index 0000000..d729004 --- /dev/null +++ b/htmlReport/index_SORT_BY_LINE.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_LINE_DESC.html b/htmlReport/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..481372c --- /dev/null +++ b/htmlReport/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_METHOD.html b/htmlReport/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..43bbda2 --- /dev/null +++ b/htmlReport/index_SORT_BY_METHOD.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_METHOD_DESC.html b/htmlReport/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..73e9f8f --- /dev/null +++ b/htmlReport/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+
+ + + + + + + diff --git a/htmlReport/index_SORT_BY_NAME_DESC.html b/htmlReport/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..6052ce0 --- /dev/null +++ b/htmlReport/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,432 @@ + + + + + + + Coverage Report > Summary + + + + + + +
+ + +

Overall Coverage Summary

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
all classes + + 76.7% + + + (33/43) + + + + 59.2% + + + (126/213) + + + + 34.5% + + + (41/119) + + + + 49.6% + + + (286/577) + +
+ +
+

Coverage Breakdown

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + + 45.5% + + + (5/11) + +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + + 55.3% + + + (63/114) + +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+
+ + + + + + + diff --git a/htmlReport/js/highlight.min.js b/htmlReport/js/highlight.min.js new file mode 100644 index 0000000..e887315 --- /dev/null +++ b/htmlReport/js/highlight.min.js @@ -0,0 +1,1388 @@ +/* + Highlight.js 10.7.2 (00233d63) + License: BSD-3-Clause + Copyright (c) 2006-2021, Ivan Sagalaev + + BSD 3-Clause License + + Copyright (c) 2006-2021, Ivan Sagalaev. + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. + + * Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + + * Neither the name of the copyright holder nor the names of its + contributors may be used to endorse or promote products derived from + this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE + FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER + CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, + OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ +var hljs=function(){"use strict";function e(t){ +return t instanceof Map?t.clear=t.delete=t.set=()=>{ +throw Error("map is read-only")}:t instanceof Set&&(t.add=t.clear=t.delete=()=>{ +throw Error("set is read-only") +}),Object.freeze(t),Object.getOwnPropertyNames(t).forEach((n=>{var i=t[n] +;"object"!=typeof i||Object.isFrozen(i)||e(i)})),t}var t=e,n=e;t.default=n +;class i{constructor(e){ +void 0===e.data&&(e.data={}),this.data=e.data,this.isMatchIgnored=!1} +ignoreMatch(){this.isMatchIgnored=!0}}function s(e){ +return e.replace(/&/g,"&").replace(//g,">").replace(/"/g,""").replace(/'/g,"'") +}function a(e,...t){const n=Object.create(null);for(const t in e)n[t]=e[t] +;return t.forEach((e=>{for(const t in e)n[t]=e[t]})),n}const r=e=>!!e.kind +;class l{constructor(e,t){ +this.buffer="",this.classPrefix=t.classPrefix,e.walk(this)}addText(e){ +this.buffer+=s(e)}openNode(e){if(!r(e))return;let t=e.kind +;e.sublanguage||(t=`${this.classPrefix}${t}`),this.span(t)}closeNode(e){ +r(e)&&(this.buffer+="")}value(){return this.buffer}span(e){ +this.buffer+=``}}class o{constructor(){this.rootNode={ +children:[]},this.stack=[this.rootNode]}get top(){ +return this.stack[this.stack.length-1]}get root(){return this.rootNode}add(e){ +this.top.children.push(e)}openNode(e){const t={kind:e,children:[]} +;this.add(t),this.stack.push(t)}closeNode(){ +if(this.stack.length>1)return this.stack.pop()}closeAllNodes(){ +for(;this.closeNode(););}toJSON(){return JSON.stringify(this.rootNode,null,4)} +walk(e){return this.constructor._walk(e,this.rootNode)}static _walk(e,t){ +return"string"==typeof t?e.addText(t):t.children&&(e.openNode(t), +t.children.forEach((t=>this._walk(e,t))),e.closeNode(t)),e}static _collapse(e){ +"string"!=typeof e&&e.children&&(e.children.every((e=>"string"==typeof e))?e.children=[e.children.join("")]:e.children.forEach((e=>{ +o._collapse(e)})))}}class c extends o{constructor(e){super(),this.options=e} +addKeyword(e,t){""!==e&&(this.openNode(t),this.addText(e),this.closeNode())} +addText(e){""!==e&&this.add(e)}addSublanguage(e,t){const n=e.root +;n.kind=t,n.sublanguage=!0,this.add(n)}toHTML(){ +return new l(this,this.options).value()}finalize(){return!0}}function g(e){ +return e?"string"==typeof e?e:e.source:null} +const u=/\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./,h="[a-zA-Z]\\w*",d="[a-zA-Z_]\\w*",f="\\b\\d+(\\.\\d+)?",p="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",m="\\b(0b[01]+)",b={ +begin:"\\\\[\\s\\S]",relevance:0},E={className:"string",begin:"'",end:"'", +illegal:"\\n",contains:[b]},x={className:"string",begin:'"',end:'"', +illegal:"\\n",contains:[b]},v={ +begin:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/ +},w=(e,t,n={})=>{const i=a({className:"comment",begin:e,end:t,contains:[]},n) +;return i.contains.push(v),i.contains.push({className:"doctag", +begin:"(?:TODO|FIXME|NOTE|BUG|OPTIMIZE|HACK|XXX):",relevance:0}),i +},y=w("//","$"),N=w("/\\*","\\*/"),R=w("#","$");var _=Object.freeze({ +__proto__:null,MATCH_NOTHING_RE:/\b\B/,IDENT_RE:h,UNDERSCORE_IDENT_RE:d, +NUMBER_RE:f,C_NUMBER_RE:p,BINARY_NUMBER_RE:m, +RE_STARTERS_RE:"!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~", +SHEBANG:(e={})=>{const t=/^#![ ]*\// +;return e.binary&&(e.begin=((...e)=>e.map((e=>g(e))).join(""))(t,/.*\b/,e.binary,/\b.*/)), +a({className:"meta",begin:t,end:/$/,relevance:0,"on:begin":(e,t)=>{ +0!==e.index&&t.ignoreMatch()}},e)},BACKSLASH_ESCAPE:b,APOS_STRING_MODE:E, +QUOTE_STRING_MODE:x,PHRASAL_WORDS_MODE:v,COMMENT:w,C_LINE_COMMENT_MODE:y, +C_BLOCK_COMMENT_MODE:N,HASH_COMMENT_MODE:R,NUMBER_MODE:{className:"number", +begin:f,relevance:0},C_NUMBER_MODE:{className:"number",begin:p,relevance:0}, +BINARY_NUMBER_MODE:{className:"number",begin:m,relevance:0},CSS_NUMBER_MODE:{ +className:"number", +begin:f+"(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?", +relevance:0},REGEXP_MODE:{begin:/(?=\/[^/\n]*\/)/,contains:[{className:"regexp", +begin:/\//,end:/\/[gimuy]*/,illegal:/\n/,contains:[b,{begin:/\[/,end:/\]/, +relevance:0,contains:[b]}]}]},TITLE_MODE:{className:"title",begin:h,relevance:0 +},UNDERSCORE_TITLE_MODE:{className:"title",begin:d,relevance:0},METHOD_GUARD:{ +begin:"\\.\\s*[a-zA-Z_]\\w*",relevance:0},END_SAME_AS_BEGIN:e=>Object.assign(e,{ +"on:begin":(e,t)=>{t.data._beginMatch=e[1]},"on:end":(e,t)=>{ +t.data._beginMatch!==e[1]&&t.ignoreMatch()}})});function k(e,t){ +"."===e.input[e.index-1]&&t.ignoreMatch()}function M(e,t){ +t&&e.beginKeywords&&(e.begin="\\b("+e.beginKeywords.split(" ").join("|")+")(?!\\.)(?=\\b|\\s)", +e.__beforeBegin=k,e.keywords=e.keywords||e.beginKeywords,delete e.beginKeywords, +void 0===e.relevance&&(e.relevance=0))}function O(e,t){ +Array.isArray(e.illegal)&&(e.illegal=((...e)=>"("+e.map((e=>g(e))).join("|")+")")(...e.illegal)) +}function A(e,t){if(e.match){ +if(e.begin||e.end)throw Error("begin & end are not supported with match") +;e.begin=e.match,delete e.match}}function L(e,t){ +void 0===e.relevance&&(e.relevance=1)} +const I=["of","and","for","in","not","or","if","then","parent","list","value"] +;function j(e,t,n="keyword"){const i={} +;return"string"==typeof e?s(n,e.split(" ")):Array.isArray(e)?s(n,e):Object.keys(e).forEach((n=>{ +Object.assign(i,j(e[n],t,n))})),i;function s(e,n){ +t&&(n=n.map((e=>e.toLowerCase()))),n.forEach((t=>{const n=t.split("|") +;i[n[0]]=[e,B(n[0],n[1])]}))}}function B(e,t){ +return t?Number(t):(e=>I.includes(e.toLowerCase()))(e)?0:1} +function T(e,{plugins:t}){function n(t,n){ +return RegExp(g(t),"m"+(e.case_insensitive?"i":"")+(n?"g":""))}class i{ +constructor(){ +this.matchIndexes={},this.regexes=[],this.matchAt=1,this.position=0} +addRule(e,t){ +t.position=this.position++,this.matchIndexes[this.matchAt]=t,this.regexes.push([t,e]), +this.matchAt+=(e=>RegExp(e.toString()+"|").exec("").length-1)(e)+1}compile(){ +0===this.regexes.length&&(this.exec=()=>null) +;const e=this.regexes.map((e=>e[1]));this.matcherRe=n(((e,t="|")=>{let n=0 +;return e.map((e=>{n+=1;const t=n;let i=g(e),s="";for(;i.length>0;){ +const e=u.exec(i);if(!e){s+=i;break} +s+=i.substring(0,e.index),i=i.substring(e.index+e[0].length), +"\\"===e[0][0]&&e[1]?s+="\\"+(Number(e[1])+t):(s+=e[0],"("===e[0]&&n++)}return s +})).map((e=>`(${e})`)).join(t)})(e),!0),this.lastIndex=0}exec(e){ +this.matcherRe.lastIndex=this.lastIndex;const t=this.matcherRe.exec(e) +;if(!t)return null +;const n=t.findIndex(((e,t)=>t>0&&void 0!==e)),i=this.matchIndexes[n] +;return t.splice(0,n),Object.assign(t,i)}}class s{constructor(){ +this.rules=[],this.multiRegexes=[], +this.count=0,this.lastIndex=0,this.regexIndex=0}getMatcher(e){ +if(this.multiRegexes[e])return this.multiRegexes[e];const t=new i +;return this.rules.slice(e).forEach((([e,n])=>t.addRule(e,n))), +t.compile(),this.multiRegexes[e]=t,t}resumingScanAtSamePosition(){ +return 0!==this.regexIndex}considerAll(){this.regexIndex=0}addRule(e,t){ +this.rules.push([e,t]),"begin"===t.type&&this.count++}exec(e){ +const t=this.getMatcher(this.regexIndex);t.lastIndex=this.lastIndex +;let n=t.exec(e) +;if(this.resumingScanAtSamePosition())if(n&&n.index===this.lastIndex);else{ +const t=this.getMatcher(0);t.lastIndex=this.lastIndex+1,n=t.exec(e)} +return n&&(this.regexIndex+=n.position+1, +this.regexIndex===this.count&&this.considerAll()),n}} +if(e.compilerExtensions||(e.compilerExtensions=[]), +e.contains&&e.contains.includes("self"))throw Error("ERR: contains `self` is not supported at the top-level of a language. See documentation.") +;return e.classNameAliases=a(e.classNameAliases||{}),function t(i,r){const l=i +;if(i.isCompiled)return l +;[A].forEach((e=>e(i,r))),e.compilerExtensions.forEach((e=>e(i,r))), +i.__beforeBegin=null,[M,O,L].forEach((e=>e(i,r))),i.isCompiled=!0;let o=null +;if("object"==typeof i.keywords&&(o=i.keywords.$pattern, +delete i.keywords.$pattern), +i.keywords&&(i.keywords=j(i.keywords,e.case_insensitive)), +i.lexemes&&o)throw Error("ERR: Prefer `keywords.$pattern` to `mode.lexemes`, BOTH are not allowed. (see mode reference) ") +;return o=o||i.lexemes||/\w+/, +l.keywordPatternRe=n(o,!0),r&&(i.begin||(i.begin=/\B|\b/), +l.beginRe=n(i.begin),i.endSameAsBegin&&(i.end=i.begin), +i.end||i.endsWithParent||(i.end=/\B|\b/), +i.end&&(l.endRe=n(i.end)),l.terminatorEnd=g(i.end)||"", +i.endsWithParent&&r.terminatorEnd&&(l.terminatorEnd+=(i.end?"|":"")+r.terminatorEnd)), +i.illegal&&(l.illegalRe=n(i.illegal)), +i.contains||(i.contains=[]),i.contains=[].concat(...i.contains.map((e=>(e=>(e.variants&&!e.cachedVariants&&(e.cachedVariants=e.variants.map((t=>a(e,{ +variants:null},t)))),e.cachedVariants?e.cachedVariants:S(e)?a(e,{ +starts:e.starts?a(e.starts):null +}):Object.isFrozen(e)?a(e):e))("self"===e?i:e)))),i.contains.forEach((e=>{t(e,l) +})),i.starts&&t(i.starts,r),l.matcher=(e=>{const t=new s +;return e.contains.forEach((e=>t.addRule(e.begin,{rule:e,type:"begin" +}))),e.terminatorEnd&&t.addRule(e.terminatorEnd,{type:"end" +}),e.illegal&&t.addRule(e.illegal,{type:"illegal"}),t})(l),l}(e)}function S(e){ +return!!e&&(e.endsWithParent||S(e.starts))}function P(e){const t={ +props:["language","code","autodetect"],data:()=>({detectedLanguage:"", +unknownLanguage:!1}),computed:{className(){ +return this.unknownLanguage?"":"hljs "+this.detectedLanguage},highlighted(){ +if(!this.autoDetect&&!e.getLanguage(this.language))return console.warn(`The language "${this.language}" you specified could not be found.`), +this.unknownLanguage=!0,s(this.code);let t={} +;return this.autoDetect?(t=e.highlightAuto(this.code), +this.detectedLanguage=t.language):(t=e.highlight(this.language,this.code,this.ignoreIllegals), +this.detectedLanguage=this.language),t.value},autoDetect(){ +return!(this.language&&(e=this.autodetect,!e&&""!==e));var e}, +ignoreIllegals:()=>!0},render(e){return e("pre",{},[e("code",{ +class:this.className,domProps:{innerHTML:this.highlighted}})])}};return{ +Component:t,VuePlugin:{install(e){e.component("highlightjs",t)}}}}const D={ +"after:highlightElement":({el:e,result:t,text:n})=>{const i=H(e) +;if(!i.length)return;const a=document.createElement("div") +;a.innerHTML=t.value,t.value=((e,t,n)=>{let i=0,a="";const r=[];function l(){ +return e.length&&t.length?e[0].offset!==t[0].offset?e[0].offset"}function c(e){ +a+=""}function g(e){("start"===e.event?o:c)(e.node)} +for(;e.length||t.length;){let t=l() +;if(a+=s(n.substring(i,t[0].offset)),i=t[0].offset,t===e){r.reverse().forEach(c) +;do{g(t.splice(0,1)[0]),t=l()}while(t===e&&t.length&&t[0].offset===i) +;r.reverse().forEach(o) +}else"start"===t[0].event?r.push(t[0].node):r.pop(),g(t.splice(0,1)[0])} +return a+s(n.substr(i))})(i,H(a),n)}};function C(e){ +return e.nodeName.toLowerCase()}function H(e){const t=[];return function e(n,i){ +for(let s=n.firstChild;s;s=s.nextSibling)3===s.nodeType?i+=s.nodeValue.length:1===s.nodeType&&(t.push({ +event:"start",offset:i,node:s}),i=e(s,i),C(s).match(/br|hr|img|input/)||t.push({ +event:"stop",offset:i,node:s}));return i}(e,0),t}const $={},U=e=>{ +console.error(e)},z=(e,...t)=>{console.log("WARN: "+e,...t)},K=(e,t)=>{ +$[`${e}/${t}`]||(console.log(`Deprecated as of ${e}. ${t}`),$[`${e}/${t}`]=!0) +},G=s,V=a,W=Symbol("nomatch");return(e=>{ +const n=Object.create(null),s=Object.create(null),a=[];let r=!0 +;const l=/(^(<[^>]+>|\t|)+|\n)/gm,o="Could not find the language '{}', did you forget to load/include a language module?",g={ +disableAutodetect:!0,name:"Plain text",contains:[]};let u={ +noHighlightRe:/^(no-?highlight)$/i, +languageDetectRe:/\blang(?:uage)?-([\w-]+)\b/i,classPrefix:"hljs-", +tabReplace:null,useBR:!1,languages:null,__emitter:c};function h(e){ +return u.noHighlightRe.test(e)}function d(e,t,n,i){let s="",a="" +;"object"==typeof t?(s=e, +n=t.ignoreIllegals,a=t.language,i=void 0):(K("10.7.0","highlight(lang, code, ...args) has been deprecated."), +K("10.7.0","Please use highlight(code, options) instead.\nhttps://github.com/highlightjs/highlight.js/issues/2277"), +a=e,s=t);const r={code:s,language:a};M("before:highlight",r) +;const l=r.result?r.result:f(r.language,r.code,n,i) +;return l.code=r.code,M("after:highlight",l),l}function f(e,t,s,l){ +function c(e,t){const n=v.case_insensitive?t[0].toLowerCase():t[0] +;return Object.prototype.hasOwnProperty.call(e.keywords,n)&&e.keywords[n]} +function g(){null!=R.subLanguage?(()=>{if(""===M)return;let e=null +;if("string"==typeof R.subLanguage){ +if(!n[R.subLanguage])return void k.addText(M) +;e=f(R.subLanguage,M,!0,_[R.subLanguage]),_[R.subLanguage]=e.top +}else e=p(M,R.subLanguage.length?R.subLanguage:null) +;R.relevance>0&&(O+=e.relevance),k.addSublanguage(e.emitter,e.language) +})():(()=>{if(!R.keywords)return void k.addText(M);let e=0 +;R.keywordPatternRe.lastIndex=0;let t=R.keywordPatternRe.exec(M),n="";for(;t;){ +n+=M.substring(e,t.index);const i=c(R,t);if(i){const[e,s]=i +;if(k.addText(n),n="",O+=s,e.startsWith("_"))n+=t[0];else{ +const n=v.classNameAliases[e]||e;k.addKeyword(t[0],n)}}else n+=t[0] +;e=R.keywordPatternRe.lastIndex,t=R.keywordPatternRe.exec(M)} +n+=M.substr(e),k.addText(n)})(),M=""}function h(e){ +return e.className&&k.openNode(v.classNameAliases[e.className]||e.className), +R=Object.create(e,{parent:{value:R}}),R}function d(e,t,n){let s=((e,t)=>{ +const n=e&&e.exec(t);return n&&0===n.index})(e.endRe,n);if(s){if(e["on:end"]){ +const n=new i(e);e["on:end"](t,n),n.isMatchIgnored&&(s=!1)}if(s){ +for(;e.endsParent&&e.parent;)e=e.parent;return e}} +if(e.endsWithParent)return d(e.parent,t,n)}function m(e){ +return 0===R.matcher.regexIndex?(M+=e[0],1):(I=!0,0)}function b(e){ +const n=e[0],i=t.substr(e.index),s=d(R,e,i);if(!s)return W;const a=R +;a.skip?M+=n:(a.returnEnd||a.excludeEnd||(M+=n),g(),a.excludeEnd&&(M=n));do{ +R.className&&k.closeNode(),R.skip||R.subLanguage||(O+=R.relevance),R=R.parent +}while(R!==s.parent) +;return s.starts&&(s.endSameAsBegin&&(s.starts.endRe=s.endRe), +h(s.starts)),a.returnEnd?0:n.length}let E={};function x(n,a){const l=a&&a[0] +;if(M+=n,null==l)return g(),0 +;if("begin"===E.type&&"end"===a.type&&E.index===a.index&&""===l){ +if(M+=t.slice(a.index,a.index+1),!r){const t=Error("0 width match regex") +;throw t.languageName=e,t.badRule=E.rule,t}return 1} +if(E=a,"begin"===a.type)return function(e){ +const t=e[0],n=e.rule,s=new i(n),a=[n.__beforeBegin,n["on:begin"]] +;for(const n of a)if(n&&(n(e,s),s.isMatchIgnored))return m(t) +;return n&&n.endSameAsBegin&&(n.endRe=RegExp(t.replace(/[-/\\^$*+?.()|[\]{}]/g,"\\$&"),"m")), +n.skip?M+=t:(n.excludeBegin&&(M+=t), +g(),n.returnBegin||n.excludeBegin||(M=t)),h(n),n.returnBegin?0:t.length}(a) +;if("illegal"===a.type&&!s){ +const e=Error('Illegal lexeme "'+l+'" for mode "'+(R.className||"")+'"') +;throw e.mode=R,e}if("end"===a.type){const e=b(a);if(e!==W)return e} +if("illegal"===a.type&&""===l)return 1 +;if(L>1e5&&L>3*a.index)throw Error("potential infinite loop, way more iterations than matches") +;return M+=l,l.length}const v=N(e) +;if(!v)throw U(o.replace("{}",e)),Error('Unknown language: "'+e+'"') +;const w=T(v,{plugins:a});let y="",R=l||w;const _={},k=new u.__emitter(u);(()=>{ +const e=[];for(let t=R;t!==v;t=t.parent)t.className&&e.unshift(t.className) +;e.forEach((e=>k.openNode(e)))})();let M="",O=0,A=0,L=0,I=!1;try{ +for(R.matcher.considerAll();;){ +L++,I?I=!1:R.matcher.considerAll(),R.matcher.lastIndex=A +;const e=R.matcher.exec(t);if(!e)break;const n=x(t.substring(A,e.index),e) +;A=e.index+n}return x(t.substr(A)),k.closeAllNodes(),k.finalize(),y=k.toHTML(),{ +relevance:Math.floor(O),value:y,language:e,illegal:!1,emitter:k,top:R}}catch(n){ +if(n.message&&n.message.includes("Illegal"))return{illegal:!0,illegalBy:{ +msg:n.message,context:t.slice(A-100,A+100),mode:n.mode},sofar:y,relevance:0, +value:G(t),emitter:k};if(r)return{illegal:!1,relevance:0,value:G(t),emitter:k, +language:e,top:R,errorRaised:n};throw n}}function p(e,t){ +t=t||u.languages||Object.keys(n);const i=(e=>{const t={relevance:0, +emitter:new u.__emitter(u),value:G(e),illegal:!1,top:g} +;return t.emitter.addText(e),t})(e),s=t.filter(N).filter(k).map((t=>f(t,e,!1))) +;s.unshift(i);const a=s.sort(((e,t)=>{ +if(e.relevance!==t.relevance)return t.relevance-e.relevance +;if(e.language&&t.language){if(N(e.language).supersetOf===t.language)return 1 +;if(N(t.language).supersetOf===e.language)return-1}return 0})),[r,l]=a,o=r +;return o.second_best=l,o}const m={"before:highlightElement":({el:e})=>{ +u.useBR&&(e.innerHTML=e.innerHTML.replace(/\n/g,"").replace(//g,"\n")) +},"after:highlightElement":({result:e})=>{ +u.useBR&&(e.value=e.value.replace(/\n/g,"
"))}},b=/^(<[^>]+>|\t)+/gm,E={ +"after:highlightElement":({result:e})=>{ +u.tabReplace&&(e.value=e.value.replace(b,(e=>e.replace(/\t/g,u.tabReplace))))}} +;function x(e){let t=null;const n=(e=>{let t=e.className+" " +;t+=e.parentNode?e.parentNode.className:"";const n=u.languageDetectRe.exec(t) +;if(n){const t=N(n[1]) +;return t||(z(o.replace("{}",n[1])),z("Falling back to no-highlight mode for this block.",e)), +t?n[1]:"no-highlight"}return t.split(/\s+/).find((e=>h(e)||N(e)))})(e) +;if(h(n))return;M("before:highlightElement",{el:e,language:n}),t=e +;const i=t.textContent,a=n?d(i,{language:n,ignoreIllegals:!0}):p(i) +;M("after:highlightElement",{el:e,result:a,text:i +}),e.innerHTML=a.value,((e,t,n)=>{const i=t?s[t]:n +;e.classList.add("hljs"),i&&e.classList.add(i)})(e,n,a.language),e.result={ +language:a.language,re:a.relevance,relavance:a.relevance +},a.second_best&&(e.second_best={language:a.second_best.language, +re:a.second_best.relevance,relavance:a.second_best.relevance})}const v=()=>{ +v.called||(v.called=!0, +K("10.6.0","initHighlighting() is deprecated. Use highlightAll() instead."), +document.querySelectorAll("pre code").forEach(x))};let w=!1;function y(){ +"loading"!==document.readyState?document.querySelectorAll("pre code").forEach(x):w=!0 +}function N(e){return e=(e||"").toLowerCase(),n[e]||n[s[e]]} +function R(e,{languageName:t}){"string"==typeof e&&(e=[e]),e.forEach((e=>{ +s[e.toLowerCase()]=t}))}function k(e){const t=N(e) +;return t&&!t.disableAutodetect}function M(e,t){const n=e;a.forEach((e=>{ +e[n]&&e[n](t)}))} +"undefined"!=typeof window&&window.addEventListener&&window.addEventListener("DOMContentLoaded",(()=>{ +w&&y()}),!1),Object.assign(e,{highlight:d,highlightAuto:p,highlightAll:y, +fixMarkup:e=>{ +return K("10.2.0","fixMarkup will be removed entirely in v11.0"),K("10.2.0","Please see https://github.com/highlightjs/highlight.js/issues/2534"), +t=e, +u.tabReplace||u.useBR?t.replace(l,(e=>"\n"===e?u.useBR?"
":e:u.tabReplace?e.replace(/\t/g,u.tabReplace):e)):t +;var t},highlightElement:x, +highlightBlock:e=>(K("10.7.0","highlightBlock will be removed entirely in v12.0"), +K("10.7.0","Please use highlightElement now."),x(e)),configure:e=>{ +e.useBR&&(K("10.3.0","'useBR' will be removed entirely in v11.0"), +K("10.3.0","Please see https://github.com/highlightjs/highlight.js/issues/2559")), +u=V(u,e)},initHighlighting:v,initHighlightingOnLoad:()=>{ +K("10.6.0","initHighlightingOnLoad() is deprecated. Use highlightAll() instead."), +w=!0},registerLanguage:(t,i)=>{let s=null;try{s=i(e)}catch(e){ +if(U("Language definition for '{}' could not be registered.".replace("{}",t)), +!r)throw e;U(e),s=g} +s.name||(s.name=t),n[t]=s,s.rawDefinition=i.bind(null,e),s.aliases&&R(s.aliases,{ +languageName:t})},unregisterLanguage:e=>{delete n[e] +;for(const t of Object.keys(s))s[t]===e&&delete s[t]}, +listLanguages:()=>Object.keys(n),getLanguage:N,registerAliases:R, +requireLanguage:e=>{ +K("10.4.0","requireLanguage will be removed entirely in v11."), +K("10.4.0","Please see https://github.com/highlightjs/highlight.js/pull/2844") +;const t=N(e);if(t)return t +;throw Error("The '{}' language is required, but not loaded.".replace("{}",e))}, +autoDetection:k,inherit:V,addPlugin:e=>{(e=>{ +e["before:highlightBlock"]&&!e["before:highlightElement"]&&(e["before:highlightElement"]=t=>{ +e["before:highlightBlock"](Object.assign({block:t.el},t)) +}),e["after:highlightBlock"]&&!e["after:highlightElement"]&&(e["after:highlightElement"]=t=>{ +e["after:highlightBlock"](Object.assign({block:t.el},t))})})(e),a.push(e)}, +vuePlugin:P(e).VuePlugin}),e.debugMode=()=>{r=!1},e.safeMode=()=>{r=!0 +},e.versionString="10.7.2";for(const e in _)"object"==typeof _[e]&&t(_[e]) +;return Object.assign(e,_),e.addPlugin(m),e.addPlugin(D),e.addPlugin(E),e})({}) +}();"object"==typeof exports&&"undefined"!=typeof module&&(module.exports=hljs); +hljs.registerLanguage("apache",(()=>{"use strict";return e=>{const n={ +className:"number",begin:/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}(:\d{1,5})?/} +;return{name:"Apache config",aliases:["apacheconf"],case_insensitive:!0, +contains:[e.HASH_COMMENT_MODE,{className:"section",begin:/<\/?/,end:/>/, +contains:[n,{className:"number",begin:/:\d{1,5}/ +},e.inherit(e.QUOTE_STRING_MODE,{relevance:0})]},{className:"attribute", +begin:/\w+/,relevance:0,keywords:{ +nomarkup:"order deny allow setenv rewriterule rewriteengine rewritecond documentroot sethandler errordocument loadmodule options header listen serverroot servername" +},starts:{end:/$/,relevance:0,keywords:{literal:"on off all deny allow"}, +contains:[{className:"meta",begin:/\s\[/,end:/\]$/},{className:"variable", +begin:/[\$%]\{/,end:/\}/,contains:["self",{className:"number",begin:/[$%]\d+/}] +},n,{className:"number",begin:/\d+/},e.QUOTE_STRING_MODE]}}],illegal:/\S/}} +})()); +hljs.registerLanguage("bash",(()=>{"use strict";function e(...e){ +return e.map((e=>{return(s=e)?"string"==typeof s?s:s.source:null;var s +})).join("")}return s=>{const n={},t={begin:/\$\{/,end:/\}/,contains:["self",{ +begin:/:-/,contains:[n]}]};Object.assign(n,{className:"variable",variants:[{ +begin:e(/\$[\w\d#@][\w\d_]*/,"(?![\\w\\d])(?![$])")},t]});const a={ +className:"subst",begin:/\$\(/,end:/\)/,contains:[s.BACKSLASH_ESCAPE]},i={ +begin:/<<-?\s*(?=\w+)/,starts:{contains:[s.END_SAME_AS_BEGIN({begin:/(\w+)/, +end:/(\w+)/,className:"string"})]}},c={className:"string",begin:/"/,end:/"/, +contains:[s.BACKSLASH_ESCAPE,n,a]};a.contains.push(c);const o={begin:/\$\(\(/, +end:/\)\)/,contains:[{begin:/\d+#[0-9a-f]+/,className:"number"},s.NUMBER_MODE,n] +},r=s.SHEBANG({binary:"(fish|bash|zsh|sh|csh|ksh|tcsh|dash|scsh)",relevance:10 +}),l={className:"function",begin:/\w[\w\d_]*\s*\(\s*\)\s*\{/,returnBegin:!0, +contains:[s.inherit(s.TITLE_MODE,{begin:/\w[\w\d_]*/})],relevance:0};return{ +name:"Bash",aliases:["sh","zsh"],keywords:{$pattern:/\b[a-z._-]+\b/, +keyword:"if then else elif fi for while in do done case esac function", +literal:"true false", +built_in:"break cd continue eval exec exit export getopts hash pwd readonly return shift test times trap umask unset alias bind builtin caller command declare echo enable help let local logout mapfile printf read readarray source type typeset ulimit unalias set shopt autoload bg bindkey bye cap chdir clone comparguments compcall compctl compdescribe compfiles compgroups compquote comptags comptry compvalues dirs disable disown echotc echoti emulate fc fg float functions getcap getln history integer jobs kill limit log noglob popd print pushd pushln rehash sched setcap setopt stat suspend ttyctl unfunction unhash unlimit unsetopt vared wait whence where which zcompile zformat zftp zle zmodload zparseopts zprof zpty zregexparse zsocket zstyle ztcp" +},contains:[r,s.SHEBANG(),l,o,s.HASH_COMMENT_MODE,i,c,{className:"",begin:/\\"/ +},{className:"string",begin:/'/,end:/'/},n]}}})()); +hljs.registerLanguage("c",(()=>{"use strict";function e(e){ +return((...e)=>e.map((e=>(e=>e?"string"==typeof e?e:e.source:null)(e))).join(""))("(",e,")?") +}return t=>{const n=t.COMMENT("//","$",{contains:[{begin:/\\\n/}] +}),r="[a-zA-Z_]\\w*::",a="(decltype\\(auto\\)|"+e(r)+"[a-zA-Z_]\\w*"+e("<[^<>]+>")+")",i={ +className:"keyword",begin:"\\b[a-z\\d_]*_t\\b"},s={className:"string", +variants:[{begin:'(u8?|U|L)?"',end:'"',illegal:"\\n", +contains:[t.BACKSLASH_ESCAPE]},{ +begin:"(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", +end:"'",illegal:"."},t.END_SAME_AS_BEGIN({ +begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\(/,end:/\)([^()\\ ]{0,16})"/})]},o={ +className:"number",variants:[{begin:"\\b(0b[01']+)"},{ +begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)((ll|LL|l|L)(u|U)?|(u|U)(ll|LL|l|L)?|f|F|b|B)" +},{ +begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" +}],relevance:0},c={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{ +"meta-keyword":"if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" +},contains:[{begin:/\\\n/,relevance:0},t.inherit(s,{className:"meta-string"}),{ +className:"meta-string",begin:/<.*?>/},n,t.C_BLOCK_COMMENT_MODE]},l={ +className:"title",begin:e(r)+t.IDENT_RE,relevance:0 +},d=e(r)+t.IDENT_RE+"\\s*\\(",u={ +keyword:"int float while private char char8_t char16_t char32_t catch import module export virtual operator sizeof dynamic_cast|10 typedef const_cast|10 const for static_cast|10 union namespace unsigned long volatile static protected bool template mutable if public friend do goto auto void enum else break extern using asm case typeid wchar_t short reinterpret_cast|10 default double register explicit signed typename try this switch continue inline delete alignas alignof constexpr consteval constinit decltype concept co_await co_return co_yield requires noexcept static_assert thread_local restrict final override atomic_bool atomic_char atomic_schar atomic_uchar atomic_short atomic_ushort atomic_int atomic_uint atomic_long atomic_ulong atomic_llong atomic_ullong new throw return and and_eq bitand bitor compl not not_eq or or_eq xor xor_eq", +built_in:"std string wstring cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream auto_ptr deque list queue stack vector map set pair bitset multiset multimap unordered_set unordered_map unordered_multiset unordered_multimap priority_queue make_pair array shared_ptr abort terminate abs acos asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp fscanf future isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan vfprintf vprintf vsprintf endl initializer_list unique_ptr _Bool complex _Complex imaginary _Imaginary", +literal:"true false nullptr NULL"},m=[c,i,n,t.C_BLOCK_COMMENT_MODE,o,s],p={ +variants:[{begin:/=/,end:/;/},{begin:/\(/,end:/\)/},{ +beginKeywords:"new throw return else",end:/;/}],keywords:u,contains:m.concat([{ +begin:/\(/,end:/\)/,keywords:u,contains:m.concat(["self"]),relevance:0}]), +relevance:0},_={className:"function",begin:"("+a+"[\\*&\\s]+)+"+d, +returnBegin:!0,end:/[{;=]/,excludeEnd:!0,keywords:u,illegal:/[^\w\s\*&:<>.]/, +contains:[{begin:"decltype\\(auto\\)",keywords:u,relevance:0},{begin:d, +returnBegin:!0,contains:[l],relevance:0},{className:"params",begin:/\(/, +end:/\)/,keywords:u,relevance:0,contains:[n,t.C_BLOCK_COMMENT_MODE,s,o,i,{ +begin:/\(/,end:/\)/,keywords:u,relevance:0, +contains:["self",n,t.C_BLOCK_COMMENT_MODE,s,o,i]}] +},i,n,t.C_BLOCK_COMMENT_MODE,c]};return{name:"C",aliases:["h"],keywords:u, +disableAutodetect:!0,illegal:"",keywords:u,contains:["self",i]},{begin:t.IDENT_RE+"::",keywords:u},{ +className:"class",beginKeywords:"enum class struct union",end:/[{;:<>=]/, +contains:[{beginKeywords:"final class struct"},t.TITLE_MODE]}]),exports:{ +preprocessor:c,strings:s,keywords:u}}}})()); +hljs.registerLanguage("coffeescript",(()=>{"use strict" +;const e=["as","in","of","if","for","while","finally","var","new","function","do","return","void","else","break","catch","instanceof","with","throw","case","default","try","switch","continue","typeof","delete","let","yield","const","class","debugger","async","await","static","import","from","export","extends"],n=["true","false","null","undefined","NaN","Infinity"],a=[].concat(["setInterval","setTimeout","clearInterval","clearTimeout","require","exports","eval","isFinite","isNaN","parseFloat","parseInt","decodeURI","decodeURIComponent","encodeURI","encodeURIComponent","escape","unescape"],["arguments","this","super","console","window","document","localStorage","module","global"],["Intl","DataView","Number","Math","Date","String","RegExp","Object","Function","Boolean","Error","Symbol","Set","Map","WeakSet","WeakMap","Proxy","Reflect","JSON","Promise","Float64Array","Int16Array","Int32Array","Int8Array","Uint16Array","Uint32Array","Float32Array","Array","Uint8Array","Uint8ClampedArray","ArrayBuffer","BigInt64Array","BigUint64Array","BigInt"],["EvalError","InternalError","RangeError","ReferenceError","SyntaxError","TypeError","URIError"]) +;return r=>{const t={ +keyword:e.concat(["then","unless","until","loop","by","when","and","or","is","isnt","not"]).filter((i=["var","const","let","function","static"], +e=>!i.includes(e))),literal:n.concat(["yes","no","on","off"]), +built_in:a.concat(["npm","print"])};var i;const s="[A-Za-z$_][0-9A-Za-z$_]*",o={ +className:"subst",begin:/#\{/,end:/\}/,keywords:t +},c=[r.BINARY_NUMBER_MODE,r.inherit(r.C_NUMBER_MODE,{starts:{end:"(\\s*/)?", +relevance:0}}),{className:"string",variants:[{begin:/'''/,end:/'''/, +contains:[r.BACKSLASH_ESCAPE]},{begin:/'/,end:/'/,contains:[r.BACKSLASH_ESCAPE] +},{begin:/"""/,end:/"""/,contains:[r.BACKSLASH_ESCAPE,o]},{begin:/"/,end:/"/, +contains:[r.BACKSLASH_ESCAPE,o]}]},{className:"regexp",variants:[{begin:"///", +end:"///",contains:[o,r.HASH_COMMENT_MODE]},{begin:"//[gim]{0,3}(?=\\W)", +relevance:0},{begin:/\/(?![ *]).*?(?![\\]).\/[gim]{0,3}(?=\W)/}]},{begin:"@"+s +},{subLanguage:"javascript",excludeBegin:!0,excludeEnd:!0,variants:[{ +begin:"```",end:"```"},{begin:"`",end:"`"}]}];o.contains=c +;const l=r.inherit(r.TITLE_MODE,{begin:s}),d="(\\(.*\\)\\s*)?\\B[-=]>",g={ +className:"params",begin:"\\([^\\(]",returnBegin:!0,contains:[{begin:/\(/, +end:/\)/,keywords:t,contains:["self"].concat(c)}]};return{name:"CoffeeScript", +aliases:["coffee","cson","iced"],keywords:t,illegal:/\/\*/, +contains:c.concat([r.COMMENT("###","###"),r.HASH_COMMENT_MODE,{ +className:"function",begin:"^\\s*"+s+"\\s*=\\s*"+d,end:"[-=]>",returnBegin:!0, +contains:[l,g]},{begin:/[:\(,=]\s*/,relevance:0,contains:[{className:"function", +begin:d,end:"[-=]>",returnBegin:!0,contains:[g]}]},{className:"class", +beginKeywords:"class",end:"$",illegal:/[:="\[\]]/,contains:[{ +beginKeywords:"extends",endsWithParent:!0,illegal:/[:="\[\]]/,contains:[l]},l] +},{begin:s+":",end:":",returnBegin:!0,returnEnd:!0,relevance:0}])}}})()); +hljs.registerLanguage("cpp",(()=>{"use strict";function e(e){ +return t("(",e,")?")}function t(...e){return e.map((e=>{ +return(t=e)?"string"==typeof t?t:t.source:null;var t})).join("")}return n=>{ +const r=n.COMMENT("//","$",{contains:[{begin:/\\\n/}] +}),a="[a-zA-Z_]\\w*::",i="(decltype\\(auto\\)|"+e(a)+"[a-zA-Z_]\\w*"+e("<[^<>]+>")+")",s={ +className:"keyword",begin:"\\b[a-z\\d_]*_t\\b"},c={className:"string", +variants:[{begin:'(u8?|U|L)?"',end:'"',illegal:"\\n", +contains:[n.BACKSLASH_ESCAPE]},{ +begin:"(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", +end:"'",illegal:"."},n.END_SAME_AS_BEGIN({ +begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\(/,end:/\)([^()\\ ]{0,16})"/})]},o={ +className:"number",variants:[{begin:"\\b(0b[01']+)"},{ +begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)((ll|LL|l|L)(u|U)?|(u|U)(ll|LL|l|L)?|f|F|b|B)" +},{ +begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" +}],relevance:0},l={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{ +"meta-keyword":"if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" +},contains:[{begin:/\\\n/,relevance:0},n.inherit(c,{className:"meta-string"}),{ +className:"meta-string",begin:/<.*?>/},r,n.C_BLOCK_COMMENT_MODE]},d={ +className:"title",begin:e(a)+n.IDENT_RE,relevance:0 +},u=e(a)+n.IDENT_RE+"\\s*\\(",m={ +keyword:"int float while private char char8_t char16_t char32_t catch import module export virtual operator sizeof dynamic_cast|10 typedef const_cast|10 const for static_cast|10 union namespace unsigned long volatile static protected bool template mutable if public friend do goto auto void enum else break extern using asm case typeid wchar_t short reinterpret_cast|10 default double register explicit signed typename try this switch continue inline delete alignas alignof constexpr consteval constinit decltype concept co_await co_return co_yield requires noexcept static_assert thread_local restrict final override atomic_bool atomic_char atomic_schar atomic_uchar atomic_short atomic_ushort atomic_int atomic_uint atomic_long atomic_ulong atomic_llong atomic_ullong new throw return and and_eq bitand bitor compl not not_eq or or_eq xor xor_eq", +built_in:"_Bool _Complex _Imaginary", +_relevance_hints:["asin","atan2","atan","calloc","ceil","cosh","cos","exit","exp","fabs","floor","fmod","fprintf","fputs","free","frexp","auto_ptr","deque","list","queue","stack","vector","map","set","pair","bitset","multiset","multimap","unordered_set","fscanf","future","isalnum","isalpha","iscntrl","isdigit","isgraph","islower","isprint","ispunct","isspace","isupper","isxdigit","tolower","toupper","labs","ldexp","log10","log","malloc","realloc","memchr","memcmp","memcpy","memset","modf","pow","printf","putchar","puts","scanf","sinh","sin","snprintf","sprintf","sqrt","sscanf","strcat","strchr","strcmp","strcpy","strcspn","strlen","strncat","strncmp","strncpy","strpbrk","strrchr","strspn","strstr","tanh","tan","unordered_map","unordered_multiset","unordered_multimap","priority_queue","make_pair","array","shared_ptr","abort","terminate","abs","acos","vfprintf","vprintf","vsprintf","endl","initializer_list","unique_ptr","complex","imaginary","std","string","wstring","cin","cout","cerr","clog","stdin","stdout","stderr","stringstream","istringstream","ostringstream"], +literal:"true false nullptr NULL"},p={className:"function.dispatch",relevance:0, +keywords:m, +begin:t(/\b/,/(?!decltype)/,/(?!if)/,/(?!for)/,/(?!while)/,n.IDENT_RE,(_=/\s*\(/, +t("(?=",_,")")))};var _;const g=[p,l,s,r,n.C_BLOCK_COMMENT_MODE,o,c],b={ +variants:[{begin:/=/,end:/;/},{begin:/\(/,end:/\)/},{ +beginKeywords:"new throw return else",end:/;/}],keywords:m,contains:g.concat([{ +begin:/\(/,end:/\)/,keywords:m,contains:g.concat(["self"]),relevance:0}]), +relevance:0},f={className:"function",begin:"("+i+"[\\*&\\s]+)+"+u, +returnBegin:!0,end:/[{;=]/,excludeEnd:!0,keywords:m,illegal:/[^\w\s\*&:<>.]/, +contains:[{begin:"decltype\\(auto\\)",keywords:m,relevance:0},{begin:u, +returnBegin:!0,contains:[d],relevance:0},{begin:/::/,relevance:0},{begin:/:/, +endsWithParent:!0,contains:[c,o]},{className:"params",begin:/\(/,end:/\)/, +keywords:m,relevance:0,contains:[r,n.C_BLOCK_COMMENT_MODE,c,o,s,{begin:/\(/, +end:/\)/,keywords:m,relevance:0,contains:["self",r,n.C_BLOCK_COMMENT_MODE,c,o,s] +}]},s,r,n.C_BLOCK_COMMENT_MODE,l]};return{name:"C++", +aliases:["cc","c++","h++","hpp","hh","hxx","cxx"],keywords:m,illegal:"",keywords:m,contains:["self",s]},{begin:n.IDENT_RE+"::",keywords:m},{ +className:"class",beginKeywords:"enum class struct union",end:/[{;:<>=]/, +contains:[{beginKeywords:"final class struct"},n.TITLE_MODE]}]),exports:{ +preprocessor:l,strings:c,keywords:m}}}})()); +hljs.registerLanguage("csharp",(()=>{"use strict";return e=>{const n={ +keyword:["abstract","as","base","break","case","class","const","continue","do","else","event","explicit","extern","finally","fixed","for","foreach","goto","if","implicit","in","interface","internal","is","lock","namespace","new","operator","out","override","params","private","protected","public","readonly","record","ref","return","sealed","sizeof","stackalloc","static","struct","switch","this","throw","try","typeof","unchecked","unsafe","using","virtual","void","volatile","while"].concat(["add","alias","and","ascending","async","await","by","descending","equals","from","get","global","group","init","into","join","let","nameof","not","notnull","on","or","orderby","partial","remove","select","set","unmanaged","value|0","var","when","where","with","yield"]), +built_in:["bool","byte","char","decimal","delegate","double","dynamic","enum","float","int","long","nint","nuint","object","sbyte","short","string","ulong","uint","ushort"], +literal:["default","false","null","true"]},a=e.inherit(e.TITLE_MODE,{ +begin:"[a-zA-Z](\\.?\\w)*"}),i={className:"number",variants:[{ +begin:"\\b(0b[01']+)"},{ +begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{ +begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" +}],relevance:0},s={className:"string",begin:'@"',end:'"',contains:[{begin:'""'}] +},t=e.inherit(s,{illegal:/\n/}),r={className:"subst",begin:/\{/,end:/\}/, +keywords:n},l=e.inherit(r,{illegal:/\n/}),c={className:"string",begin:/\$"/, +end:'"',illegal:/\n/,contains:[{begin:/\{\{/},{begin:/\}\}/ +},e.BACKSLASH_ESCAPE,l]},o={className:"string",begin:/\$@"/,end:'"',contains:[{ +begin:/\{\{/},{begin:/\}\}/},{begin:'""'},r]},d=e.inherit(o,{illegal:/\n/, +contains:[{begin:/\{\{/},{begin:/\}\}/},{begin:'""'},l]}) +;r.contains=[o,c,s,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,i,e.C_BLOCK_COMMENT_MODE], +l.contains=[d,c,t,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,i,e.inherit(e.C_BLOCK_COMMENT_MODE,{ +illegal:/\n/})];const g={variants:[o,c,s,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE] +},E={begin:"<",end:">",contains:[{beginKeywords:"in out"},a] +},_=e.IDENT_RE+"(<"+e.IDENT_RE+"(\\s*,\\s*"+e.IDENT_RE+")*>)?(\\[\\])?",b={ +begin:"@"+e.IDENT_RE,relevance:0};return{name:"C#",aliases:["cs","c#"], +keywords:n,illegal:/::/,contains:[e.COMMENT("///","$",{returnBegin:!0, +contains:[{className:"doctag",variants:[{begin:"///",relevance:0},{ +begin:"\x3c!--|--\x3e"},{begin:""}]}] +}),e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"#", +end:"$",keywords:{ +"meta-keyword":"if else elif endif define undef warning error line region endregion pragma checksum" +}},g,i,{beginKeywords:"class interface",relevance:0,end:/[{;=]/, +illegal:/[^\s:,]/,contains:[{beginKeywords:"where class" +},a,E,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{beginKeywords:"namespace", +relevance:0,end:/[{;=]/,illegal:/[^\s:]/, +contains:[a,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{ +beginKeywords:"record",relevance:0,end:/[{;=]/,illegal:/[^\s:]/, +contains:[a,E,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{className:"meta", +begin:"^\\s*\\[",excludeBegin:!0,end:"\\]",excludeEnd:!0,contains:[{ +className:"meta-string",begin:/"/,end:/"/}]},{ +beginKeywords:"new return throw await else",relevance:0},{className:"function", +begin:"("+_+"\\s+)+"+e.IDENT_RE+"\\s*(<.+>\\s*)?\\(",returnBegin:!0, +end:/\s*[{;=]/,excludeEnd:!0,keywords:n,contains:[{ +beginKeywords:"public private protected static internal protected abstract async extern override unsafe virtual new sealed partial", +relevance:0},{begin:e.IDENT_RE+"\\s*(<.+>\\s*)?\\(",returnBegin:!0, +contains:[e.TITLE_MODE,E],relevance:0},{className:"params",begin:/\(/,end:/\)/, +excludeBegin:!0,excludeEnd:!0,keywords:n,relevance:0, +contains:[g,i,e.C_BLOCK_COMMENT_MODE] +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},b]}}})()); +hljs.registerLanguage("css",(()=>{"use strict" +;const e=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","main","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"],t=["any-hover","any-pointer","aspect-ratio","color","color-gamut","color-index","device-aspect-ratio","device-height","device-width","display-mode","forced-colors","grid","height","hover","inverted-colors","monochrome","orientation","overflow-block","overflow-inline","pointer","prefers-color-scheme","prefers-contrast","prefers-reduced-motion","prefers-reduced-transparency","resolution","scan","scripting","update","width","min-width","max-width","min-height","max-height"],i=["active","any-link","blank","checked","current","default","defined","dir","disabled","drop","empty","enabled","first","first-child","first-of-type","fullscreen","future","focus","focus-visible","focus-within","has","host","host-context","hover","indeterminate","in-range","invalid","is","lang","last-child","last-of-type","left","link","local-link","not","nth-child","nth-col","nth-last-child","nth-last-col","nth-last-of-type","nth-of-type","only-child","only-of-type","optional","out-of-range","past","placeholder-shown","read-only","read-write","required","right","root","scope","target","target-within","user-invalid","valid","visited","where"],o=["after","backdrop","before","cue","cue-region","first-letter","first-line","grammar-error","marker","part","placeholder","selection","slotted","spelling-error"],r=["align-content","align-items","align-self","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","auto","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","clear","clip","clip-path","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","content","counter-increment","counter-reset","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","font","font-display","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-smoothing","font-stretch","font-style","font-variant","font-variant-ligatures","font-variation-settings","font-weight","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inherit","initial","justify-content","left","letter-spacing","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marks","mask","max-height","max-width","min-height","min-width","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","perspective","perspective-origin","pointer-events","position","quotes","resize","right","src","tab-size","table-layout","text-align","text-align-last","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-indent","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","white-space","widows","width","word-break","word-spacing","word-wrap","z-index"].reverse() +;return n=>{const a=(e=>({IMPORTANT:{className:"meta",begin:"!important"}, +HEXCOLOR:{className:"number",begin:"#([a-fA-F0-9]{6}|[a-fA-F0-9]{3})"}, +ATTRIBUTE_SELECTOR_MODE:{className:"selector-attr",begin:/\[/,end:/\]/, +illegal:"$",contains:[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE]} +}))(n),l=[n.APOS_STRING_MODE,n.QUOTE_STRING_MODE];return{name:"CSS", +case_insensitive:!0,illegal:/[=|'\$]/,keywords:{keyframePosition:"from to"}, +classNameAliases:{keyframePosition:"selector-tag"}, +contains:[n.C_BLOCK_COMMENT_MODE,{begin:/-(webkit|moz|ms|o)-(?=[a-z])/ +},n.CSS_NUMBER_MODE,{className:"selector-id",begin:/#[A-Za-z0-9_-]+/,relevance:0 +},{className:"selector-class",begin:"\\.[a-zA-Z-][a-zA-Z0-9_-]*",relevance:0 +},a.ATTRIBUTE_SELECTOR_MODE,{className:"selector-pseudo",variants:[{ +begin:":("+i.join("|")+")"},{begin:"::("+o.join("|")+")"}]},{ +className:"attribute",begin:"\\b("+r.join("|")+")\\b"},{begin:":",end:"[;}]", +contains:[a.HEXCOLOR,a.IMPORTANT,n.CSS_NUMBER_MODE,...l,{ +begin:/(url|data-uri)\(/,end:/\)/,relevance:0,keywords:{built_in:"url data-uri" +},contains:[{className:"string",begin:/[^)]/,endsWithParent:!0,excludeEnd:!0}] +},{className:"built_in",begin:/[\w-]+(?=\()/}]},{ +begin:(s=/@/,((...e)=>e.map((e=>(e=>e?"string"==typeof e?e:e.source:null)(e))).join(""))("(?=",s,")")), +end:"[{;]",relevance:0,illegal:/:/,contains:[{className:"keyword", +begin:/@-?\w[\w]*(-\w+)*/},{begin:/\s/,endsWithParent:!0,excludeEnd:!0, +relevance:0,keywords:{$pattern:/[a-z-]+/,keyword:"and or not only", +attribute:t.join(" ")},contains:[{begin:/[a-z-]+(?=:)/,className:"attribute" +},...l,n.CSS_NUMBER_MODE]}]},{className:"selector-tag", +begin:"\\b("+e.join("|")+")\\b"}]};var s}})()); +hljs.registerLanguage("diff",(()=>{"use strict";return e=>({name:"Diff", +aliases:["patch"],contains:[{className:"meta",relevance:10,variants:[{ +begin:/^@@ +-\d+,\d+ +\+\d+,\d+ +@@/},{begin:/^\*\*\* +\d+,\d+ +\*\*\*\*$/},{ +begin:/^--- +\d+,\d+ +----$/}]},{className:"comment",variants:[{begin:/Index: /, +end:/$/},{begin:/^index/,end:/$/},{begin:/={3,}/,end:/$/},{begin:/^-{3}/,end:/$/ +},{begin:/^\*{3} /,end:/$/},{begin:/^\+{3}/,end:/$/},{begin:/^\*{15}$/},{ +begin:/^diff --git/,end:/$/}]},{className:"addition",begin:/^\+/,end:/$/},{ +className:"deletion",begin:/^-/,end:/$/},{className:"addition",begin:/^!/, +end:/$/}]})})()); +hljs.registerLanguage("go",(()=>{"use strict";return e=>{const n={ +keyword:"break default func interface select case map struct chan else goto package switch const fallthrough if range type continue for import return var go defer bool byte complex64 complex128 float32 float64 int8 int16 int32 int64 string uint8 uint16 uint32 uint64 int uint uintptr rune", +literal:"true false iota nil", +built_in:"append cap close complex copy imag len make new panic print println real recover delete" +};return{name:"Go",aliases:["golang"],keywords:n,illegal:"{"use strict";function e(...e){ +return e.map((e=>{return(n=e)?"string"==typeof n?n:n.source:null;var n +})).join("")}return n=>{const a="HTTP/(2|1\\.[01])",s={className:"attribute", +begin:e("^",/[A-Za-z][A-Za-z0-9-]*/,"(?=\\:\\s)"),starts:{contains:[{ +className:"punctuation",begin:/: /,relevance:0,starts:{end:"$",relevance:0}}]} +},t=[s,{begin:"\\n\\n",starts:{subLanguage:[],endsWithParent:!0}}];return{ +name:"HTTP",aliases:["https"],illegal:/\S/,contains:[{begin:"^(?="+a+" \\d{3})", +end:/$/,contains:[{className:"meta",begin:a},{className:"number", +begin:"\\b\\d{3}\\b"}],starts:{end:/\b\B/,illegal:/\S/,contains:t}},{ +begin:"(?=^[A-Z]+ (.*?) "+a+"$)",end:/$/,contains:[{className:"string", +begin:" ",end:" ",excludeBegin:!0,excludeEnd:!0},{className:"meta",begin:a},{ +className:"keyword",begin:"[A-Z]+"}],starts:{end:/\b\B/,illegal:/\S/,contains:t} +},n.inherit(s,{relevance:0})]}}})()); +hljs.registerLanguage("ini",(()=>{"use strict";function e(e){ +return e?"string"==typeof e?e:e.source:null}function n(...n){ +return n.map((n=>e(n))).join("")}return s=>{const a={className:"number", +relevance:0,variants:[{begin:/([+-]+)?[\d]+_[\d_]+/},{begin:s.NUMBER_RE}] +},i=s.COMMENT();i.variants=[{begin:/;/,end:/$/},{begin:/#/,end:/$/}];const t={ +className:"variable",variants:[{begin:/\$[\w\d"][\w\d_]*/},{begin:/\$\{(.*?)\}/ +}]},r={className:"literal",begin:/\bon|off|true|false|yes|no\b/},l={ +className:"string",contains:[s.BACKSLASH_ESCAPE],variants:[{begin:"'''", +end:"'''",relevance:10},{begin:'"""',end:'"""',relevance:10},{begin:'"',end:'"' +},{begin:"'",end:"'"}]},c={begin:/\[/,end:/\]/,contains:[i,r,t,l,a,"self"], +relevance:0 +},g="("+[/[A-Za-z0-9_-]+/,/"(\\"|[^"])*"/,/'[^']*'/].map((n=>e(n))).join("|")+")" +;return{name:"TOML, also INI",aliases:["toml"],case_insensitive:!0,illegal:/\S/, +contains:[i,{className:"section",begin:/\[+/,end:/\]+/},{ +begin:n(g,"(\\s*\\.\\s*",g,")*",n("(?=",/\s*=\s*[^#\s]/,")")),className:"attr", +starts:{end:/$/,contains:[i,c,r,t,l,a]}}]}}})()); +hljs.registerLanguage("java",(()=>{"use strict" +;var e="\\.([0-9](_*[0-9])*)",n="[0-9a-fA-F](_*[0-9a-fA-F])*",a={ +className:"number",variants:[{ +begin:`(\\b([0-9](_*[0-9])*)((${e})|\\.)?|(${e}))[eE][+-]?([0-9](_*[0-9])*)[fFdD]?\\b` +},{begin:`\\b([0-9](_*[0-9])*)((${e})[fFdD]?\\b|\\.([fFdD]\\b)?)`},{ +begin:`(${e})[fFdD]?\\b`},{begin:"\\b([0-9](_*[0-9])*)[fFdD]\\b"},{ +begin:`\\b0[xX]((${n})\\.?|(${n})?\\.(${n}))[pP][+-]?([0-9](_*[0-9])*)[fFdD]?\\b` +},{begin:"\\b(0|[1-9](_*[0-9])*)[lL]?\\b"},{begin:`\\b0[xX](${n})[lL]?\\b`},{ +begin:"\\b0(_*[0-7])*[lL]?\\b"},{begin:"\\b0[bB][01](_*[01])*[lL]?\\b"}], +relevance:0};return e=>{ +var n="false synchronized int abstract float private char boolean var static null if const for true while long strictfp finally protected import native final void enum else break transient catch instanceof byte super volatile case assert short package default double public try this switch continue throws protected public private module requires exports do",s={ +className:"meta",begin:"@[\xc0-\u02b8a-zA-Z_$][\xc0-\u02b8a-zA-Z_$0-9]*", +contains:[{begin:/\(/,end:/\)/,contains:["self"]}]};const r=a;return{ +name:"Java",aliases:["jsp"],keywords:n,illegal:/<\/|#/, +contains:[e.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/, +relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),{ +begin:/import java\.[a-z]+\./,keywords:"import",relevance:2 +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,{ +className:"class",beginKeywords:"class interface enum",end:/[{;=]/, +excludeEnd:!0,relevance:1,keywords:"class interface enum",illegal:/[:"\[\]]/, +contains:[{beginKeywords:"extends implements"},e.UNDERSCORE_TITLE_MODE]},{ +beginKeywords:"new throw return else",relevance:0},{className:"class", +begin:"record\\s+"+e.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:!0,excludeEnd:!0, +end:/[{;=]/,keywords:n,contains:[{beginKeywords:"record"},{ +begin:e.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:!0,relevance:0, +contains:[e.UNDERSCORE_TITLE_MODE]},{className:"params",begin:/\(/,end:/\)/, +keywords:n,relevance:0,contains:[e.C_BLOCK_COMMENT_MODE] +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{className:"function", +begin:"([\xc0-\u02b8a-zA-Z_$][\xc0-\u02b8a-zA-Z_$0-9]*(<[\xc0-\u02b8a-zA-Z_$][\xc0-\u02b8a-zA-Z_$0-9]*(\\s*,\\s*[\xc0-\u02b8a-zA-Z_$][\xc0-\u02b8a-zA-Z_$0-9]*)*>)?\\s+)+"+e.UNDERSCORE_IDENT_RE+"\\s*\\(", +returnBegin:!0,end:/[{;=]/,excludeEnd:!0,keywords:n,contains:[{ +begin:e.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:!0,relevance:0, +contains:[e.UNDERSCORE_TITLE_MODE]},{className:"params",begin:/\(/,end:/\)/, +keywords:n,relevance:0, +contains:[s,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,r,e.C_BLOCK_COMMENT_MODE] +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},r,s]}}})()); +hljs.registerLanguage("javascript",(()=>{"use strict" +;const e="[A-Za-z$_][0-9A-Za-z$_]*",n=["as","in","of","if","for","while","finally","var","new","function","do","return","void","else","break","catch","instanceof","with","throw","case","default","try","switch","continue","typeof","delete","let","yield","const","class","debugger","async","await","static","import","from","export","extends"],a=["true","false","null","undefined","NaN","Infinity"],s=[].concat(["setInterval","setTimeout","clearInterval","clearTimeout","require","exports","eval","isFinite","isNaN","parseFloat","parseInt","decodeURI","decodeURIComponent","encodeURI","encodeURIComponent","escape","unescape"],["arguments","this","super","console","window","document","localStorage","module","global"],["Intl","DataView","Number","Math","Date","String","RegExp","Object","Function","Boolean","Error","Symbol","Set","Map","WeakSet","WeakMap","Proxy","Reflect","JSON","Promise","Float64Array","Int16Array","Int32Array","Int8Array","Uint16Array","Uint32Array","Float32Array","Array","Uint8Array","Uint8ClampedArray","ArrayBuffer","BigInt64Array","BigUint64Array","BigInt"],["EvalError","InternalError","RangeError","ReferenceError","SyntaxError","TypeError","URIError"]) +;function r(e){return t("(?=",e,")")}function t(...e){return e.map((e=>{ +return(n=e)?"string"==typeof n?n:n.source:null;var n})).join("")}return i=>{ +const c=e,o={begin:/<[A-Za-z0-9\\._:-]+/,end:/\/[A-Za-z0-9\\._:-]+>|\/>/, +isTrulyOpeningTag:(e,n)=>{const a=e[0].length+e.index,s=e.input[a] +;"<"!==s?">"===s&&(((e,{after:n})=>{const a="", +returnBegin:!0,end:"\\s*=>",contains:[{className:"params",variants:[{ +begin:i.UNDERSCORE_IDENT_RE,relevance:0},{className:null,begin:/\(\s*\)/,skip:!0 +},{begin:/\(/,end:/\)/,excludeBegin:!0,excludeEnd:!0,keywords:l,contains:f}]}] +},{begin:/,/,relevance:0},{className:"",begin:/\s/,end:/\s*/,skip:!0},{ +variants:[{begin:"<>",end:""},{begin:o.begin,"on:begin":o.isTrulyOpeningTag, +end:o.end}],subLanguage:"xml",contains:[{begin:o.begin,end:o.end,skip:!0, +contains:["self"]}]}],relevance:0},{className:"function", +beginKeywords:"function",end:/[{;]/,excludeEnd:!0,keywords:l, +contains:["self",i.inherit(i.TITLE_MODE,{begin:c}),p],illegal:/%/},{ +beginKeywords:"while if switch catch for"},{className:"function", +begin:i.UNDERSCORE_IDENT_RE+"\\([^()]*(\\([^()]*(\\([^()]*\\)[^()]*)*\\)[^()]*)*\\)\\s*\\{", +returnBegin:!0,contains:[p,i.inherit(i.TITLE_MODE,{begin:c})]},{variants:[{ +begin:"\\."+c},{begin:"\\$"+c}],relevance:0},{className:"class", +beginKeywords:"class",end:/[{;=]/,excludeEnd:!0,illegal:/[:"[\]]/,contains:[{ +beginKeywords:"extends"},i.UNDERSCORE_TITLE_MODE]},{begin:/\b(?=constructor)/, +end:/[{;]/,excludeEnd:!0,contains:[i.inherit(i.TITLE_MODE,{begin:c}),"self",p] +},{begin:"(get|set)\\s+(?="+c+"\\()",end:/\{/,keywords:"get set", +contains:[i.inherit(i.TITLE_MODE,{begin:c}),{begin:/\(\)/},p]},{begin:/\$[(.]/}] +}}})()); +hljs.registerLanguage("json",(()=>{"use strict";return n=>{const e={ +literal:"true false null" +},i=[n.C_LINE_COMMENT_MODE,n.C_BLOCK_COMMENT_MODE],a=[n.QUOTE_STRING_MODE,n.C_NUMBER_MODE],l={ +end:",",endsWithParent:!0,excludeEnd:!0,contains:a,keywords:e},t={begin:/\{/, +end:/\}/,contains:[{className:"attr",begin:/"/,end:/"/, +contains:[n.BACKSLASH_ESCAPE],illegal:"\\n"},n.inherit(l,{begin:/:/ +})].concat(i),illegal:"\\S"},s={begin:"\\[",end:"\\]",contains:[n.inherit(l)], +illegal:"\\S"};return a.push(t,s),i.forEach((n=>{a.push(n)})),{name:"JSON", +contains:a,keywords:e,illegal:"\\S"}}})()); +hljs.registerLanguage("kotlin",(()=>{"use strict" +;var e="\\.([0-9](_*[0-9])*)",n="[0-9a-fA-F](_*[0-9a-fA-F])*",a={ +className:"number",variants:[{ +begin:`(\\b([0-9](_*[0-9])*)((${e})|\\.)?|(${e}))[eE][+-]?([0-9](_*[0-9])*)[fFdD]?\\b` +},{begin:`\\b([0-9](_*[0-9])*)((${e})[fFdD]?\\b|\\.([fFdD]\\b)?)`},{ +begin:`(${e})[fFdD]?\\b`},{begin:"\\b([0-9](_*[0-9])*)[fFdD]\\b"},{ +begin:`\\b0[xX]((${n})\\.?|(${n})?\\.(${n}))[pP][+-]?([0-9](_*[0-9])*)[fFdD]?\\b` +},{begin:"\\b(0|[1-9](_*[0-9])*)[lL]?\\b"},{begin:`\\b0[xX](${n})[lL]?\\b`},{ +begin:"\\b0(_*[0-7])*[lL]?\\b"},{begin:"\\b0[bB][01](_*[01])*[lL]?\\b"}], +relevance:0};return e=>{const n={ +keyword:"abstract as val var vararg get set class object open private protected public noinline crossinline dynamic final enum if else do while for when throw try catch finally import package is in fun override companion reified inline lateinit init interface annotation data sealed internal infix operator out by constructor super tailrec where const inner suspend typealias external expect actual", +built_in:"Byte Short Char Int Long Boolean Float Double Void Unit Nothing", +literal:"true false null"},i={className:"symbol",begin:e.UNDERSCORE_IDENT_RE+"@" +},s={className:"subst",begin:/\$\{/,end:/\}/,contains:[e.C_NUMBER_MODE]},t={ +className:"variable",begin:"\\$"+e.UNDERSCORE_IDENT_RE},r={className:"string", +variants:[{begin:'"""',end:'"""(?=[^"])',contains:[t,s]},{begin:"'",end:"'", +illegal:/\n/,contains:[e.BACKSLASH_ESCAPE]},{begin:'"',end:'"',illegal:/\n/, +contains:[e.BACKSLASH_ESCAPE,t,s]}]};s.contains.push(r);const l={ +className:"meta", +begin:"@(?:file|property|field|get|set|receiver|param|setparam|delegate)\\s*:(?:\\s*"+e.UNDERSCORE_IDENT_RE+")?" +},c={className:"meta",begin:"@"+e.UNDERSCORE_IDENT_RE,contains:[{begin:/\(/, +end:/\)/,contains:[e.inherit(r,{className:"meta-string"})]}] +},o=a,b=e.COMMENT("/\\*","\\*/",{contains:[e.C_BLOCK_COMMENT_MODE]}),E={ +variants:[{className:"type",begin:e.UNDERSCORE_IDENT_RE},{begin:/\(/,end:/\)/, +contains:[]}]},d=E;return d.variants[1].contains=[E],E.variants[1].contains=[d], +{name:"Kotlin",aliases:["kt","kts"],keywords:n, +contains:[e.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag", +begin:"@[A-Za-z]+"}]}),e.C_LINE_COMMENT_MODE,b,{className:"keyword", +begin:/\b(break|continue|return|this)\b/,starts:{contains:[{className:"symbol", +begin:/@\w+/}]}},i,l,c,{className:"function",beginKeywords:"fun",end:"[(]|$", +returnBegin:!0,excludeEnd:!0,keywords:n,relevance:5,contains:[{ +begin:e.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:!0,relevance:0, +contains:[e.UNDERSCORE_TITLE_MODE]},{className:"type",begin://, +keywords:"reified",relevance:0},{className:"params",begin:/\(/,end:/\)/, +endsParent:!0,keywords:n,relevance:0,contains:[{begin:/:/,end:/[=,\/]/, +endsWithParent:!0,contains:[E,e.C_LINE_COMMENT_MODE,b],relevance:0 +},e.C_LINE_COMMENT_MODE,b,l,c,r,e.C_NUMBER_MODE]},b]},{className:"class", +beginKeywords:"class interface trait",end:/[:\{(]|$/,excludeEnd:!0, +illegal:"extends implements",contains:[{ +beginKeywords:"public protected internal private constructor" +},e.UNDERSCORE_TITLE_MODE,{className:"type",begin://,excludeBegin:!0, +excludeEnd:!0,relevance:0},{className:"type",begin:/[,:]\s*/,end:/[<\(,]|$/, +excludeBegin:!0,returnEnd:!0},l,c]},r,{className:"meta",begin:"^#!/usr/bin/env", +end:"$",illegal:"\n"},o]}}})()); +hljs.registerLanguage("less",(()=>{"use strict" +;const e=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","main","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"],t=["any-hover","any-pointer","aspect-ratio","color","color-gamut","color-index","device-aspect-ratio","device-height","device-width","display-mode","forced-colors","grid","height","hover","inverted-colors","monochrome","orientation","overflow-block","overflow-inline","pointer","prefers-color-scheme","prefers-contrast","prefers-reduced-motion","prefers-reduced-transparency","resolution","scan","scripting","update","width","min-width","max-width","min-height","max-height"],i=["active","any-link","blank","checked","current","default","defined","dir","disabled","drop","empty","enabled","first","first-child","first-of-type","fullscreen","future","focus","focus-visible","focus-within","has","host","host-context","hover","indeterminate","in-range","invalid","is","lang","last-child","last-of-type","left","link","local-link","not","nth-child","nth-col","nth-last-child","nth-last-col","nth-last-of-type","nth-of-type","only-child","only-of-type","optional","out-of-range","past","placeholder-shown","read-only","read-write","required","right","root","scope","target","target-within","user-invalid","valid","visited","where"],o=["after","backdrop","before","cue","cue-region","first-letter","first-line","grammar-error","marker","part","placeholder","selection","slotted","spelling-error"],n=["align-content","align-items","align-self","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","auto","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","clear","clip","clip-path","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","content","counter-increment","counter-reset","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","font","font-display","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-smoothing","font-stretch","font-style","font-variant","font-variant-ligatures","font-variation-settings","font-weight","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inherit","initial","justify-content","left","letter-spacing","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marks","mask","max-height","max-width","min-height","min-width","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","perspective","perspective-origin","pointer-events","position","quotes","resize","right","src","tab-size","table-layout","text-align","text-align-last","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-indent","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","white-space","widows","width","word-break","word-spacing","word-wrap","z-index"].reverse(),r=i.concat(o) +;return a=>{const s=(e=>({IMPORTANT:{className:"meta",begin:"!important"}, +HEXCOLOR:{className:"number",begin:"#([a-fA-F0-9]{6}|[a-fA-F0-9]{3})"}, +ATTRIBUTE_SELECTOR_MODE:{className:"selector-attr",begin:/\[/,end:/\]/, +illegal:"$",contains:[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE]} +}))(a),l=r,d="([\\w-]+|@\\{[\\w-]+\\})",c=[],g=[],b=e=>({className:"string", +begin:"~?"+e+".*?"+e}),m=(e,t,i)=>({className:e,begin:t,relevance:i}),u={ +$pattern:/[a-z-]+/,keyword:"and or not only",attribute:t.join(" ")},p={ +begin:"\\(",end:"\\)",contains:g,keywords:u,relevance:0} +;g.push(a.C_LINE_COMMENT_MODE,a.C_BLOCK_COMMENT_MODE,b("'"),b('"'),a.CSS_NUMBER_MODE,{ +begin:"(url|data-uri)\\(",starts:{className:"string",end:"[\\)\\n]", +excludeEnd:!0} +},s.HEXCOLOR,p,m("variable","@@?[\\w-]+",10),m("variable","@\\{[\\w-]+\\}"),m("built_in","~?`[^`]*?`"),{ +className:"attribute",begin:"[\\w-]+\\s*:",end:":",returnBegin:!0,excludeEnd:!0 +},s.IMPORTANT);const f=g.concat({begin:/\{/,end:/\}/,contains:c}),h={ +beginKeywords:"when",endsWithParent:!0,contains:[{beginKeywords:"and not" +}].concat(g)},w={begin:d+"\\s*:",returnBegin:!0,end:/[;}]/,relevance:0, +contains:[{begin:/-(webkit|moz|ms|o)-/},{className:"attribute", +begin:"\\b("+n.join("|")+")\\b",end:/(?=:)/,starts:{endsWithParent:!0, +illegal:"[<=$]",relevance:0,contains:g}}]},v={className:"keyword", +begin:"@(import|media|charset|font-face|(-[a-z]+-)?keyframes|supports|document|namespace|page|viewport|host)\\b", +starts:{end:"[;{}]",keywords:u,returnEnd:!0,contains:g,relevance:0}},y={ +className:"variable",variants:[{begin:"@[\\w-]+\\s*:",relevance:15},{ +begin:"@[\\w-]+"}],starts:{end:"[;}]",returnEnd:!0,contains:f}},k={variants:[{ +begin:"[\\.#:&\\[>]",end:"[;{}]"},{begin:d,end:/\{/}],returnBegin:!0, +returnEnd:!0,illegal:"[<='$\"]",relevance:0, +contains:[a.C_LINE_COMMENT_MODE,a.C_BLOCK_COMMENT_MODE,h,m("keyword","all\\b"),m("variable","@\\{[\\w-]+\\}"),{ +begin:"\\b("+e.join("|")+")\\b",className:"selector-tag" +},m("selector-tag",d+"%?",0),m("selector-id","#"+d),m("selector-class","\\."+d,0),m("selector-tag","&",0),s.ATTRIBUTE_SELECTOR_MODE,{ +className:"selector-pseudo",begin:":("+i.join("|")+")"},{ +className:"selector-pseudo",begin:"::("+o.join("|")+")"},{begin:"\\(",end:"\\)", +contains:f},{begin:"!important"}]},E={begin:`[\\w-]+:(:)?(${l.join("|")})`, +returnBegin:!0,contains:[k]} +;return c.push(a.C_LINE_COMMENT_MODE,a.C_BLOCK_COMMENT_MODE,v,y,E,w,k),{ +name:"Less",case_insensitive:!0,illegal:"[=>'/<($\"]",contains:c}}})()); +hljs.registerLanguage("lua",(()=>{"use strict";return e=>{ +const t="\\[=*\\[",a="\\]=*\\]",n={begin:t,end:a,contains:["self"] +},o=[e.COMMENT("--(?!\\[=*\\[)","$"),e.COMMENT("--\\[=*\\[",a,{contains:[n], +relevance:10})];return{name:"Lua",keywords:{$pattern:e.UNDERSCORE_IDENT_RE, +literal:"true false nil", +keyword:"and break do else elseif end for goto if in local not or repeat return then until while", +built_in:"_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len __gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstring module next pairs pcall print rawequal rawget rawset require select setfenv setmetatable tonumber tostring type unpack xpcall arg self coroutine resume yield status wrap create running debug getupvalue debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv io lines write close flush open output type read stderr stdin input stdout popen tmpfile math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower table setn insert getn foreachi maxn foreach concat sort remove" +},contains:o.concat([{className:"function",beginKeywords:"function",end:"\\)", +contains:[e.inherit(e.TITLE_MODE,{ +begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),{className:"params", +begin:"\\(",endsWithParent:!0,contains:o}].concat(o) +},e.C_NUMBER_MODE,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,{className:"string", +begin:t,end:a,contains:[n],relevance:5}])}}})()); +hljs.registerLanguage("makefile",(()=>{"use strict";return e=>{const i={ +className:"variable",variants:[{begin:"\\$\\("+e.UNDERSCORE_IDENT_RE+"\\)", +contains:[e.BACKSLASH_ESCAPE]},{begin:/\$[@%{"use strict";function e(e){ +return e?"string"==typeof e?e:e.source:null}function n(e){return a("(?=",e,")")} +function a(...n){return n.map((n=>e(n))).join("")}function s(...n){ +return"("+n.map((n=>e(n))).join("|")+")"}return e=>{ +const t=a(/[A-Z_]/,a("(",/[A-Z0-9_.-]*:/,")?"),/[A-Z0-9_.-]*/),i={ +className:"symbol",begin:/&[a-z]+;|&#[0-9]+;|&#x[a-f0-9]+;/},r={begin:/\s/, +contains:[{className:"meta-keyword",begin:/#?[a-z_][a-z1-9_-]+/,illegal:/\n/}] +},c=e.inherit(r,{begin:/\(/,end:/\)/}),l=e.inherit(e.APOS_STRING_MODE,{ +className:"meta-string"}),g=e.inherit(e.QUOTE_STRING_MODE,{ +className:"meta-string"}),m={endsWithParent:!0,illegal:/`]+/}]}] +}]};return{name:"HTML, XML", +aliases:["html","xhtml","rss","atom","xjb","xsd","xsl","plist","wsf","svg"], +case_insensitive:!0,contains:[{className:"meta",begin://, +relevance:10,contains:[r,g,l,c,{begin:/\[/,end:/\]/,contains:[{className:"meta", +begin://,contains:[r,c,g,l]}]}]},e.COMMENT(//,{ +relevance:10}),{begin://,relevance:10},i,{ +className:"meta",begin:/<\?xml/,end:/\?>/,relevance:10},{className:"tag", +begin:/)/,end:/>/,keywords:{name:"style"},contains:[m],starts:{ +end:/<\/style>/,returnEnd:!0,subLanguage:["css","xml"]}},{className:"tag", +begin:/)/,end:/>/,keywords:{name:"script"},contains:[m],starts:{ +end:/<\/script>/,returnEnd:!0,subLanguage:["javascript","handlebars","xml"]}},{ +className:"tag",begin:/<>|<\/>/},{className:"tag", +begin:a(//,/>/,/\s/)))),end:/\/?>/,contains:[{className:"name", +begin:t,relevance:0,starts:m}]},{className:"tag",begin:a(/<\//,n(a(t,/>/))), +contains:[{className:"name",begin:t,relevance:0},{begin:/>/,relevance:0, +endsParent:!0}]}]}}})()); +hljs.registerLanguage("markdown",(()=>{"use strict";function n(...n){ +return n.map((n=>{return(e=n)?"string"==typeof e?e:e.source:null;var e +})).join("")}return e=>{const a={begin:/<\/?[A-Za-z_]/,end:">", +subLanguage:"xml",relevance:0},i={variants:[{begin:/\[.+?\]\[.*?\]/,relevance:0 +},{begin:/\[.+?\]\(((data|javascript|mailto):|(?:http|ftp)s?:\/\/).*?\)/, +relevance:2},{begin:n(/\[.+?\]\(/,/[A-Za-z][A-Za-z0-9+.-]*/,/:\/\/.*?\)/), +relevance:2},{begin:/\[.+?\]\([./?&#].*?\)/,relevance:1},{ +begin:/\[.+?\]\(.*?\)/,relevance:0}],returnBegin:!0,contains:[{ +className:"string",relevance:0,begin:"\\[",end:"\\]",excludeBegin:!0, +returnEnd:!0},{className:"link",relevance:0,begin:"\\]\\(",end:"\\)", +excludeBegin:!0,excludeEnd:!0},{className:"symbol",relevance:0,begin:"\\]\\[", +end:"\\]",excludeBegin:!0,excludeEnd:!0}]},s={className:"strong",contains:[], +variants:[{begin:/_{2}/,end:/_{2}/},{begin:/\*{2}/,end:/\*{2}/}]},c={ +className:"emphasis",contains:[],variants:[{begin:/\*(?!\*)/,end:/\*/},{ +begin:/_(?!_)/,end:/_/,relevance:0}]};s.contains.push(c),c.contains.push(s) +;let t=[a,i] +;return s.contains=s.contains.concat(t),c.contains=c.contains.concat(t), +t=t.concat(s,c),{name:"Markdown",aliases:["md","mkdown","mkd"],contains:[{ +className:"section",variants:[{begin:"^#{1,6}",end:"$",contains:t},{ +begin:"(?=^.+?\\n[=-]{2,}$)",contains:[{begin:"^[=-]*$"},{begin:"^",end:"\\n", +contains:t}]}]},a,{className:"bullet",begin:"^[ \t]*([*+-]|(\\d+\\.))(?=\\s+)", +end:"\\s+",excludeEnd:!0},s,c,{className:"quote",begin:"^>\\s+",contains:t, +end:"$"},{className:"code",variants:[{begin:"(`{3,})[^`](.|\\n)*?\\1`*[ ]*"},{ +begin:"(~{3,})[^~](.|\\n)*?\\1~*[ ]*"},{begin:"```",end:"```+[ ]*$"},{ +begin:"~~~",end:"~~~+[ ]*$"},{begin:"`.+?`"},{begin:"(?=^( {4}|\\t))", +contains:[{begin:"^( {4}|\\t)",end:"(\\n)$"}],relevance:0}]},{ +begin:"^[-\\*]{3,}",end:"$"},i,{begin:/^\[[^\n]+\]:/,returnBegin:!0,contains:[{ +className:"symbol",begin:/\[/,end:/\]/,excludeBegin:!0,excludeEnd:!0},{ +className:"link",begin:/:\s*/,end:/$/,excludeBegin:!0}]}]}}})()); +hljs.registerLanguage("nginx",(()=>{"use strict";return e=>{const n={ +className:"variable",variants:[{begin:/\$\d+/},{begin:/\$\{/,end:/\}/},{ +begin:/[$@]/+e.UNDERSCORE_IDENT_RE}]},a={endsWithParent:!0,keywords:{ +$pattern:"[a-z/_]+", +literal:"on off yes no true false none blocked debug info notice warn error crit select break last permanent redirect kqueue rtsig epoll poll /dev/poll" +},relevance:0,illegal:"=>",contains:[e.HASH_COMMENT_MODE,{className:"string", +contains:[e.BACKSLASH_ESCAPE,n],variants:[{begin:/"/,end:/"/},{begin:/'/,end:/'/ +}]},{begin:"([a-z]+):/",end:"\\s",endsWithParent:!0,excludeEnd:!0,contains:[n] +},{className:"regexp",contains:[e.BACKSLASH_ESCAPE,n],variants:[{begin:"\\s\\^", +end:"\\s|\\{|;",returnEnd:!0},{begin:"~\\*?\\s+",end:"\\s|\\{|;",returnEnd:!0},{ +begin:"\\*(\\.[a-z\\-]+)+"},{begin:"([a-z\\-]+\\.)+\\*"}]},{className:"number", +begin:"\\b\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}(:\\d{1,5})?\\b"},{ +className:"number",begin:"\\b\\d+[kKmMgGdshdwy]*\\b",relevance:0},n]};return{ +name:"Nginx config",aliases:["nginxconf"],contains:[e.HASH_COMMENT_MODE,{ +begin:e.UNDERSCORE_IDENT_RE+"\\s+\\{",returnBegin:!0,end:/\{/,contains:[{ +className:"section",begin:e.UNDERSCORE_IDENT_RE}],relevance:0},{ +begin:e.UNDERSCORE_IDENT_RE+"\\s",end:";|\\{",returnBegin:!0,contains:[{ +className:"attribute",begin:e.UNDERSCORE_IDENT_RE,starts:a}],relevance:0}], +illegal:"[^\\s\\}]"}}})()); +hljs.registerLanguage("objectivec",(()=>{"use strict";return e=>{ +const n=/[a-zA-Z@][a-zA-Z0-9_]*/,_={$pattern:n, +keyword:"@interface @class @protocol @implementation"};return{ +name:"Objective-C",aliases:["mm","objc","obj-c","obj-c++","objective-c++"], +keywords:{$pattern:n, +keyword:"int float while char export sizeof typedef const struct for union unsigned long volatile static bool mutable if do return goto void enum else break extern asm case short default double register explicit signed typename this switch continue wchar_t inline readonly assign readwrite self @synchronized id typeof nonatomic super unichar IBOutlet IBAction strong weak copy in out inout bycopy byref oneway __strong __weak __block __autoreleasing @private @protected @public @try @property @end @throw @catch @finally @autoreleasepool @synthesize @dynamic @selector @optional @required @encode @package @import @defs @compatibility_alias __bridge __bridge_transfer __bridge_retained __bridge_retain __covariant __contravariant __kindof _Nonnull _Nullable _Null_unspecified __FUNCTION__ __PRETTY_FUNCTION__ __attribute__ getter setter retain unsafe_unretained nonnull nullable null_unspecified null_resettable class instancetype NS_DESIGNATED_INITIALIZER NS_UNAVAILABLE NS_REQUIRES_SUPER NS_RETURNS_INNER_POINTER NS_INLINE NS_AVAILABLE NS_DEPRECATED NS_ENUM NS_OPTIONS NS_SWIFT_UNAVAILABLE NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_END NS_REFINED_FOR_SWIFT NS_SWIFT_NAME NS_SWIFT_NOTHROW NS_DURING NS_HANDLER NS_ENDHANDLER NS_VALUERETURN NS_VOIDRETURN", +literal:"false true FALSE TRUE nil YES NO NULL", +built_in:"BOOL dispatch_once_t dispatch_queue_t dispatch_sync dispatch_async dispatch_once" +},illegal:"/,end:/$/, +illegal:"\\n"},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{ +className:"class",begin:"("+_.keyword.split(" ").join("|")+")\\b",end:/(\{|$)/, +excludeEnd:!0,keywords:_,contains:[e.UNDERSCORE_TITLE_MODE]},{ +begin:"\\."+e.UNDERSCORE_IDENT_RE,relevance:0}]}}})()); +hljs.registerLanguage("perl",(()=>{"use strict";function e(e){ +return e?"string"==typeof e?e:e.source:null}function n(...n){ +return n.map((n=>e(n))).join("")}function t(...n){ +return"("+n.map((n=>e(n))).join("|")+")"}return e=>{ +const r=/[dualxmsipngr]{0,12}/,s={$pattern:/[\w.]+/, +keyword:"abs accept alarm and atan2 bind binmode bless break caller chdir chmod chomp chop chown chr chroot close closedir connect continue cos crypt dbmclose dbmopen defined delete die do dump each else elsif endgrent endhostent endnetent endprotoent endpwent endservent eof eval exec exists exit exp fcntl fileno flock for foreach fork format formline getc getgrent getgrgid getgrnam gethostbyaddr gethostbyname gethostent getlogin getnetbyaddr getnetbyname getnetent getpeername getpgrp getpriority getprotobyname getprotobynumber getprotoent getpwent getpwnam getpwuid getservbyname getservbyport getservent getsockname getsockopt given glob gmtime goto grep gt hex if index int ioctl join keys kill last lc lcfirst length link listen local localtime log lstat lt ma map mkdir msgctl msgget msgrcv msgsnd my ne next no not oct open opendir or ord our pack package pipe pop pos print printf prototype push q|0 qq quotemeta qw qx rand read readdir readline readlink readpipe recv redo ref rename require reset return reverse rewinddir rindex rmdir say scalar seek seekdir select semctl semget semop send setgrent sethostent setnetent setpgrp setpriority setprotoent setpwent setservent setsockopt shift shmctl shmget shmread shmwrite shutdown sin sleep socket socketpair sort splice split sprintf sqrt srand stat state study sub substr symlink syscall sysopen sysread sysseek system syswrite tell telldir tie tied time times tr truncate uc ucfirst umask undef unless unlink unpack unshift untie until use utime values vec wait waitpid wantarray warn when while write x|0 xor y|0" +},i={className:"subst",begin:"[$@]\\{",end:"\\}",keywords:s},a={begin:/->\{/, +end:/\}/},o={variants:[{begin:/\$\d/},{ +begin:n(/[$%@](\^\w\b|#\w+(::\w+)*|\{\w+\}|\w+(::\w*)*)/,"(?![A-Za-z])(?![@$%])") +},{begin:/[$%@][^\s\w{]/,relevance:0}] +},c=[e.BACKSLASH_ESCAPE,i,o],g=[/!/,/\//,/\|/,/\?/,/'/,/"/,/#/],l=(e,t,s="\\1")=>{ +const i="\\1"===s?s:n(s,t) +;return n(n("(?:",e,")"),t,/(?:\\.|[^\\\/])*?/,i,/(?:\\.|[^\\\/])*?/,s,r) +},d=(e,t,s)=>n(n("(?:",e,")"),t,/(?:\\.|[^\\\/])*?/,s,r),p=[o,e.HASH_COMMENT_MODE,e.COMMENT(/^=\w/,/=cut/,{ +endsWithParent:!0}),a,{className:"string",contains:c,variants:[{ +begin:"q[qwxr]?\\s*\\(",end:"\\)",relevance:5},{begin:"q[qwxr]?\\s*\\[", +end:"\\]",relevance:5},{begin:"q[qwxr]?\\s*\\{",end:"\\}",relevance:5},{ +begin:"q[qwxr]?\\s*\\|",end:"\\|",relevance:5},{begin:"q[qwxr]?\\s*<",end:">", +relevance:5},{begin:"qw\\s+q",end:"q",relevance:5},{begin:"'",end:"'", +contains:[e.BACKSLASH_ESCAPE]},{begin:'"',end:'"'},{begin:"`",end:"`", +contains:[e.BACKSLASH_ESCAPE]},{begin:/\{\w+\}/,relevance:0},{ +begin:"-?\\w+\\s*=>",relevance:0}]},{className:"number", +begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b", +relevance:0},{ +begin:"(\\/\\/|"+e.RE_STARTERS_RE+"|\\b(split|return|print|reverse|grep)\\b)\\s*", +keywords:"split return print reverse grep",relevance:0, +contains:[e.HASH_COMMENT_MODE,{className:"regexp",variants:[{ +begin:l("s|tr|y",t(...g))},{begin:l("s|tr|y","\\(","\\)")},{ +begin:l("s|tr|y","\\[","\\]")},{begin:l("s|tr|y","\\{","\\}")}],relevance:2},{ +className:"regexp",variants:[{begin:/(m|qr)\/\//,relevance:0},{ +begin:d("(?:m|qr)?",/\//,/\//)},{begin:d("m|qr",t(...g),/\1/)},{ +begin:d("m|qr",/\(/,/\)/)},{begin:d("m|qr",/\[/,/\]/)},{ +begin:d("m|qr",/\{/,/\}/)}]}]},{className:"function",beginKeywords:"sub", +end:"(\\s*\\(.*?\\))?[;{]",excludeEnd:!0,relevance:5,contains:[e.TITLE_MODE]},{ +begin:"-\\w\\b",relevance:0},{begin:"^__DATA__$",end:"^__END__$", +subLanguage:"mojolicious",contains:[{begin:"^@@.*",end:"$",className:"comment"}] +}];return i.contains=p,a.contains=p,{name:"Perl",aliases:["pl","pm"],keywords:s, +contains:p}}})()); +hljs.registerLanguage("php",(()=>{"use strict";return e=>{const r={ +className:"variable", +begin:"\\$+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*(?![A-Za-z0-9])(?![$])"},t={ +className:"meta",variants:[{begin:/<\?php/,relevance:10},{begin:/<\?[=]?/},{ +begin:/\?>/}]},a={className:"subst",variants:[{begin:/\$\w+/},{begin:/\{\$/, +end:/\}/}]},n=e.inherit(e.APOS_STRING_MODE,{illegal:null +}),i=e.inherit(e.QUOTE_STRING_MODE,{illegal:null, +contains:e.QUOTE_STRING_MODE.contains.concat(a)}),o=e.END_SAME_AS_BEGIN({ +begin:/<<<[ \t]*(\w+)\n/,end:/[ \t]*(\w+)\b/, +contains:e.QUOTE_STRING_MODE.contains.concat(a)}),l={className:"string", +contains:[e.BACKSLASH_ESCAPE,t],variants:[e.inherit(n,{begin:"b'",end:"'" +}),e.inherit(i,{begin:'b"',end:'"'}),i,n,o]},s={className:"number",variants:[{ +begin:"\\b0b[01]+(?:_[01]+)*\\b"},{begin:"\\b0o[0-7]+(?:_[0-7]+)*\\b"},{ +begin:"\\b0x[\\da-f]+(?:_[\\da-f]+)*\\b"},{ +begin:"(?:\\b\\d+(?:_\\d+)*(\\.(?:\\d+(?:_\\d+)*))?|\\B\\.\\d+)(?:e[+-]?\\d+)?" +}],relevance:0},c={ +keyword:"__CLASS__ __DIR__ __FILE__ __FUNCTION__ __LINE__ __METHOD__ __NAMESPACE__ __TRAIT__ die echo exit include include_once print require require_once array abstract and as binary bool boolean break callable case catch class clone const continue declare default do double else elseif empty enddeclare endfor endforeach endif endswitch endwhile enum eval extends final finally float for foreach from global goto if implements instanceof insteadof int integer interface isset iterable list match|0 mixed new object or private protected public real return string switch throw trait try unset use var void while xor yield", +literal:"false null true", +built_in:"Error|0 AppendIterator ArgumentCountError ArithmeticError ArrayIterator ArrayObject AssertionError BadFunctionCallException BadMethodCallException CachingIterator CallbackFilterIterator CompileError Countable DirectoryIterator DivisionByZeroError DomainException EmptyIterator ErrorException Exception FilesystemIterator FilterIterator GlobIterator InfiniteIterator InvalidArgumentException IteratorIterator LengthException LimitIterator LogicException MultipleIterator NoRewindIterator OutOfBoundsException OutOfRangeException OuterIterator OverflowException ParentIterator ParseError RangeException RecursiveArrayIterator RecursiveCachingIterator RecursiveCallbackFilterIterator RecursiveDirectoryIterator RecursiveFilterIterator RecursiveIterator RecursiveIteratorIterator RecursiveRegexIterator RecursiveTreeIterator RegexIterator RuntimeException SeekableIterator SplDoublyLinkedList SplFileInfo SplFileObject SplFixedArray SplHeap SplMaxHeap SplMinHeap SplObjectStorage SplObserver SplObserver SplPriorityQueue SplQueue SplStack SplSubject SplSubject SplTempFileObject TypeError UnderflowException UnexpectedValueException UnhandledMatchError ArrayAccess Closure Generator Iterator IteratorAggregate Serializable Stringable Throwable Traversable WeakReference WeakMap Directory __PHP_Incomplete_Class parent php_user_filter self static stdClass" +};return{aliases:["php3","php4","php5","php6","php7","php8"], +case_insensitive:!0,keywords:c, +contains:[e.HASH_COMMENT_MODE,e.COMMENT("//","$",{contains:[t] +}),e.COMMENT("/\\*","\\*/",{contains:[{className:"doctag",begin:"@[A-Za-z]+"}] +}),e.COMMENT("__halt_compiler.+?;",!1,{endsWithParent:!0, +keywords:"__halt_compiler"}),t,{className:"keyword",begin:/\$this\b/},r,{ +begin:/(::|->)+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/},{className:"function", +relevance:0,beginKeywords:"fn function",end:/[;{]/,excludeEnd:!0, +illegal:"[$%\\[]",contains:[{beginKeywords:"use"},e.UNDERSCORE_TITLE_MODE,{ +begin:"=>",endsParent:!0},{className:"params",begin:"\\(",end:"\\)", +excludeBegin:!0,excludeEnd:!0,keywords:c, +contains:["self",r,e.C_BLOCK_COMMENT_MODE,l,s]}]},{className:"class",variants:[{ +beginKeywords:"enum",illegal:/[($"]/},{beginKeywords:"class interface trait", +illegal:/[:($"]/}],relevance:0,end:/\{/,excludeEnd:!0,contains:[{ +beginKeywords:"extends implements"},e.UNDERSCORE_TITLE_MODE]},{ +beginKeywords:"namespace",relevance:0,end:";",illegal:/[.']/, +contains:[e.UNDERSCORE_TITLE_MODE]},{beginKeywords:"use",relevance:0,end:";", +contains:[e.UNDERSCORE_TITLE_MODE]},l,s]}}})()); +hljs.registerLanguage("php-template",(()=>{"use strict";return n=>({ +name:"PHP template",subLanguage:"xml",contains:[{begin:/<\?(php|=)?/,end:/\?>/, +subLanguage:"php",contains:[{begin:"/\\*",end:"\\*/",skip:!0},{begin:'b"', +end:'"',skip:!0},{begin:"b'",end:"'",skip:!0},n.inherit(n.APOS_STRING_MODE,{ +illegal:null,className:null,contains:null,skip:!0 +}),n.inherit(n.QUOTE_STRING_MODE,{illegal:null,className:null,contains:null, +skip:!0})]}]})})()); +hljs.registerLanguage("plaintext",(()=>{"use strict";return t=>({ +name:"Plain text",aliases:["text","txt"],disableAutodetect:!0})})()); +hljs.registerLanguage("properties",(()=>{"use strict";return e=>{ +var n="[ \\t\\f]*",a=n+"[:=]"+n,t="("+a+"|[ \\t\\f]+)",r="([^\\\\\\W:= \\t\\f\\n]|\\\\.)+",s="([^\\\\:= \\t\\f\\n]|\\\\.)+",i={ +end:t,relevance:0,starts:{className:"string",end:/$/,relevance:0,contains:[{ +begin:"\\\\\\\\"},{begin:"\\\\\\n"}]}};return{name:".properties", +case_insensitive:!0,illegal:/\S/,contains:[e.COMMENT("^\\s*[!#]","$"),{ +returnBegin:!0,variants:[{begin:r+a,relevance:1},{begin:r+"[ \\t\\f]+", +relevance:0}],contains:[{className:"attr",begin:r,endsParent:!0,relevance:0}], +starts:i},{begin:s+t,returnBegin:!0,relevance:0,contains:[{className:"meta", +begin:s,endsParent:!0,relevance:0}],starts:i},{className:"attr",relevance:0, +begin:s+n+"$"}]}}})()); +hljs.registerLanguage("python",(()=>{"use strict";return e=>{const n={ +$pattern:/[A-Za-z]\w+|__\w+__/, +keyword:["and","as","assert","async","await","break","class","continue","def","del","elif","else","except","finally","for","from","global","if","import","in","is","lambda","nonlocal|10","not","or","pass","raise","return","try","while","with","yield"], +built_in:["__import__","abs","all","any","ascii","bin","bool","breakpoint","bytearray","bytes","callable","chr","classmethod","compile","complex","delattr","dict","dir","divmod","enumerate","eval","exec","filter","float","format","frozenset","getattr","globals","hasattr","hash","help","hex","id","input","int","isinstance","issubclass","iter","len","list","locals","map","max","memoryview","min","next","object","oct","open","ord","pow","print","property","range","repr","reversed","round","set","setattr","slice","sorted","staticmethod","str","sum","super","tuple","type","vars","zip"], +literal:["__debug__","Ellipsis","False","None","NotImplemented","True"], +type:["Any","Callable","Coroutine","Dict","List","Literal","Generic","Optional","Sequence","Set","Tuple","Type","Union"] +},a={className:"meta",begin:/^(>>>|\.\.\.) /},i={className:"subst",begin:/\{/, +end:/\}/,keywords:n,illegal:/#/},s={begin:/\{\{/,relevance:0},t={ +className:"string",contains:[e.BACKSLASH_ESCAPE],variants:[{ +begin:/([uU]|[bB]|[rR]|[bB][rR]|[rR][bB])?'''/,end:/'''/, +contains:[e.BACKSLASH_ESCAPE,a],relevance:10},{ +begin:/([uU]|[bB]|[rR]|[bB][rR]|[rR][bB])?"""/,end:/"""/, +contains:[e.BACKSLASH_ESCAPE,a],relevance:10},{ +begin:/([fF][rR]|[rR][fF]|[fF])'''/,end:/'''/, +contains:[e.BACKSLASH_ESCAPE,a,s,i]},{begin:/([fF][rR]|[rR][fF]|[fF])"""/, +end:/"""/,contains:[e.BACKSLASH_ESCAPE,a,s,i]},{begin:/([uU]|[rR])'/,end:/'/, +relevance:10},{begin:/([uU]|[rR])"/,end:/"/,relevance:10},{ +begin:/([bB]|[bB][rR]|[rR][bB])'/,end:/'/},{begin:/([bB]|[bB][rR]|[rR][bB])"/, +end:/"/},{begin:/([fF][rR]|[rR][fF]|[fF])'/,end:/'/, +contains:[e.BACKSLASH_ESCAPE,s,i]},{begin:/([fF][rR]|[rR][fF]|[fF])"/,end:/"/, +contains:[e.BACKSLASH_ESCAPE,s,i]},e.APOS_STRING_MODE,e.QUOTE_STRING_MODE] +},r="[0-9](_?[0-9])*",l=`(\\b(${r}))?\\.(${r})|\\b(${r})\\.`,b={ +className:"number",relevance:0,variants:[{ +begin:`(\\b(${r})|(${l}))[eE][+-]?(${r})[jJ]?\\b`},{begin:`(${l})[jJ]?`},{ +begin:"\\b([1-9](_?[0-9])*|0+(_?0)*)[lLjJ]?\\b"},{ +begin:"\\b0[bB](_?[01])+[lL]?\\b"},{begin:"\\b0[oO](_?[0-7])+[lL]?\\b"},{ +begin:"\\b0[xX](_?[0-9a-fA-F])+[lL]?\\b"},{begin:`\\b(${r})[jJ]\\b`}]},o={ +className:"comment", +begin:(d=/# type:/,((...e)=>e.map((e=>(e=>e?"string"==typeof e?e:e.source:null)(e))).join(""))("(?=",d,")")), +end:/$/,keywords:n,contains:[{begin:/# type:/},{begin:/#/,end:/\b\B/, +endsWithParent:!0}]},c={className:"params",variants:[{className:"", +begin:/\(\s*\)/,skip:!0},{begin:/\(/,end:/\)/,excludeBegin:!0,excludeEnd:!0, +keywords:n,contains:["self",a,b,t,e.HASH_COMMENT_MODE]}]};var d +;return i.contains=[t,b,a],{name:"Python",aliases:["py","gyp","ipython"], +keywords:n,illegal:/(<\/|->|\?)|=>/,contains:[a,b,{begin:/\bself\b/},{ +beginKeywords:"if",relevance:0},t,o,e.HASH_COMMENT_MODE,{variants:[{ +className:"function",beginKeywords:"def"},{className:"class", +beginKeywords:"class"}],end:/:/,illegal:/[${=;\n,]/, +contains:[e.UNDERSCORE_TITLE_MODE,c,{begin:/->/,endsWithParent:!0,keywords:n}] +},{className:"meta",begin:/^[\t ]*@/,end:/(?=#)|$/,contains:[b,c,t]}]}}})()); +hljs.registerLanguage("python-repl",(()=>{"use strict";return s=>({ +aliases:["pycon"],contains:[{className:"meta",starts:{end:/ |$/,starts:{end:"$", +subLanguage:"python"}},variants:[{begin:/^>>>(?=[ ]|$)/},{ +begin:/^\.\.\.(?=[ ]|$)/}]}]})})()); +hljs.registerLanguage("r",(()=>{"use strict";function e(...e){return e.map((e=>{ +return(a=e)?"string"==typeof a?a:a.source:null;var a})).join("")}return a=>{ +const n=/(?:(?:[a-zA-Z]|\.[._a-zA-Z])[._a-zA-Z0-9]*)|\.(?!\d)/;return{name:"R", +illegal:/->/,keywords:{$pattern:n, +keyword:"function if in break next repeat else for while", +literal:"NULL NA TRUE FALSE Inf NaN NA_integer_|10 NA_real_|10 NA_character_|10 NA_complex_|10", +built_in:"LETTERS letters month.abb month.name pi T F abs acos acosh all any anyNA Arg as.call as.character as.complex as.double as.environment as.integer as.logical as.null.default as.numeric as.raw asin asinh atan atanh attr attributes baseenv browser c call ceiling class Conj cos cosh cospi cummax cummin cumprod cumsum digamma dim dimnames emptyenv exp expression floor forceAndCall gamma gc.time globalenv Im interactive invisible is.array is.atomic is.call is.character is.complex is.double is.environment is.expression is.finite is.function is.infinite is.integer is.language is.list is.logical is.matrix is.na is.name is.nan is.null is.numeric is.object is.pairlist is.raw is.recursive is.single is.symbol lazyLoadDBfetch length lgamma list log max min missing Mod names nargs nzchar oldClass on.exit pos.to.env proc.time prod quote range Re rep retracemem return round seq_along seq_len seq.int sign signif sin sinh sinpi sqrt standardGeneric substitute sum switch tan tanh tanpi tracemem trigamma trunc unclass untracemem UseMethod xtfrm" +},compilerExtensions:[(a,n)=>{if(!a.beforeMatch)return +;if(a.starts)throw Error("beforeMatch cannot be used with starts") +;const i=Object.assign({},a);Object.keys(a).forEach((e=>{delete a[e] +})),a.begin=e(i.beforeMatch,e("(?=",i.begin,")")),a.starts={relevance:0, +contains:[Object.assign(i,{endsParent:!0})]},a.relevance=0,delete i.beforeMatch +}],contains:[a.COMMENT(/#'/,/$/,{contains:[{className:"doctag", +begin:"@examples",starts:{contains:[{begin:/\n/},{begin:/#'\s*(?=@[a-zA-Z]+)/, +endsParent:!0},{begin:/#'/,end:/$/,excludeBegin:!0}]}},{className:"doctag", +begin:"@param",end:/$/,contains:[{className:"variable",variants:[{begin:n},{ +begin:/`(?:\\.|[^`\\])+`/}],endsParent:!0}]},{className:"doctag", +begin:/@[a-zA-Z]+/},{className:"meta-keyword",begin:/\\[a-zA-Z]+/}] +}),a.HASH_COMMENT_MODE,{className:"string",contains:[a.BACKSLASH_ESCAPE], +variants:[a.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\(/,end:/\)(-*)"/ +}),a.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\{/,end:/\}(-*)"/ +}),a.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\[/,end:/\](-*)"/ +}),a.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\(/,end:/\)(-*)'/ +}),a.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\{/,end:/\}(-*)'/ +}),a.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\[/,end:/\](-*)'/}),{begin:'"',end:'"', +relevance:0},{begin:"'",end:"'",relevance:0}]},{className:"number",relevance:0, +beforeMatch:/([^a-zA-Z0-9._])/,variants:[{ +match:/0[xX][0-9a-fA-F]+\.[0-9a-fA-F]*[pP][+-]?\d+i?/},{ +match:/0[xX][0-9a-fA-F]+([pP][+-]?\d+)?[Li]?/},{ +match:/(\d+(\.\d*)?|\.\d+)([eE][+-]?\d+)?[Li]?/}]},{begin:"%",end:"%"},{ +begin:e(/[a-zA-Z][a-zA-Z_0-9]*/,"\\s+<-\\s+")},{begin:"`",end:"`",contains:[{ +begin:/\\./}]}]}}})()); +hljs.registerLanguage("ruby",(()=>{"use strict";function e(...e){ +return e.map((e=>{return(n=e)?"string"==typeof n?n:n.source:null;var n +})).join("")}return n=>{ +const a="([a-zA-Z_]\\w*[!?=]?|[-+~]@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?)",i={ +keyword:"and then defined module in return redo if BEGIN retry end for self when next until do begin unless END rescue else break undef not super class case require yield alias while ensure elsif or include attr_reader attr_writer attr_accessor __FILE__", +built_in:"proc lambda",literal:"true false nil"},s={className:"doctag", +begin:"@[A-Za-z]+"},r={begin:"#<",end:">"},b=[n.COMMENT("#","$",{contains:[s] +}),n.COMMENT("^=begin","^=end",{contains:[s],relevance:10 +}),n.COMMENT("^__END__","\\n$")],c={className:"subst",begin:/#\{/,end:/\}/, +keywords:i},t={className:"string",contains:[n.BACKSLASH_ESCAPE,c],variants:[{ +begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{begin:/%[qQwWx]?\(/, +end:/\)/},{begin:/%[qQwWx]?\[/,end:/\]/},{begin:/%[qQwWx]?\{/,end:/\}/},{ +begin:/%[qQwWx]?/},{begin:/%[qQwWx]?\//,end:/\//},{begin:/%[qQwWx]?%/, +end:/%/},{begin:/%[qQwWx]?-/,end:/-/},{begin:/%[qQwWx]?\|/,end:/\|/},{ +begin:/\B\?(\\\d{1,3})/},{begin:/\B\?(\\x[A-Fa-f0-9]{1,2})/},{ +begin:/\B\?(\\u\{?[A-Fa-f0-9]{1,6}\}?)/},{ +begin:/\B\?(\\M-\\C-|\\M-\\c|\\c\\M-|\\M-|\\C-\\M-)[\x20-\x7e]/},{ +begin:/\B\?\\(c|C-)[\x20-\x7e]/},{begin:/\B\?\\?\S/},{ +begin:/<<[-~]?'?(\w+)\n(?:[^\n]*\n)*?\s*\1\b/,returnBegin:!0,contains:[{ +begin:/<<[-~]?'?/},n.END_SAME_AS_BEGIN({begin:/(\w+)/,end:/(\w+)/, +contains:[n.BACKSLASH_ESCAPE,c]})]}]},g="[0-9](_?[0-9])*",d={className:"number", +relevance:0,variants:[{ +begin:`\\b([1-9](_?[0-9])*|0)(\\.(${g}))?([eE][+-]?(${g})|r)?i?\\b`},{ +begin:"\\b0[dD][0-9](_?[0-9])*r?i?\\b"},{begin:"\\b0[bB][0-1](_?[0-1])*r?i?\\b" +},{begin:"\\b0[oO][0-7](_?[0-7])*r?i?\\b"},{ +begin:"\\b0[xX][0-9a-fA-F](_?[0-9a-fA-F])*r?i?\\b"},{ +begin:"\\b0(_?[0-7])+r?i?\\b"}]},l={className:"params",begin:"\\(",end:"\\)", +endsParent:!0,keywords:i},o=[t,{className:"class",beginKeywords:"class module", +end:"$|;",illegal:/=/,contains:[n.inherit(n.TITLE_MODE,{ +begin:"[A-Za-z_]\\w*(::\\w+)*(\\?|!)?"}),{begin:"<\\s*",contains:[{ +begin:"("+n.IDENT_RE+"::)?"+n.IDENT_RE,relevance:0}]}].concat(b)},{ +className:"function",begin:e(/def\s+/,(_=a+"\\s*(\\(|;|$)",e("(?=",_,")"))), +relevance:0,keywords:"def",end:"$|;",contains:[n.inherit(n.TITLE_MODE,{begin:a +}),l].concat(b)},{begin:n.IDENT_RE+"::"},{className:"symbol", +begin:n.UNDERSCORE_IDENT_RE+"(!|\\?)?:",relevance:0},{className:"symbol", +begin:":(?!\\s)",contains:[t,{begin:a}],relevance:0},d,{className:"variable", +begin:"(\\$\\W)|((\\$|@@?)(\\w+))(?=[^@$?])(?![A-Za-z])(?![@$?'])"},{ +className:"params",begin:/\|/,end:/\|/,relevance:0,keywords:i},{ +begin:"("+n.RE_STARTERS_RE+"|unless)\\s*",keywords:"unless",contains:[{ +className:"regexp",contains:[n.BACKSLASH_ESCAPE,c],illegal:/\n/,variants:[{ +begin:"/",end:"/[a-z]*"},{begin:/%r\{/,end:/\}[a-z]*/},{begin:"%r\\(", +end:"\\)[a-z]*"},{begin:"%r!",end:"![a-z]*"},{begin:"%r\\[",end:"\\][a-z]*"}] +}].concat(r,b),relevance:0}].concat(r,b);var _;c.contains=o,l.contains=o +;const E=[{begin:/^\s*=>/,starts:{end:"$",contains:o}},{className:"meta", +begin:"^([>?]>|[\\w#]+\\(\\w+\\):\\d+:\\d+>|(\\w+-)?\\d+\\.\\d+\\.\\d+(p\\d+)?[^\\d][^>]+>)(?=[ ])", +starts:{end:"$",contains:o}}];return b.unshift(r),{name:"Ruby", +aliases:["rb","gemspec","podspec","thor","irb"],keywords:i,illegal:/\/\*/, +contains:[n.SHEBANG({binary:"ruby"})].concat(E).concat(b).concat(o)}}})()); +hljs.registerLanguage("rust",(()=>{"use strict";return e=>{ +const n="([ui](8|16|32|64|128|size)|f(32|64))?",t="drop i8 i16 i32 i64 i128 isize u8 u16 u32 u64 u128 usize f32 f64 str char bool Box Option Result String Vec Copy Send Sized Sync Drop Fn FnMut FnOnce ToOwned Clone Debug PartialEq PartialOrd Eq Ord AsRef AsMut Into From Default Iterator Extend IntoIterator DoubleEndedIterator ExactSizeIterator SliceConcatExt ToString assert! assert_eq! bitflags! bytes! cfg! col! concat! concat_idents! debug_assert! debug_assert_eq! env! panic! file! format! format_args! include_bin! include_str! line! local_data_key! module_path! option_env! print! println! select! stringify! try! unimplemented! unreachable! vec! write! writeln! macro_rules! assert_ne! debug_assert_ne!" +;return{name:"Rust",aliases:["rs"],keywords:{$pattern:e.IDENT_RE+"!?", +keyword:"abstract as async await become box break const continue crate do dyn else enum extern false final fn for if impl in let loop macro match mod move mut override priv pub ref return self Self static struct super trait true try type typeof unsafe unsized use virtual where while yield", +literal:"true false Some None Ok Err",built_in:t},illegal:""}]}}})()); +hljs.registerLanguage("scss",(()=>{"use strict" +;const e=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","main","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"],t=["any-hover","any-pointer","aspect-ratio","color","color-gamut","color-index","device-aspect-ratio","device-height","device-width","display-mode","forced-colors","grid","height","hover","inverted-colors","monochrome","orientation","overflow-block","overflow-inline","pointer","prefers-color-scheme","prefers-contrast","prefers-reduced-motion","prefers-reduced-transparency","resolution","scan","scripting","update","width","min-width","max-width","min-height","max-height"],i=["active","any-link","blank","checked","current","default","defined","dir","disabled","drop","empty","enabled","first","first-child","first-of-type","fullscreen","future","focus","focus-visible","focus-within","has","host","host-context","hover","indeterminate","in-range","invalid","is","lang","last-child","last-of-type","left","link","local-link","not","nth-child","nth-col","nth-last-child","nth-last-col","nth-last-of-type","nth-of-type","only-child","only-of-type","optional","out-of-range","past","placeholder-shown","read-only","read-write","required","right","root","scope","target","target-within","user-invalid","valid","visited","where"],o=["after","backdrop","before","cue","cue-region","first-letter","first-line","grammar-error","marker","part","placeholder","selection","slotted","spelling-error"],r=["align-content","align-items","align-self","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","auto","backface-visibility","background","background-attachment","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","border","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","clear","clip","clip-path","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","content","counter-increment","counter-reset","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","font","font-display","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-smoothing","font-stretch","font-style","font-variant","font-variant-ligatures","font-variation-settings","font-weight","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inherit","initial","justify-content","left","letter-spacing","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-bottom","margin-left","margin-right","margin-top","marks","mask","max-height","max-width","min-height","min-width","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-bottom","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","perspective","perspective-origin","pointer-events","position","quotes","resize","right","src","tab-size","table-layout","text-align","text-align-last","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-indent","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","white-space","widows","width","word-break","word-spacing","word-wrap","z-index"].reverse() +;return a=>{const n=(e=>({IMPORTANT:{className:"meta",begin:"!important"}, +HEXCOLOR:{className:"number",begin:"#([a-fA-F0-9]{6}|[a-fA-F0-9]{3})"}, +ATTRIBUTE_SELECTOR_MODE:{className:"selector-attr",begin:/\[/,end:/\]/, +illegal:"$",contains:[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE]} +}))(a),l=o,s=i,d="@[a-z-]+",c={className:"variable", +begin:"(\\$[a-zA-Z-][a-zA-Z0-9_-]*)\\b"};return{name:"SCSS",case_insensitive:!0, +illegal:"[=/|']",contains:[a.C_LINE_COMMENT_MODE,a.C_BLOCK_COMMENT_MODE,{ +className:"selector-id",begin:"#[A-Za-z0-9_-]+",relevance:0},{ +className:"selector-class",begin:"\\.[A-Za-z0-9_-]+",relevance:0 +},n.ATTRIBUTE_SELECTOR_MODE,{className:"selector-tag", +begin:"\\b("+e.join("|")+")\\b",relevance:0},{className:"selector-pseudo", +begin:":("+s.join("|")+")"},{className:"selector-pseudo", +begin:"::("+l.join("|")+")"},c,{begin:/\(/,end:/\)/,contains:[a.CSS_NUMBER_MODE] +},{className:"attribute",begin:"\\b("+r.join("|")+")\\b"},{ +begin:"\\b(whitespace|wait|w-resize|visible|vertical-text|vertical-ideographic|uppercase|upper-roman|upper-alpha|underline|transparent|top|thin|thick|text|text-top|text-bottom|tb-rl|table-header-group|table-footer-group|sw-resize|super|strict|static|square|solid|small-caps|separate|se-resize|scroll|s-resize|rtl|row-resize|ridge|right|repeat|repeat-y|repeat-x|relative|progress|pointer|overline|outside|outset|oblique|nowrap|not-allowed|normal|none|nw-resize|no-repeat|no-drop|newspaper|ne-resize|n-resize|move|middle|medium|ltr|lr-tb|lowercase|lower-roman|lower-alpha|loose|list-item|line|line-through|line-edge|lighter|left|keep-all|justify|italic|inter-word|inter-ideograph|inside|inset|inline|inline-block|inherit|inactive|ideograph-space|ideograph-parenthesis|ideograph-numeric|ideograph-alpha|horizontal|hidden|help|hand|groove|fixed|ellipsis|e-resize|double|dotted|distribute|distribute-space|distribute-letter|distribute-all-lines|disc|disabled|default|decimal|dashed|crosshair|collapse|col-resize|circle|char|center|capitalize|break-word|break-all|bottom|both|bolder|bold|block|bidi-override|below|baseline|auto|always|all-scroll|absolute|table|table-cell)\\b" +},{begin:":",end:";", +contains:[c,n.HEXCOLOR,a.CSS_NUMBER_MODE,a.QUOTE_STRING_MODE,a.APOS_STRING_MODE,n.IMPORTANT] +},{begin:"@(page|font-face)",lexemes:d,keywords:"@page @font-face"},{begin:"@", +end:"[{;]",returnBegin:!0,keywords:{$pattern:/[a-z-]+/, +keyword:"and or not only",attribute:t.join(" ")},contains:[{begin:d, +className:"keyword"},{begin:/[a-z-]+(?=:)/,className:"attribute" +},c,a.QUOTE_STRING_MODE,a.APOS_STRING_MODE,n.HEXCOLOR,a.CSS_NUMBER_MODE]}]}} +})()); +hljs.registerLanguage("shell",(()=>{"use strict";return s=>({ +name:"Shell Session",aliases:["console"],contains:[{className:"meta", +begin:/^\s{0,3}[/~\w\d[\]()@-]*[>%$#]/,starts:{end:/[^\\](?=\s*$)/, +subLanguage:"bash"}}]})})()); +hljs.registerLanguage("sql",(()=>{"use strict";function e(e){ +return e?"string"==typeof e?e:e.source:null}function r(...r){ +return r.map((r=>e(r))).join("")}function t(...r){ +return"("+r.map((r=>e(r))).join("|")+")"}return e=>{ +const n=e.COMMENT("--","$"),a=["true","false","unknown"],i=["bigint","binary","blob","boolean","char","character","clob","date","dec","decfloat","decimal","float","int","integer","interval","nchar","nclob","national","numeric","real","row","smallint","time","timestamp","varchar","varying","varbinary"],s=["abs","acos","array_agg","asin","atan","avg","cast","ceil","ceiling","coalesce","corr","cos","cosh","count","covar_pop","covar_samp","cume_dist","dense_rank","deref","element","exp","extract","first_value","floor","json_array","json_arrayagg","json_exists","json_object","json_objectagg","json_query","json_table","json_table_primitive","json_value","lag","last_value","lead","listagg","ln","log","log10","lower","max","min","mod","nth_value","ntile","nullif","percent_rank","percentile_cont","percentile_disc","position","position_regex","power","rank","regr_avgx","regr_avgy","regr_count","regr_intercept","regr_r2","regr_slope","regr_sxx","regr_sxy","regr_syy","row_number","sin","sinh","sqrt","stddev_pop","stddev_samp","substring","substring_regex","sum","tan","tanh","translate","translate_regex","treat","trim","trim_array","unnest","upper","value_of","var_pop","var_samp","width_bucket"],o=["create table","insert into","primary key","foreign key","not null","alter table","add constraint","grouping sets","on overflow","character set","respect nulls","ignore nulls","nulls first","nulls last","depth first","breadth first"],c=s,l=["abs","acos","all","allocate","alter","and","any","are","array","array_agg","array_max_cardinality","as","asensitive","asin","asymmetric","at","atan","atomic","authorization","avg","begin","begin_frame","begin_partition","between","bigint","binary","blob","boolean","both","by","call","called","cardinality","cascaded","case","cast","ceil","ceiling","char","char_length","character","character_length","check","classifier","clob","close","coalesce","collate","collect","column","commit","condition","connect","constraint","contains","convert","copy","corr","corresponding","cos","cosh","count","covar_pop","covar_samp","create","cross","cube","cume_dist","current","current_catalog","current_date","current_default_transform_group","current_path","current_role","current_row","current_schema","current_time","current_timestamp","current_path","current_role","current_transform_group_for_type","current_user","cursor","cycle","date","day","deallocate","dec","decimal","decfloat","declare","default","define","delete","dense_rank","deref","describe","deterministic","disconnect","distinct","double","drop","dynamic","each","element","else","empty","end","end_frame","end_partition","end-exec","equals","escape","every","except","exec","execute","exists","exp","external","extract","false","fetch","filter","first_value","float","floor","for","foreign","frame_row","free","from","full","function","fusion","get","global","grant","group","grouping","groups","having","hold","hour","identity","in","indicator","initial","inner","inout","insensitive","insert","int","integer","intersect","intersection","interval","into","is","join","json_array","json_arrayagg","json_exists","json_object","json_objectagg","json_query","json_table","json_table_primitive","json_value","lag","language","large","last_value","lateral","lead","leading","left","like","like_regex","listagg","ln","local","localtime","localtimestamp","log","log10","lower","match","match_number","match_recognize","matches","max","member","merge","method","min","minute","mod","modifies","module","month","multiset","national","natural","nchar","nclob","new","no","none","normalize","not","nth_value","ntile","null","nullif","numeric","octet_length","occurrences_regex","of","offset","old","omit","on","one","only","open","or","order","out","outer","over","overlaps","overlay","parameter","partition","pattern","per","percent","percent_rank","percentile_cont","percentile_disc","period","portion","position","position_regex","power","precedes","precision","prepare","primary","procedure","ptf","range","rank","reads","real","recursive","ref","references","referencing","regr_avgx","regr_avgy","regr_count","regr_intercept","regr_r2","regr_slope","regr_sxx","regr_sxy","regr_syy","release","result","return","returns","revoke","right","rollback","rollup","row","row_number","rows","running","savepoint","scope","scroll","search","second","seek","select","sensitive","session_user","set","show","similar","sin","sinh","skip","smallint","some","specific","specifictype","sql","sqlexception","sqlstate","sqlwarning","sqrt","start","static","stddev_pop","stddev_samp","submultiset","subset","substring","substring_regex","succeeds","sum","symmetric","system","system_time","system_user","table","tablesample","tan","tanh","then","time","timestamp","timezone_hour","timezone_minute","to","trailing","translate","translate_regex","translation","treat","trigger","trim","trim_array","true","truncate","uescape","union","unique","unknown","unnest","update ","upper","user","using","value","values","value_of","var_pop","var_samp","varbinary","varchar","varying","versioning","when","whenever","where","width_bucket","window","with","within","without","year","add","asc","collation","desc","final","first","last","view"].filter((e=>!s.includes(e))),u={ +begin:r(/\b/,t(...c),/\s*\(/),keywords:{built_in:c}};return{name:"SQL", +case_insensitive:!0,illegal:/[{}]|<\//,keywords:{$pattern:/\b[\w\.]+/, +keyword:((e,{exceptions:r,when:t}={})=>{const n=t +;return r=r||[],e.map((e=>e.match(/\|\d+$/)||r.includes(e)?e:n(e)?e+"|0":e)) +})(l,{when:e=>e.length<3}),literal:a,type:i, +built_in:["current_catalog","current_date","current_default_transform_group","current_path","current_role","current_schema","current_transform_group_for_type","current_user","session_user","system_time","system_user","current_time","localtime","current_timestamp","localtimestamp"] +},contains:[{begin:t(...o),keywords:{$pattern:/[\w\.]+/,keyword:l.concat(o), +literal:a,type:i}},{className:"type", +begin:t("double precision","large object","with timezone","without timezone") +},u,{className:"variable",begin:/@[a-z0-9]+/},{className:"string",variants:[{ +begin:/'/,end:/'/,contains:[{begin:/''/}]}]},{begin:/"/,end:/"/,contains:[{ +begin:/""/}]},e.C_NUMBER_MODE,e.C_BLOCK_COMMENT_MODE,n,{className:"operator", +begin:/[-+*/=%^~]|&&?|\|\|?|!=?|<(?:=>?|<|>)?|>[>=]?/,relevance:0}]}}})()); +hljs.registerLanguage("swift",(()=>{"use strict";function e(e){ +return e?"string"==typeof e?e:e.source:null}function n(e){return a("(?=",e,")")} +function a(...n){return n.map((n=>e(n))).join("")}function t(...n){ +return"("+n.map((n=>e(n))).join("|")+")"} +const i=e=>a(/\b/,e,/\w$/.test(e)?/\b/:/\B/),s=["Protocol","Type"].map(i),u=["init","self"].map(i),c=["Any","Self"],r=["associatedtype","async","await",/as\?/,/as!/,"as","break","case","catch","class","continue","convenience","default","defer","deinit","didSet","do","dynamic","else","enum","extension","fallthrough",/fileprivate\(set\)/,"fileprivate","final","for","func","get","guard","if","import","indirect","infix",/init\?/,/init!/,"inout",/internal\(set\)/,"internal","in","is","lazy","let","mutating","nonmutating",/open\(set\)/,"open","operator","optional","override","postfix","precedencegroup","prefix",/private\(set\)/,"private","protocol",/public\(set\)/,"public","repeat","required","rethrows","return","set","some","static","struct","subscript","super","switch","throws","throw",/try\?/,/try!/,"try","typealias",/unowned\(safe\)/,/unowned\(unsafe\)/,"unowned","var","weak","where","while","willSet"],o=["false","nil","true"],l=["assignment","associativity","higherThan","left","lowerThan","none","right"],m=["#colorLiteral","#column","#dsohandle","#else","#elseif","#endif","#error","#file","#fileID","#fileLiteral","#filePath","#function","#if","#imageLiteral","#keyPath","#line","#selector","#sourceLocation","#warn_unqualified_access","#warning"],d=["abs","all","any","assert","assertionFailure","debugPrint","dump","fatalError","getVaList","isKnownUniquelyReferenced","max","min","numericCast","pointwiseMax","pointwiseMin","precondition","preconditionFailure","print","readLine","repeatElement","sequence","stride","swap","swift_unboxFromSwiftValueWithType","transcode","type","unsafeBitCast","unsafeDowncast","withExtendedLifetime","withUnsafeMutablePointer","withUnsafePointer","withVaList","withoutActuallyEscaping","zip"],p=t(/[/=\-+!*%<>&|^~?]/,/[\u00A1-\u00A7]/,/[\u00A9\u00AB]/,/[\u00AC\u00AE]/,/[\u00B0\u00B1]/,/[\u00B6\u00BB\u00BF\u00D7\u00F7]/,/[\u2016-\u2017]/,/[\u2020-\u2027]/,/[\u2030-\u203E]/,/[\u2041-\u2053]/,/[\u2055-\u205E]/,/[\u2190-\u23FF]/,/[\u2500-\u2775]/,/[\u2794-\u2BFF]/,/[\u2E00-\u2E7F]/,/[\u3001-\u3003]/,/[\u3008-\u3020]/,/[\u3030]/),F=t(p,/[\u0300-\u036F]/,/[\u1DC0-\u1DFF]/,/[\u20D0-\u20FF]/,/[\uFE00-\uFE0F]/,/[\uFE20-\uFE2F]/),b=a(p,F,"*"),h=t(/[a-zA-Z_]/,/[\u00A8\u00AA\u00AD\u00AF\u00B2-\u00B5\u00B7-\u00BA]/,/[\u00BC-\u00BE\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u00FF]/,/[\u0100-\u02FF\u0370-\u167F\u1681-\u180D\u180F-\u1DBF]/,/[\u1E00-\u1FFF]/,/[\u200B-\u200D\u202A-\u202E\u203F-\u2040\u2054\u2060-\u206F]/,/[\u2070-\u20CF\u2100-\u218F\u2460-\u24FF\u2776-\u2793]/,/[\u2C00-\u2DFF\u2E80-\u2FFF]/,/[\u3004-\u3007\u3021-\u302F\u3031-\u303F\u3040-\uD7FF]/,/[\uF900-\uFD3D\uFD40-\uFDCF\uFDF0-\uFE1F\uFE30-\uFE44]/,/[\uFE47-\uFEFE\uFF00-\uFFFD]/),f=t(h,/\d/,/[\u0300-\u036F\u1DC0-\u1DFF\u20D0-\u20FF\uFE20-\uFE2F]/),w=a(h,f,"*"),y=a(/[A-Z]/,f,"*"),g=["autoclosure",a(/convention\(/,t("swift","block","c"),/\)/),"discardableResult","dynamicCallable","dynamicMemberLookup","escaping","frozen","GKInspectable","IBAction","IBDesignable","IBInspectable","IBOutlet","IBSegueAction","inlinable","main","nonobjc","NSApplicationMain","NSCopying","NSManaged",a(/objc\(/,w,/\)/),"objc","objcMembers","propertyWrapper","requires_stored_property_inits","testable","UIApplicationMain","unknown","usableFromInline"],E=["iOS","iOSApplicationExtension","macOS","macOSApplicationExtension","macCatalyst","macCatalystApplicationExtension","watchOS","watchOSApplicationExtension","tvOS","tvOSApplicationExtension","swift"] +;return e=>{const p={match:/\s+/,relevance:0},h=e.COMMENT("/\\*","\\*/",{ +contains:["self"]}),v=[e.C_LINE_COMMENT_MODE,h],N={className:"keyword", +begin:a(/\./,n(t(...s,...u))),end:t(...s,...u),excludeBegin:!0},A={ +match:a(/\./,t(...r)),relevance:0 +},C=r.filter((e=>"string"==typeof e)).concat(["_|0"]),_={variants:[{ +className:"keyword", +match:t(...r.filter((e=>"string"!=typeof e)).concat(c).map(i),...u)}]},D={ +$pattern:t(/\b\w+/,/#\w+/),keyword:C.concat(m),literal:o},B=[N,A,_],k=[{ +match:a(/\./,t(...d)),relevance:0},{className:"built_in", +match:a(/\b/,t(...d),/(?=\()/)}],M={match:/->/,relevance:0},S=[M,{ +className:"operator",relevance:0,variants:[{match:b},{match:`\\.(\\.|${F})+`}] +}],x="([0-9a-fA-F]_*)+",I={className:"number",relevance:0,variants:[{ +match:"\\b(([0-9]_*)+)(\\.(([0-9]_*)+))?([eE][+-]?(([0-9]_*)+))?\\b"},{ +match:`\\b0x(${x})(\\.(${x}))?([pP][+-]?(([0-9]_*)+))?\\b`},{ +match:/\b0o([0-7]_*)+\b/},{match:/\b0b([01]_*)+\b/}]},O=(e="")=>({ +className:"subst",variants:[{match:a(/\\/,e,/[0\\tnr"']/)},{ +match:a(/\\/,e,/u\{[0-9a-fA-F]{1,8}\}/)}]}),T=(e="")=>({className:"subst", +match:a(/\\/,e,/[\t ]*(?:[\r\n]|\r\n)/)}),L=(e="")=>({className:"subst", +label:"interpol",begin:a(/\\/,e,/\(/),end:/\)/}),P=(e="")=>({begin:a(e,/"""/), +end:a(/"""/,e),contains:[O(e),T(e),L(e)]}),$=(e="")=>({begin:a(e,/"/), +end:a(/"/,e),contains:[O(e),L(e)]}),K={className:"string", +variants:[P(),P("#"),P("##"),P("###"),$(),$("#"),$("##"),$("###")]},j={ +match:a(/`/,w,/`/)},z=[j,{className:"variable",match:/\$\d+/},{ +className:"variable",match:`\\$${f}+`}],q=[{match:/(@|#)available/, +className:"keyword",starts:{contains:[{begin:/\(/,end:/\)/,keywords:E, +contains:[...S,I,K]}]}},{className:"keyword",match:a(/@/,t(...g))},{ +className:"meta",match:a(/@/,w)}],U={match:n(/\b[A-Z]/),relevance:0,contains:[{ +className:"type", +match:a(/(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)/,f,"+") +},{className:"type",match:y,relevance:0},{match:/[?!]+/,relevance:0},{ +match:/\.\.\./,relevance:0},{match:a(/\s+&\s+/,n(y)),relevance:0}]},Z={ +begin://,keywords:D,contains:[...v,...B,...q,M,U]};U.contains.push(Z) +;const G={begin:/\(/,end:/\)/,relevance:0,keywords:D,contains:["self",{ +match:a(w,/\s*:/),keywords:"_|0",relevance:0 +},...v,...B,...k,...S,I,K,...z,...q,U]},H={beginKeywords:"func",contains:[{ +className:"title",match:t(j.match,w,b),endsParent:!0,relevance:0},p]},R={ +begin://,contains:[...v,U]},V={begin:/\(/,end:/\)/,keywords:D, +contains:[{begin:t(n(a(w,/\s*:/)),n(a(w,/\s+/,w,/\s*:/))),end:/:/,relevance:0, +contains:[{className:"keyword",match:/\b_\b/},{className:"params",match:w}] +},...v,...B,...S,I,K,...q,U,G],endsParent:!0,illegal:/["']/},W={ +className:"function",match:n(/\bfunc\b/),contains:[H,R,V,p],illegal:[/\[/,/%/] +},X={className:"function",match:/\b(subscript|init[?!]?)\s*(?=[<(])/,keywords:{ +keyword:"subscript init init? init!",$pattern:/\w+[?!]?/},contains:[R,V,p], +illegal:/\[|%/},J={beginKeywords:"operator",end:e.MATCH_NOTHING_RE,contains:[{ +className:"title",match:b,endsParent:!0,relevance:0}]},Q={ +beginKeywords:"precedencegroup",end:e.MATCH_NOTHING_RE,contains:[{ +className:"title",match:y,relevance:0},{begin:/{/,end:/}/,relevance:0, +endsParent:!0,keywords:[...l,...o],contains:[U]}]};for(const e of K.variants){ +const n=e.contains.find((e=>"interpol"===e.label));n.keywords=D +;const a=[...B,...k,...S,I,K,...z];n.contains=[...a,{begin:/\(/,end:/\)/, +contains:["self",...a]}]}return{name:"Swift",keywords:D,contains:[...v,W,X,{ +className:"class",beginKeywords:"struct protocol class extension enum", +end:"\\{",excludeEnd:!0,keywords:D,contains:[e.inherit(e.TITLE_MODE,{ +begin:/[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/}),...B]},J,Q,{ +beginKeywords:"import",end:/$/,contains:[...v],relevance:0 +},...B,...k,...S,I,K,...z,...q,U,G]}}})()); +hljs.registerLanguage("typescript",(()=>{"use strict" +;const e="[A-Za-z$_][0-9A-Za-z$_]*",n=["as","in","of","if","for","while","finally","var","new","function","do","return","void","else","break","catch","instanceof","with","throw","case","default","try","switch","continue","typeof","delete","let","yield","const","class","debugger","async","await","static","import","from","export","extends"],a=["true","false","null","undefined","NaN","Infinity"],s=[].concat(["setInterval","setTimeout","clearInterval","clearTimeout","require","exports","eval","isFinite","isNaN","parseFloat","parseInt","decodeURI","decodeURIComponent","encodeURI","encodeURIComponent","escape","unescape"],["arguments","this","super","console","window","document","localStorage","module","global"],["Intl","DataView","Number","Math","Date","String","RegExp","Object","Function","Boolean","Error","Symbol","Set","Map","WeakSet","WeakMap","Proxy","Reflect","JSON","Promise","Float64Array","Int16Array","Int32Array","Int8Array","Uint16Array","Uint32Array","Float32Array","Array","Uint8Array","Uint8ClampedArray","ArrayBuffer","BigInt64Array","BigUint64Array","BigInt"],["EvalError","InternalError","RangeError","ReferenceError","SyntaxError","TypeError","URIError"]) +;function t(e){return r("(?=",e,")")}function r(...e){return e.map((e=>{ +return(n=e)?"string"==typeof n?n:n.source:null;var n})).join("")}return i=>{ +const c={$pattern:e, +keyword:n.concat(["type","namespace","typedef","interface","public","private","protected","implements","declare","abstract","readonly"]), +literal:a, +built_in:s.concat(["any","void","number","boolean","string","object","never","enum"]) +},o={className:"meta",begin:"@[A-Za-z$_][0-9A-Za-z$_]*"},l=(e,n,a)=>{ +const s=e.contains.findIndex((e=>e.label===n)) +;if(-1===s)throw Error("can not find mode to replace");e.contains.splice(s,1,a) +},b=(i=>{const c=e,o={begin:/<[A-Za-z0-9\\._:-]+/, +end:/\/[A-Za-z0-9\\._:-]+>|\/>/,isTrulyOpeningTag:(e,n)=>{ +const a=e[0].length+e.index,s=e.input[a];"<"!==s?">"===s&&(((e,{after:n})=>{ +const a="", +returnBegin:!0,end:"\\s*=>",contains:[{className:"params",variants:[{ +begin:i.UNDERSCORE_IDENT_RE,relevance:0},{className:null,begin:/\(\s*\)/,skip:!0 +},{begin:/\(/,end:/\)/,excludeBegin:!0,excludeEnd:!0,keywords:l,contains:f}]}] +},{begin:/,/,relevance:0},{className:"",begin:/\s/,end:/\s*/,skip:!0},{ +variants:[{begin:"<>",end:""},{begin:o.begin,"on:begin":o.isTrulyOpeningTag, +end:o.end}],subLanguage:"xml",contains:[{begin:o.begin,end:o.end,skip:!0, +contains:["self"]}]}],relevance:0},{className:"function", +beginKeywords:"function",end:/[{;]/,excludeEnd:!0,keywords:l, +contains:["self",i.inherit(i.TITLE_MODE,{begin:c}),A],illegal:/%/},{ +beginKeywords:"while if switch catch for"},{className:"function", +begin:i.UNDERSCORE_IDENT_RE+"\\([^()]*(\\([^()]*(\\([^()]*\\)[^()]*)*\\)[^()]*)*\\)\\s*\\{", +returnBegin:!0,contains:[A,i.inherit(i.TITLE_MODE,{begin:c})]},{variants:[{ +begin:"\\."+c},{begin:"\\$"+c}],relevance:0},{className:"class", +beginKeywords:"class",end:/[{;=]/,excludeEnd:!0,illegal:/[:"[\]]/,contains:[{ +beginKeywords:"extends"},i.UNDERSCORE_TITLE_MODE]},{begin:/\b(?=constructor)/, +end:/[{;]/,excludeEnd:!0,contains:[i.inherit(i.TITLE_MODE,{begin:c}),"self",A] +},{begin:"(get|set)\\s+(?="+c+"\\()",end:/\{/,keywords:"get set", +contains:[i.inherit(i.TITLE_MODE,{begin:c}),{begin:/\(\)/},A]},{begin:/\$[(.]/}] +}})(i) +;return Object.assign(b.keywords,c),b.exports.PARAMS_CONTAINS.push(o),b.contains=b.contains.concat([o,{ +beginKeywords:"namespace",end:/\{/,excludeEnd:!0},{beginKeywords:"interface", +end:/\{/,excludeEnd:!0,keywords:"interface extends" +}]),l(b,"shebang",i.SHEBANG()),l(b,"use_strict",{className:"meta",relevance:10, +begin:/^\s*['"]use strict['"]/ +}),b.contains.find((e=>"function"===e.className)).relevance=0,Object.assign(b,{ +name:"TypeScript",aliases:["ts","tsx"]}),b}})()); +hljs.registerLanguage("vbnet",(()=>{"use strict";function e(e){ +return e?"string"==typeof e?e:e.source:null}function n(...n){ +return n.map((n=>e(n))).join("")}function t(...n){ +return"("+n.map((n=>e(n))).join("|")+")"}return e=>{ +const a=/\d{1,2}\/\d{1,2}\/\d{4}/,i=/\d{4}-\d{1,2}-\d{1,2}/,s=/(\d|1[012])(:\d+){0,2} *(AM|PM)/,r=/\d{1,2}(:\d{1,2}){1,2}/,o={ +className:"literal",variants:[{begin:n(/# */,t(i,a),/ *#/)},{ +begin:n(/# */,r,/ *#/)},{begin:n(/# */,s,/ *#/)},{ +begin:n(/# */,t(i,a),/ +/,t(s,r),/ *#/)}]},l=e.COMMENT(/'''/,/$/,{contains:[{ +className:"doctag",begin:/<\/?/,end:/>/}]}),c=e.COMMENT(null,/$/,{variants:[{ +begin:/'/},{begin:/([\t ]|^)REM(?=\s)/}]});return{name:"Visual Basic .NET", +aliases:["vb"],case_insensitive:!0,classNameAliases:{label:"symbol"},keywords:{ +keyword:"addhandler alias aggregate ansi as async assembly auto binary by byref byval call case catch class compare const continue custom declare default delegate dim distinct do each equals else elseif end enum erase error event exit explicit finally for friend from function get global goto group handles if implements imports in inherits interface into iterator join key let lib loop me mid module mustinherit mustoverride mybase myclass namespace narrowing new next notinheritable notoverridable of off on operator option optional order overloads overridable overrides paramarray partial preserve private property protected public raiseevent readonly redim removehandler resume return select set shadows shared skip static step stop structure strict sub synclock take text then throw to try unicode until using when where while widening with withevents writeonly yield", +built_in:"addressof and andalso await directcast gettype getxmlnamespace is isfalse isnot istrue like mod nameof new not or orelse trycast typeof xor cbool cbyte cchar cdate cdbl cdec cint clng cobj csbyte cshort csng cstr cuint culng cushort", +type:"boolean byte char date decimal double integer long object sbyte short single string uinteger ulong ushort", +literal:"true false nothing"}, +illegal:"//|\\{|\\}|endif|gosub|variant|wend|^\\$ ",contains:[{ +className:"string",begin:/"(""|[^/n])"C\b/},{className:"string",begin:/"/, +end:/"/,illegal:/\n/,contains:[{begin:/""/}]},o,{className:"number",relevance:0, +variants:[{begin:/\b\d[\d_]*((\.[\d_]+(E[+-]?[\d_]+)?)|(E[+-]?[\d_]+))[RFD@!#]?/ +},{begin:/\b\d[\d_]*((U?[SIL])|[%&])?/},{begin:/&H[\dA-F_]+((U?[SIL])|[%&])?/},{ +begin:/&O[0-7_]+((U?[SIL])|[%&])?/},{begin:/&B[01_]+((U?[SIL])|[%&])?/}]},{ +className:"label",begin:/^\w+:/},l,c,{className:"meta", +begin:/[\t ]*#(const|disable|else|elseif|enable|end|externalsource|if|region)\b/, +end:/$/,keywords:{ +"meta-keyword":"const disable else elseif enable end externalsource if region then" +},contains:[c]}]}}})()); +hljs.registerLanguage("yaml",(()=>{"use strict";return e=>{ +var n="true false yes no null",a="[\\w#;/?:@&=+$,.~*'()[\\]]+",s={ +className:"string",relevance:0,variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/ +},{begin:/\S+/}],contains:[e.BACKSLASH_ESCAPE,{className:"template-variable", +variants:[{begin:/\{\{/,end:/\}\}/},{begin:/%\{/,end:/\}/}]}]},i=e.inherit(s,{ +variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/[^\s,{}[\]]+/}]}),l={ +end:",",endsWithParent:!0,excludeEnd:!0,keywords:n,relevance:0},t={begin:/\{/, +end:/\}/,contains:[l],illegal:"\\n",relevance:0},g={begin:"\\[",end:"\\]", +contains:[l],illegal:"\\n",relevance:0},b=[{className:"attr",variants:[{ +begin:"\\w[\\w :\\/.-]*:(?=[ \t]|$)"},{begin:'"\\w[\\w :\\/.-]*":(?=[ \t]|$)'},{ +begin:"'\\w[\\w :\\/.-]*':(?=[ \t]|$)"}]},{className:"meta",begin:"^---\\s*$", +relevance:10},{className:"string", +begin:"[\\|>]([1-9]?[+-])?[ ]*\\n( +)[^ ][^\\n]*\\n(\\2[^\\n]+\\n?)*"},{ +begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:!0,excludeEnd:!0, +relevance:0},{className:"type",begin:"!\\w+!"+a},{className:"type", +begin:"!<"+a+">"},{className:"type",begin:"!"+a},{className:"type",begin:"!!"+a +},{className:"meta",begin:"&"+e.UNDERSCORE_IDENT_RE+"$"},{className:"meta", +begin:"\\*"+e.UNDERSCORE_IDENT_RE+"$"},{className:"bullet",begin:"-(?=[ ]|$)", +relevance:0},e.HASH_COMMENT_MODE,{beginKeywords:n,keywords:{literal:n}},{ +className:"number", +begin:"\\b[0-9]{4}(-[0-9][0-9]){0,2}([Tt \\t][0-9][0-9]?(:[0-9][0-9]){2})?(\\.[0-9]*)?([ \\t])*(Z|[-+][0-9][0-9]?(:[0-9][0-9])?)?\\b" +},{className:"number",begin:e.C_NUMBER_RE+"\\b",relevance:0},t,g,s],r=[...b] +;return r.pop(),r.push(i),l.contains=r,{name:"YAML",case_insensitive:!0, +aliases:["yml"],contains:b}}})()); \ No newline at end of file diff --git a/htmlReport/js/highlightjs-line-numbers.min.js b/htmlReport/js/highlightjs-line-numbers.min.js new file mode 100644 index 0000000..8548576 --- /dev/null +++ b/htmlReport/js/highlightjs-line-numbers.min.js @@ -0,0 +1,24 @@ +/* +The MIT License (MIT) + +Copyright (c) 2017 Yauheni Pakala + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + */ +!function(r,o){"use strict";var e,i="hljs-ln",l="hljs-ln-line",h="hljs-ln-code",s="hljs-ln-numbers",c="hljs-ln-n",m="data-line-number",a=/\r\n|\r|\n/g;function u(e){for(var n=e.toString(),t=e.anchorNode;"TD"!==t.nodeName;)t=t.parentNode;for(var r=e.focusNode;"TD"!==r.nodeName;)r=r.parentNode;var o=parseInt(t.dataset.lineNumber),a=parseInt(r.dataset.lineNumber);if(o==a)return n;var i,l=t.textContent,s=r.textContent;for(a
{6}',[l,s,c,m,h,o+n.startFrom,0{1}',[i,r])}return e}(e.innerHTML,o)}function v(e){var n=e.className;if(/hljs-/.test(n)){for(var t=g(e.innerHTML),r=0,o="";r{1}
\n',[n,0 + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_BLOCK.html b/htmlReport/ns-1/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..b21590f --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_BLOCK.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-1/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..5af3334 --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_CLASS.html b/htmlReport/ns-1/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..2e49e0b --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_CLASS.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-1/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..0cbc38a --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_LINE.html b/htmlReport/ns-1/index_SORT_BY_LINE.html new file mode 100644 index 0000000..c56c023 --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_LINE.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-1/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..981bac6 --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_METHOD.html b/htmlReport/ns-1/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..1a65aab --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_METHOD.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-1/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..5f01a3d --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-1/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..370f765 --- /dev/null +++ b/htmlReport/ns-1/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web + + 50% + + + (1/2) + + + + 66.7% + + + (4/6) + + + + 0% + + + (0/5) + + + + 21.4% + + + (6/28) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
App + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + + 100% + + + (6/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-1/sources/source-1.html b/htmlReport/ns-1/sources/source-1.html new file mode 100644 index 0000000..a804d20 --- /dev/null +++ b/htmlReport/ns-1/sources/source-1.html @@ -0,0 +1,206 @@ + + + + + + + + Coverage Report > App + + + + + + +
+ + +

Coverage Summary for Class: App (ng.cove.web)

+ + + + + + + + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
App + + 100% + + + (4/4) + + + + 100% + + + (6/6) + +
App$$SpringCGLIB$$0
App$$SpringCGLIB$$FastClass$$0
App$$SpringCGLIB$$FastClass$$1
Total + + 100% + + + (4/4) + + + + 100% + + + (6/6) + +
+ +
+
+ + +
+ package ng.cove.web
+ 
+ import com.github.benmanes.caffeine.cache.Caffeine
+ import com.google.auth.oauth2.GoogleCredentials
+ import com.google.cloud.secretmanager.v1.SecretManagerServiceClient
+ import com.google.cloud.secretmanager.v1.SecretVersionName
+ import com.google.firebase.FirebaseApp
+ import com.google.firebase.FirebaseOptions
+ import de.flapdoodle.embed.mongo.spring.autoconfigure.EmbeddedMongoAutoConfiguration
+ import ng.cove.web.component.SmsOtpService
+ import org.springframework.boot.SpringApplication
+ import org.springframework.boot.autoconfigure.SpringBootApplication
+ import org.springframework.boot.autoconfigure.data.mongo.MongoDataAutoConfiguration
+ import org.springframework.boot.context.event.ApplicationStartedEvent
+ import org.springframework.cache.annotation.EnableCaching
+ import org.springframework.cache.caffeine.CaffeineCacheManager
+ import org.springframework.context.ApplicationListener
+ import org.springframework.context.annotation.Bean
+ import org.springframework.scheduling.annotation.EnableAsync
+ import org.springframework.scheduling.annotation.EnableScheduling
+ import org.springframework.web.client.RestTemplate
+ import java.util.concurrent.TimeUnit
+ 
+ @EnableAsync
+ @EnableCaching
+ @SpringBootApplication(exclude = [MongoDataAutoConfiguration::class, EmbeddedMongoAutoConfiguration::class])
+ class App {
+ 
+     @Bean
+     fun restTemplate() = RestTemplate()
+ 
+     @Bean
+     fun caffeineConfig(): Caffeine<Any, Any> {
+         return Caffeine.newBuilder().expireAfterWrite(60, TimeUnit.MINUTES)
+     }
+ 
+     @Bean
+     fun cacheManager(caffeine: Caffeine<Any, Any>): CaffeineCacheManager {
+         val cacheManager = CaffeineCacheManager()
+         cacheManager.setCaffeine(caffeine)
+         return cacheManager
+     }
+ }
+ 
+ fun main(args: Array<String>) {
+     val app = SpringApplication(App::class.java)
+ 
+     val startedEvent = ApplicationListener<ApplicationStartedEvent> { event ->
+ 
+         val profiles = event.applicationContext.environment.activeProfiles
+         // profiles is sometimes empty when running on a production server because it
+         // has not been loaded from the application.properties file at this point
+         val profile = profiles.getOrNull(0) ?: "prod"
+ 
+         when (profile) {
+             "test" -> {}
+             "dev" -> FirebaseApp.initializeApp()
+             "prod" -> {
+                 // Get secrets from GCP
+                 val gcpProjectId = "gatedaccessdev"
+                 val firebaseSecret = SecretVersionName.of(gcpProjectId, "firebase-service-account", "1")
+                 val termiiSecret = SecretVersionName.of(gcpProjectId, "termii-key", "1")
+ 
+                 // Auto-closable
+                 SecretManagerServiceClient.create().use {
+                     // Set Termii API Key to service
+                     val termiiSecretPayload =
+                         it.accessSecretVersion(termiiSecret).payload.data.toByteArray().inputStream()
+                     val smsOtpService = event.applicationContext.getBean(SmsOtpService::class.java)
+                     smsOtpService.termiiApiKey = String(termiiSecretPayload.readAllBytes())
+                     // Init Firebase with service account
+                     val serviceAccountStream =
+                         it.accessSecretVersion(firebaseSecret).payload.data.toByteArray().inputStream()
+                     val options = FirebaseOptions.builder()
+                         .setCredentials(GoogleCredentials.fromStream(serviceAccountStream))
+                         .build()
+                     FirebaseApp.initializeApp(options)
+                 }
+             }
+         }
+ 
+     }
+ 
+     app.addListeners(startedEvent)
+     app.run(*args)
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-1/sources/source-2.html b/htmlReport/ns-1/sources/source-2.html new file mode 100644 index 0000000..1c62682 --- /dev/null +++ b/htmlReport/ns-1/sources/source-2.html @@ -0,0 +1,202 @@ + + + + + + + + Coverage Report > AppKt + + + + + + +
+ + +

Coverage Summary for Class: AppKt (ng.cove.web)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
AppKt + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/5) + + + + 0% + + + (0/22) + +
+ +
+
+ + +
+ package ng.cove.web
+ 
+ import com.github.benmanes.caffeine.cache.Caffeine
+ import com.google.auth.oauth2.GoogleCredentials
+ import com.google.cloud.secretmanager.v1.SecretManagerServiceClient
+ import com.google.cloud.secretmanager.v1.SecretVersionName
+ import com.google.firebase.FirebaseApp
+ import com.google.firebase.FirebaseOptions
+ import de.flapdoodle.embed.mongo.spring.autoconfigure.EmbeddedMongoAutoConfiguration
+ import ng.cove.web.component.SmsOtpService
+ import org.springframework.boot.SpringApplication
+ import org.springframework.boot.autoconfigure.SpringBootApplication
+ import org.springframework.boot.autoconfigure.data.mongo.MongoDataAutoConfiguration
+ import org.springframework.boot.context.event.ApplicationStartedEvent
+ import org.springframework.cache.annotation.EnableCaching
+ import org.springframework.cache.caffeine.CaffeineCacheManager
+ import org.springframework.context.ApplicationListener
+ import org.springframework.context.annotation.Bean
+ import org.springframework.scheduling.annotation.EnableAsync
+ import org.springframework.scheduling.annotation.EnableScheduling
+ import org.springframework.web.client.RestTemplate
+ import java.util.concurrent.TimeUnit
+ 
+ @EnableAsync
+ @EnableCaching
+ @SpringBootApplication(exclude = [MongoDataAutoConfiguration::class, EmbeddedMongoAutoConfiguration::class])
+ class App {
+ 
+     @Bean
+     fun restTemplate() = RestTemplate()
+ 
+     @Bean
+     fun caffeineConfig(): Caffeine<Any, Any> {
+         return Caffeine.newBuilder().expireAfterWrite(60, TimeUnit.MINUTES)
+     }
+ 
+     @Bean
+     fun cacheManager(caffeine: Caffeine<Any, Any>): CaffeineCacheManager {
+         val cacheManager = CaffeineCacheManager()
+         cacheManager.setCaffeine(caffeine)
+         return cacheManager
+     }
+ }
+ 
+ fun main(args: Array<String>) {
+     val app = SpringApplication(App::class.java)
+ 
+     val startedEvent = ApplicationListener<ApplicationStartedEvent> { event ->
+ 
+         val profiles = event.applicationContext.environment.activeProfiles
+         // profiles is sometimes empty when running on a production server because it
+         // has not been loaded from the application.properties file at this point
+         val profile = profiles.getOrNull(0) ?: "prod"
+ 
+         when (profile) {
+             "test" -> {}
+             "dev" -> FirebaseApp.initializeApp()
+             "prod" -> {
+                 // Get secrets from GCP
+                 val gcpProjectId = "gatedaccessdev"
+                 val firebaseSecret = SecretVersionName.of(gcpProjectId, "firebase-service-account", "1")
+                 val termiiSecret = SecretVersionName.of(gcpProjectId, "termii-key", "1")
+ 
+                 // Auto-closable
+                 SecretManagerServiceClient.create().use {
+                     // Set Termii API Key to service
+                     val termiiSecretPayload =
+                         it.accessSecretVersion(termiiSecret).payload.data.toByteArray().inputStream()
+                     val smsOtpService = event.applicationContext.getBean(SmsOtpService::class.java)
+                     smsOtpService.termiiApiKey = String(termiiSecretPayload.readAllBytes())
+                     // Init Firebase with service account
+                     val serviceAccountStream =
+                         it.accessSecretVersion(firebaseSecret).payload.data.toByteArray().inputStream()
+                     val options = FirebaseOptions.builder()
+                         .setCredentials(GoogleCredentials.fromStream(serviceAccountStream))
+                         .build()
+                     FirebaseApp.initializeApp(options)
+                 }
+             }
+         }
+ 
+     }
+ 
+     app.addListeners(startedEvent)
+     app.run(*args)
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-2/index.html b/htmlReport/ns-2/index.html new file mode 100644 index 0000000..c8a9d13 --- /dev/null +++ b/htmlReport/ns-2/index.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_BLOCK.html b/htmlReport/ns-2/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..4a73f6c --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_BLOCK.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-2/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..31b7673 --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_CLASS.html b/htmlReport/ns-2/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..4757c0c --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_CLASS.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-2/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..9efb8ef --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_LINE.html b/htmlReport/ns-2/index_SORT_BY_LINE.html new file mode 100644 index 0000000..90e3537 --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_LINE.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-2/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..30e5648 --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_METHOD.html b/htmlReport/ns-2/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..502d7c4 --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_METHOD.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-2/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..d4b403e --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-2/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..12f4b5c --- /dev/null +++ b/htmlReport/ns-2/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,167 @@ + + + + + + Coverage Report > ng.cove.web.component + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.component

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.component + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-2/sources/source-1.html b/htmlReport/ns-2/sources/source-1.html new file mode 100644 index 0000000..0fd4e60 --- /dev/null +++ b/htmlReport/ns-2/sources/source-1.html @@ -0,0 +1,219 @@ + + + + + + + + Coverage Report > SmsOtpService + + + + + + +
+ + +

Coverage Summary for Class: SmsOtpService (ng.cove.web.component)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
SmsOtpService + + 100% + + + (1/1) + + + + 100% + + + (5/5) + + + + 50% + + + (3/6) + + + + 84.3% + + + (43/51) + +
+ +
+
+ + +
+ package ng.cove.web.component
+ 
+ import com.google.gson.JsonObject
+ import ng.cove.web.http.body.OtpRefBody
+ import org.slf4j.Logger
+ import org.slf4j.LoggerFactory
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.beans.factory.annotation.Value
+ import org.springframework.context.annotation.Bean
+ import org.springframework.http.HttpEntity
+ import org.springframework.http.HttpHeaders
+ import org.springframework.http.HttpStatus
+ import org.springframework.http.MediaType
+ import org.springframework.stereotype.Service
+ import org.springframework.web.client.RestTemplate
+ import java.util.*
+ 
+ const val SEND_URL: String = "https://api.ng.termii.com/api/sms/otp/send"
+ const val VERIFY_URL = "https://api.ng.termii.com/api/sms/otp/verify"
+ 
+ 
+ @Service
+ class SmsOtpService {
+ 
+     val logger: Logger = LoggerFactory.getLogger(this::class.java)
+ 
+     // Set in ApplicationStartup
+     var termiiApiKey: String? = null
+ 
+     @Value("\${otp.expiry-mins}")
+     var otpExpiryMins: Int = 1
+ 
+     @Autowired
+     lateinit var template : RestTemplate
+ 
+     fun sendOtp(phone: String): OtpRefBody? {
+ 
+         val requestBody = JsonObject()
+         requestBody.addProperty("api_key", termiiApiKey)
+         requestBody.addProperty("from", "N-Alert")
+         requestBody.addProperty("to", phone)
+         requestBody.addProperty("message_type", "NUMERIC")
+         requestBody.addProperty("channel", "dnd")
+         requestBody.addProperty("pin_attempts", 5)
+         requestBody.addProperty("pin_time_to_live", otpExpiryMins)
+         requestBody.addProperty("pin_length", 6)
+         requestBody.addProperty("pin_placeholder", "<otp>")
+         requestBody.addProperty("message_text", "Your login OTP is: <otp>")
+ 
+ 
+         val headers: HttpHeaders = HttpHeaders().apply {
+             contentType = MediaType.APPLICATION_JSON
+         }
+         val entity = HttpEntity(requestBody.toString(), headers)
+         try {
+ 
+             val response = template
+                 .postForEntity(SEND_URL, entity, Map::class.java)
+ 
+             val result = response.body!!
+             // Log error when Termii returns any other status
+             result["status"].takeIf { it != 200 }?.let {
+                 throw Exception("SMS OTP provide error code: $it")
+             }
+ 
+             val ref = result["pinId"] as String
+             val futureDateTime = Date().toInstant()
+                 .plusSeconds(otpExpiryMins.toLong() * 60)
+             val expiry = Date.from(futureDateTime)
+ 
+             return OtpRefBody(ref, phone, expiry, null)
+         } catch (e: Exception) {
+             logger.error(e.localizedMessage)
+             return null
+         }
+ 
+     }
+ 
+     fun verifyOtp(otp: String, ref: String): String? {
+         val requestBody = JsonObject()
+         requestBody.addProperty("api_key", termiiApiKey)
+         requestBody.addProperty("pin_id", ref)
+         requestBody.addProperty("pin", otp)
+ 
+ 
+         val headers: HttpHeaders = HttpHeaders().apply {
+             contentType = MediaType.APPLICATION_JSON
+         }
+         val entity = HttpEntity(requestBody.toString(), headers)
+         try {
+             val result = template
+                 .postForEntity(VERIFY_URL, entity, Map::class.java)
+ 
+             if (result.statusCode != HttpStatus.OK)
+                 return null
+ 
+             return result.body!!["msisdn"] as String
+         } catch (e: Exception) {
+             logger.error(e.localizedMessage)
+             return null
+         }
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-3/index.html b/htmlReport/ns-3/index.html new file mode 100644 index 0000000..8e49fc8 --- /dev/null +++ b/htmlReport/ns-3/index.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_BLOCK.html b/htmlReport/ns-3/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..00f3429 --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_BLOCK.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-3/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..cdcc002 --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_CLASS.html b/htmlReport/ns-3/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..56abf44 --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_CLASS.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-3/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..860921d --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_LINE.html b/htmlReport/ns-3/index_SORT_BY_LINE.html new file mode 100644 index 0000000..48c64ed --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_LINE.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-3/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..f007c24 --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_METHOD.html b/htmlReport/ns-3/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..cf3284f --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_METHOD.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-3/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..130f803 --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-3/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..f578d15 --- /dev/null +++ b/htmlReport/ns-3/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,307 @@ + + + + + + Coverage Report > ng.cove.web.config + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.config

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.config + + 83.3% + + + (5/6) + + + + 86.7% + + + (13/15) + + + + 50% + + + (5/10) + + + + 50.9% + + + (29/57) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
WebConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (7/7) + +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/22) + +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + + 100% + + + (1/1) + +
MongoConfig + + 100% + + + (1/1) + + + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
AuditorConfig + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-3/sources/source-1.html b/htmlReport/ns-3/sources/source-1.html new file mode 100644 index 0000000..e3d69cf --- /dev/null +++ b/htmlReport/ns-3/sources/source-1.html @@ -0,0 +1,114 @@ + + + + + + + + Coverage Report > AuditorAwareImpl + + + + + + +
+ + +

Coverage Summary for Class: AuditorAwareImpl (ng.cove.web.config)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
AuditorAwareImpl + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 100% + + + (2/2) + +
+ +
+
+ + +
+ package ng.cove.web.config
+ 
+ import org.springframework.data.domain.AuditorAware
+ import java.util.*
+ 
+ class AuditorAwareImpl : AuditorAware<String> {
+     override fun getCurrentAuditor(): Optional<String> {
+         return Optional.of("lauren")
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-3/sources/source-2.html b/htmlReport/ns-3/sources/source-2.html new file mode 100644 index 0000000..69c9c37 --- /dev/null +++ b/htmlReport/ns-3/sources/source-2.html @@ -0,0 +1,136 @@ + + + + + + + + Coverage Report > AuditorConfig + + + + + + +
+ + +

Coverage Summary for Class: AuditorConfig (ng.cove.web.config)

+ + + + + + + + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
AuditorConfig + + 100% + + + (2/2) + + + + 100% + + + (2/2) + +
AuditorConfig$$SpringCGLIB$$0
AuditorConfig$$SpringCGLIB$$FastClass$$0
AuditorConfig$$SpringCGLIB$$FastClass$$1
Total + + 100% + + + (2/2) + + + + 100% + + + (2/2) + +
+ +
+
+ + +
+ package ng.cove.web.config
+ 
+ import org.springframework.context.annotation.Bean
+ import org.springframework.context.annotation.Configuration
+ import org.springframework.context.annotation.Profile
+ import org.springframework.data.domain.AuditorAware
+ import org.springframework.data.mongodb.config.EnableMongoAuditing
+ 
+ @Configuration
+ @EnableMongoAuditing(auditorAwareRef = "auditorProvider")
+ class AuditorConfig {
+     @Bean
+     fun auditorProvider(): AuditorAware<String> {
+         return AuditorAwareImpl()
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-3/sources/source-3.html b/htmlReport/ns-3/sources/source-3.html new file mode 100644 index 0000000..38a683f --- /dev/null +++ b/htmlReport/ns-3/sources/source-3.html @@ -0,0 +1,210 @@ + + + + + + + + Coverage Report > MongoConfig + + + + + + +
+ + +

Coverage Summary for Class: MongoConfig (ng.cove.web.config)

+ + + + + + + + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Branch, % + + Line, % +
MongoConfig + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
MongoConfig$$SpringCGLIB$$0
MongoConfig$$SpringCGLIB$$FastClass$$0
Total + + 100% + + + (6/6) + + + + 50% + + + (5/10) + + + + 73.9% + + + (17/23) + +
+ +
+
+ + +
+ package ng.cove.web.config
+ 
+ import com.google.cloud.secretmanager.v1.SecretManagerServiceClient
+ import com.google.cloud.secretmanager.v1.SecretVersionName
+ import com.mongodb.client.MongoClient
+ import com.mongodb.client.MongoClients
+ import de.flapdoodle.embed.mongo.commands.ServerAddress
+ import de.flapdoodle.embed.mongo.distribution.Version
+ import de.flapdoodle.embed.mongo.transitions.Mongod
+ import de.flapdoodle.embed.mongo.transitions.RunningMongodProcess
+ import de.flapdoodle.reverse.TransitionWalker
+ import org.springframework.context.annotation.Configuration
+ import org.springframework.data.mongodb.config.AbstractMongoClientConfiguration
+ import org.springframework.data.mongodb.repository.config.EnableMongoRepositories
+ import org.springframework.web.context.WebApplicationContext
+ import javax.annotation.PreDestroy
+ 
+ //@Profile("!test")
+ @Configuration
+ @EnableMongoRepositories("ng.cove.web.data.repo")
+ class MongoConfig(val context: WebApplicationContext) : AbstractMongoClientConfiguration() {
+ 
+     var embeddedMongo: TransitionWalker.ReachedState<RunningMongodProcess>? = null
+ 
+     override fun getDatabaseName(): String {
+         val profiles = context.environment.activeProfiles
+         return if (profiles.getOrNull(0) == "test"){
+             "test"
+         }else {
+             "dev"
+         }
+     }
+ 
+     override fun mongoClient(): MongoClient {
+         val profiles = context.environment.activeProfiles
+         // profiles is sometimes empty when running on a production server because it
+         // has not been loaded from the application.properties file at this point
+         val profile = profiles.getOrNull(0) ?: "prod"
+ 
+         when (profile) {
+             "dev" -> return MongoClients.create("mongodb://localhost:27017/dev")
+             "prod" -> {
+                 //Auto-closable client
+                 SecretManagerServiceClient.create().use {
+                     val gcpProjectId = "gatedaccessdev"
+                     val dbSecret = SecretVersionName.of(gcpProjectId, "db-uri", "1")
+                     // Access the secret version.
+                     val dbSecretPayload = it.accessSecretVersion(dbSecret).payload.data.toByteArray().inputStream()
+                     //Close the client
+                     return MongoClients.create(String(dbSecretPayload.readAllBytes()))
+                 }
+             }
+             else -> {
+ 
+                 // Embedded Mongo instance for testing
+                 embeddedMongo = Mongod.instance().start(Version.Main.V7_0)
+                 val serverAddress: ServerAddress = embeddedMongo!!.current().serverAddress
+                 val host = serverAddress.host
+                 val port = serverAddress.port
+                 return MongoClients.create("mongodb://$host:$port/test")
+             }
+         }
+ 
+     }
+ 
+     @PreDestroy
+     fun onShutDown() {
+         embeddedMongo?.close()
+     }
+ 
+     override fun autoIndexCreation(): Boolean = true
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-3/sources/source-4.html b/htmlReport/ns-3/sources/source-4.html new file mode 100644 index 0000000..ca5935c --- /dev/null +++ b/htmlReport/ns-3/sources/source-4.html @@ -0,0 +1,122 @@ + + + + + + + + Coverage Report > ScheduleConfig + + + + + + +
+ + +

Coverage Summary for Class: ScheduleConfig (ng.cove.web.config)

+ + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
ScheduleConfig + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
ScheduleConfig$$SpringCGLIB$$0
Total + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+
+ + +
+ package ng.cove.web.config
+ 
+ import org.springframework.context.annotation.Configuration
+ import org.springframework.scheduling.annotation.EnableScheduling
+ 
+ @EnableScheduling
+ @Configuration
+ class ScheduleConfig
+
+
+
+ + + + + + diff --git a/htmlReport/ns-3/sources/source-5.html b/htmlReport/ns-3/sources/source-5.html new file mode 100644 index 0000000..b7eb938 --- /dev/null +++ b/htmlReport/ns-3/sources/source-5.html @@ -0,0 +1,149 @@ + + + + + + + + Coverage Report > SwaggerConfig + + + + + + +
+ + +

Coverage Summary for Class: SwaggerConfig (ng.cove.web.config)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
SwaggerConfig + + 0% + + + (0/1) + + + + 0% + + + (0/2) + + + + 0% + + + (0/22) + +
+ +
+
+ + +
+ package ng.cove.web.config
+ 
+ import io.swagger.v3.oas.models.Components
+ import io.swagger.v3.oas.models.ExternalDocumentation
+ import io.swagger.v3.oas.models.OpenAPI
+ import io.swagger.v3.oas.models.info.Info
+ import io.swagger.v3.oas.models.info.License
+ import io.swagger.v3.oas.models.security.SecurityRequirement
+ import io.swagger.v3.oas.models.security.SecurityScheme
+ import org.springframework.context.annotation.Bean
+ import org.springframework.context.annotation.Configuration
+ import org.springframework.context.annotation.Profile
+ 
+ @Profile("!test")
+ @Configuration
+ class SwaggerConfig {
+     private var schemeName: String = "bearerAuth"
+     private var bearerFormat: String = "JWT"
+     private var scheme: String = "bearer"
+ 
+     @Bean
+     fun caseOpenAPI(): OpenAPI {
+         return OpenAPI()
+             .addSecurityItem(
+                 SecurityRequirement()
+                     .addList(schemeName)
+             ).components(
+                 Components()
+                     .addSecuritySchemes(
+                         schemeName, SecurityScheme()
+                             .name(schemeName)
+                             .type(SecurityScheme.Type.HTTP)
+                             .bearerFormat(bearerFormat)
+                             .`in`(SecurityScheme.In.HEADER)
+                             .scheme(scheme)
+                     )
+             )
+             .info(
+                 Info()
+                     .title("cove.ng web service")
+                     .description("API documentation for Cove Platform")
+                     .version("1.0")
+             )
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-3/sources/source-6.html b/htmlReport/ns-3/sources/source-6.html new file mode 100644 index 0000000..1cbccae --- /dev/null +++ b/htmlReport/ns-3/sources/source-6.html @@ -0,0 +1,136 @@ + + + + + + + + Coverage Report > WebConfig + + + + + + +
+ + +

Coverage Summary for Class: WebConfig (ng.cove.web.config)

+ + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
WebConfig + + 100% + + + (2/2) + + + + 100% + + + (7/7) + +
WebConfig$$SpringCGLIB$$0
Total + + 100% + + + (2/2) + + + + 100% + + + (7/7) + +
+ +
+
+ + +
+ package ng.cove.web.config
+ 
+ import ng.cove.web.http.interceptor.AdminInterceptor
+ import ng.cove.web.http.interceptor.SecureInterceptor
+ import org.springframework.context.annotation.Configuration
+ import org.springframework.web.context.WebApplicationContext
+ import org.springframework.web.servlet.config.annotation.EnableWebMvc
+ import org.springframework.web.servlet.config.annotation.InterceptorRegistry
+ import org.springframework.web.servlet.config.annotation.WebMvcConfigurer
+ 
+ @Configuration
+ class WebConfig(val context: WebApplicationContext) : WebMvcConfigurer {
+     override fun addInterceptors(registry: InterceptorRegistry) {
+ 
+         registry.addInterceptor(SecureInterceptor(context))
+             .addPathPatterns("/secure/**")
+             .addPathPatterns("/admin/**")
+ 
+         registry.addInterceptor(AdminInterceptor())
+             .addPathPatterns("/admin/**").order(1)
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/index.html b/htmlReport/ns-4/index.html new file mode 100644 index 0000000..6bb36fd --- /dev/null +++ b/htmlReport/ns-4/index.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_BLOCK.html b/htmlReport/ns-4/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..54c1c84 --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_BLOCK.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-4/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..0a3b694 --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_CLASS.html b/htmlReport/ns-4/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..0a19f6e --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_CLASS.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-4/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..50a14d7 --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_LINE.html b/htmlReport/ns-4/index_SORT_BY_LINE.html new file mode 100644 index 0000000..de06554 --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_LINE.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-4/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..1960827 --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_METHOD.html b/htmlReport/ns-4/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..264a80c --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_METHOD.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-4/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..8fea9bd --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-4/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..65fdc1e --- /dev/null +++ b/htmlReport/ns-4/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,521 @@ + + + + + + Coverage Report > ng.cove.web.data.model + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.model

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.data.model + + 73.3% + + + (11/15) + + + + 60.4% + + + (61/101) + + + + 55.3% + + + (63/114) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-1.html b/htmlReport/ns-4/sources/source-1.html new file mode 100644 index 0000000..bf447ba --- /dev/null +++ b/htmlReport/ns-4/sources/source-1.html @@ -0,0 +1,167 @@ + + + + + + + + Coverage Report > Access + + + + + + +
+ + +

Coverage Summary for Class: Access (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
Access + + 0% + + + (0/1) + + + + 0% + + + (0/11) + + + + 0% + + + (0/12) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonIgnore
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import jakarta.validation.constraints.Future
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.annotation.LastModifiedDate
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.DBRef
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class Access {
+ 
+     @Id
+     var id: AccessId? = null
+ 
+     @field:Field("head_count")
+     var headCount: Int = 1
+ 
+     @field:NonNull
+     @field:Indexed
+     @DBRef(lazy = true)
+     var host: Member? = null
+ 
+     @field:Indexed
+     @field:Field("checked_in_at")
+     var checkedInAt: Date? = null
+ 
+     @field:Field("checked_out_at")
+     val checkedOutAt: Date? = null
+ 
+     @DBRef(lazy = true)
+     @field:Field("checked_in_by")
+     var checkedInBy: SecurityGuard? = null
+ 
+     @DBRef(lazy = true)
+     @field:Field("checked_out_by")
+     val checkedOutBy: SecurityGuard? = null
+ 
+     @field:Field(value = "duration_of_visit")
+     var durationOfVisit: Long? = null
+ 
+     @field:Indexed
+     @field:NonNull
+     @field:Future(message = "Valid until must be in the future")
+     @field:Field("valid_until")
+     var validUntil: Date? = null
+ 
+     //    @field:CreatedDate //Only works if we use autogenerated ID
+     @field:Field("created_at")
+     var createdAt: Date? = null
+ 
+     @JsonIgnore
+     @field:Field("updated_at")
+     @field:LastModifiedDate
+     var updatedAt: Date? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-2.html b/htmlReport/ns-4/sources/source-2.html new file mode 100644 index 0000000..e0235e9 --- /dev/null +++ b/htmlReport/ns-4/sources/source-2.html @@ -0,0 +1,122 @@ + + + + + + + + Coverage Report > AccessId + + + + + + +
+ + +

Coverage Summary for Class: AccessId (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
AccessId + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/5) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import org.springframework.data.mongodb.core.mapping.Field
+ 
+ class AccessId {
+     constructor(code: String, communityId: String) {
+         this.code = code
+         this.communityId = communityId
+     }
+ 
+     @Field("code")
+     @JsonProperty("code")
+     var code: String
+     @Field("community_id")
+     @JsonProperty("community_id")
+     var communityId: String
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-3.html b/htmlReport/ns-4/sources/source-3.html new file mode 100644 index 0000000..5c75df0 --- /dev/null +++ b/htmlReport/ns-4/sources/source-3.html @@ -0,0 +1,153 @@ + + + + + + + + Coverage Report > AssignedLevy + + + + + + +
+ + +

Coverage Summary for Class: AssignedLevy (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
AssignedLevy + + 100% + + + (1/1) + + + + 71.4% + + + (5/7) + + + + 71.4% + + + (5/7) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import jakarta.validation.constraints.Future
+ import jakarta.validation.constraints.FutureOrPresent
+ import jakarta.validation.constraints.NotBlank
+ import jakarta.validation.constraints.NotNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.mongodb.core.index.CompoundIndex
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.Date
+ 
+ @Document("assigned_levy")
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ @CompoundIndex(name = "levy_member", def = "{'levy_id': 1, 'member_id': 1}", unique = true)
+ class AssignedLevy {
+     @Id
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     var id: String? = null
+ 
+     @Field("levy_id")
+     @field:NonNull
+     var levyId: String? = null
+ 
+     @Field("member_id")
+     @field:NonNull
+     var memberId: String? = null
+ 
+     @Field("assigned_by")
+     @field:NonNull
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     var assignedBy: String? = null
+ 
+     @FutureOrPresent(message = "Next payment due date cannot be in the past")
+     @field:NonNull
+     @field:NotNull
+     @Field("next_payment_due")
+     var nextPaymentDue: Date? = null
+ 
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     @Field("created_at")
+     @CreatedDate
+     var createdAt: Date? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-4.html b/htmlReport/ns-4/sources/source-4.html new file mode 100644 index 0000000..dc7fbe3 --- /dev/null +++ b/htmlReport/ns-4/sources/source-4.html @@ -0,0 +1,150 @@ + + + + + + + + Coverage Report > Community + + + + + + +
+ + +

Coverage Summary for Class: Community (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
Community + + 100% + + + (1/1) + + + + 100% + + + (12/12) + + + + 100% + + + (12/12) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import jakarta.validation.constraints.Size
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.annotation.LastModifiedDate
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ open class Community {
+     @Id
+     var id: String? = null
+     var name: @Size(max = 255) String? = null
+     var address: @Size(max = 255) String? = null
+     var country: @Size(max = 255) String? = null
+     var state: @Size(max = 255) String? = null
+     var desc: @Size(max = 255) String? = null
+ 
+     @Field("super_admin_id")
+     @Indexed(unique = true)
+     var superAdminId: String? = null
+ 
+     @NonNull
+     @Indexed
+     @Field("admin_ids")
+     var adminIds: Set<String>? = null
+ 
+     @Field("created_at")
+     @CreatedDate
+     var createdAt: Date? = null
+ 
+     @Field("banner_url")
+     var bannerUrl: String? = null
+ 
+     @Field("last_modified_at")
+     @LastModifiedDate
+     var lastModifiedAt: Date? = null
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-5.html b/htmlReport/ns-4/sources/source-5.html new file mode 100644 index 0000000..5e8e56b --- /dev/null +++ b/htmlReport/ns-4/sources/source-5.html @@ -0,0 +1,109 @@ + + + + + + + + Coverage Report > Gender + + + + + + +
+ + +

Coverage Summary for Class: Gender (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
Gender + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ enum class Gender {
+     male, female
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-6.html b/htmlReport/ns-4/sources/source-6.html new file mode 100644 index 0000000..f223425 --- /dev/null +++ b/htmlReport/ns-4/sources/source-6.html @@ -0,0 +1,159 @@ + + + + + + + + Coverage Report > JoinRequest + + + + + + +
+ + +

Coverage Summary for Class: JoinRequest (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
JoinRequest + + 100% + + + (1/1) + + + + 90% + + + (9/10) + + + + 90% + + + (9/10) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import jakarta.validation.constraints.NotBlank
+ import jakarta.validation.constraints.NotNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document("join_request")
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class JoinRequest {
+ 
+     @Id
+     @field:NotNull
+     var id: JoinRequestId? = null
+ 
+     @NonNull
+     @Field("referrer_id")
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     var referrerId: String? = null
+ 
+     @field:NonNull
+     @Field("first_name")
+     @field:NotBlank(message = "First name cannot be a null")
+     var firstName: String? = null
+ 
+     @NonNull
+     @Field("last_name")
+     @field:NotBlank(message = "Last name cannot be a null")
+     var lastName: String? = null
+ 
+     @field:NotNull
+     @field:NonNull
+     var gender: Gender? = null
+ 
+     var address: String? = null
+ 
+     @Field("created_at")
+     @CreatedDate
+     var createdAt: Date? = null
+ 
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     @Field("accepted_at")
+     var acceptedAt: Date? = null
+ 
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     @Field("approved_by")
+     var approvedBy: String? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-7.html b/htmlReport/ns-4/sources/source-7.html new file mode 100644 index 0000000..7ab8454 --- /dev/null +++ b/htmlReport/ns-4/sources/source-7.html @@ -0,0 +1,114 @@ + + + + + + + + Coverage Report > JoinRequestId + + + + + + +
+ + +

Coverage Summary for Class: JoinRequestId (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
JoinRequestId + + 100% + + + (1/1) + + + + 100% + + + (3/3) + + + + 100% + + + (3/3) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import jakarta.validation.constraints.NotBlank
+ import org.springframework.data.mongodb.core.mapping.Field
+ 
+ class JoinRequestId {
+     @field:NotBlank @Field("community_id") @field:JsonProperty("community_id") lateinit var communityId: String
+     @field:NotBlank lateinit var phone: String
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-8.html b/htmlReport/ns-4/sources/source-8.html new file mode 100644 index 0000000..41c7056 --- /dev/null +++ b/htmlReport/ns-4/sources/source-8.html @@ -0,0 +1,158 @@ + + + + + + + + Coverage Report > Levy + + + + + + +
+ + +

Coverage Summary for Class: Levy (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
Levy + + 100% + + + (1/1) + + + + 75% + + + (6/8) + + + + 75% + + + (6/8) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import jakarta.validation.constraints.Min
+ import jakarta.validation.constraints.NotNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.annotation.LastModifiedDate
+ import org.springframework.data.mongodb.core.index.CompoundIndex
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document
+ @CompoundIndex(name = "community_id_title", def = "{'community_id': 1, 'title': 1}", unique = true)
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class Levy {
+     @Id
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     var id: String? = null
+ 
+     @field:NonNull
+     var title: String? = null
+ 
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     @field:Indexed
+     @field:NonNull
+     @Field("community_id")
+     var communityId: String? = null
+ 
+     @field:NonNull
+     @field:NotNull
+     @Min(1, message = "Amount must be greater than 0")
+     @Field("amount")
+     var amount: Double? = null
+ 
+     @field:NonNull
+     @field:NotNull
+     var type: LevyType? = null
+ 
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     @CreatedDate
+     @Field("created_at")
+     var createdAt: Date? = null
+ 
+     @JsonProperty(access = JsonProperty.Access.READ_ONLY)
+     @Field("updated_at")
+     @LastModifiedDate
+     var updatedAt: Date? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-9.html b/htmlReport/ns-4/sources/source-9.html new file mode 100644 index 0000000..d41984e --- /dev/null +++ b/htmlReport/ns-4/sources/source-9.html @@ -0,0 +1,126 @@ + + + + + + + + Coverage Report > LevyPayment + + + + + + +
+ + +

Coverage Summary for Class: LevyPayment (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
LevyPayment + + 100% + + + (1/1) + + + + 80% + + + (4/5) + + + + 80% + + + (4/5) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.mongodb.lang.NonNull
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.Date
+ 
+ @Document("levy_payments")
+ class LevyPayment {
+     @Id
+     var id: String? = null
+     @field:NonNull
+     @Field("levy_id")
+     var levyId: String? = null
+     @field:NonNull
+     @Field("member_id")
+     var memberId: String? = null
+     @field:NonNull
+     @Field("due_date")
+     var dueDate: Date? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-a.html b/htmlReport/ns-4/sources/source-a.html new file mode 100644 index 0000000..70a0dc1 --- /dev/null +++ b/htmlReport/ns-4/sources/source-a.html @@ -0,0 +1,108 @@ + + + + + + + + Coverage Report > LevyType + + + + + + +
+ + +

Coverage Summary for Class: LevyType (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
LevyType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ enum class LevyType {Month, Annual
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-b.html b/htmlReport/ns-4/sources/source-b.html new file mode 100644 index 0000000..188bae6 --- /dev/null +++ b/htmlReport/ns-4/sources/source-b.html @@ -0,0 +1,176 @@ + + + + + + + + Coverage Report > Member + + + + + + +
+ + +

Coverage Summary for Class: Member (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
Member + + 100% + + + (1/1) + + + + 93.8% + + + (15/16) + + + + 93.8% + + + (15/16) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonIgnore
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.annotation.LastModifiedDate
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.DBRef
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ open class Member {
+     @Id
+     var id: String? = null
+ 
+     @Field("first_name")
+     @NonNull
+     var firstName: String? = null
+ 
+     @Field("last_name")
+     @NonNull
+     var lastName: String? = null
+ 
+     @NonNull
+     var gender: String? = null
+ 
+     @Indexed(unique = true)
+     var phone: String? = null
+ 
+     @Field(value = "phone_verified_at")
+     var phoneVerifiedAt: Date? = null
+ 
+     var address: String? = null
+ 
+     @Field(value = "photo_url")
+     var photoUrl: String? = null
+ 
+     @DBRef
+     @NonNull
+     var community: Community? = null
+ 
+     @CreatedDate
+     @Field("created_at")
+     var createdAt: Date? = null
+ 
+     @LastModifiedDate
+     @Field("last_modified_at")
+     var lastModifiedAt: Date? = null
+ 
+     /** Device info start**/
+     @Field("device_name")
+     var deviceName: String? = null
+ 
+     @JsonIgnore
+     @Field("fcm_token")
+     var fcmToken: String? = null
+ 
+     @Field("last_login_at")
+     var lastLoginAt: Date? = null
+ 
+     /** Device info end**/
+ 
+     @JsonIgnore
+     @Field("test_otp")
+     var testOtp: String? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-c.html b/htmlReport/ns-4/sources/source-c.html new file mode 100644 index 0000000..45a080c --- /dev/null +++ b/htmlReport/ns-4/sources/source-c.html @@ -0,0 +1,135 @@ + + + + + + + + Coverage Report > Occupant + + + + + + +
+ + +

Coverage Summary for Class: Occupant (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
Occupant + + 0% + + + (0/1) + + + + 0% + + + (0/5) + + + + 0% + + + (0/13) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import com.mongodb.lang.NonNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.DBRef
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document
+ class Occupant(
+     @Id var id: String,
+     @Field(
+         "first_name"
+     ) @field:Indexed var firstName: String,
+     @Field(
+         "last_name"
+     ) @Indexed var lastName: String,
+     var phone: String,
+     var gender: String,
+     @DBRef(
+         lazy = true
+     ) @field:NonNull @param:NonNull var guardian: Member,
+     @field:Field(
+         "photo_url"
+     ) var photoUrl: String,
+     @field:CreatedDate @field:JsonProperty("created_at") @field:Field("created_at") var createdAt: Date
+ )
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-d.html b/htmlReport/ns-4/sources/source-d.html new file mode 100644 index 0000000..87ca1dc --- /dev/null +++ b/htmlReport/ns-4/sources/source-d.html @@ -0,0 +1,137 @@ + + + + + + + + Coverage Report > PhoneOtp + + + + + + +
+ + +

Coverage Summary for Class: PhoneOtp (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
PhoneOtp + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 71.4% + + + (5/7) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.mongodb.lang.NonNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.mongodb.core.index.CompoundIndex
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document("phone_otp")
+ @CompoundIndex(name = "phone_ref", def = "{'phone': 1, 'ref': 1}", unique = true)
+ class PhoneOtp {
+ 
+     @Id
+     var id: String? = null
+ 
+     @field:NonNull
+     var phone: String? = null
+ 
+     @field:NonNull
+     var ref: String? = null
+ 
+     var type: UserType = UserType.Member
+ 
+     @field:Field("expire_at")
+     var expireAt: Date? = null
+ 
+     @CreatedDate
+     @Field("created_at")
+     var createdAt: Date? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-e.html b/htmlReport/ns-4/sources/source-e.html new file mode 100644 index 0000000..2339553 --- /dev/null +++ b/htmlReport/ns-4/sources/source-e.html @@ -0,0 +1,164 @@ + + + + + + + + Coverage Report > SecurityGuard + + + + + + +
+ + +

Coverage Summary for Class: SecurityGuard (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
SecurityGuard + + 0% + + + (0/1) + + + + 0% + + + (0/12) + + + + 0% + + + (0/12) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.annotation.JsonIgnore
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import org.springframework.data.annotation.CreatedDate
+ import org.springframework.data.annotation.Id
+ import org.springframework.data.mongodb.core.index.Indexed
+ import org.springframework.data.mongodb.core.mapping.Document
+ import org.springframework.data.mongodb.core.mapping.Field
+ import java.util.*
+ 
+ @Document("security_guard")
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class SecurityGuard {
+     @Id
+     var id: String? = null
+ 
+     @Field("first_name")
+     @NonNull
+     var firstName: String? = null
+ 
+     @Field("last_name")
+     @NonNull
+     var lastName: String? = null
+ 
+     @NonNull
+     @Indexed(unique = true)
+     var phone: String? = null
+ 
+     @Field("phone_verified_at")
+     var phoneVerifiedAt: Date? = null
+ 
+     @NonNull
+     @Indexed
+     @Field("community_id")
+     var communityId: String? = null
+ 
+     @CreatedDate
+     @Field("created_at")
+     var createdAt: Date? = null
+ 
+     /** Device info start**/
+     @Field("device_name")
+     var deviceName: String? = null
+ 
+     @JsonIgnore
+     @Field("fcm_token")
+     var fcmToken: String? = null
+ 
+     @Field("last_login_at")
+     var lastLoginAt: Date? = null
+     /** Device info end**/
+ 
+     @JsonIgnore
+     @Field("test_otp")
+     var testOtp: String? = null
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-4/sources/source-f.html b/htmlReport/ns-4/sources/source-f.html new file mode 100644 index 0000000..abcee63 --- /dev/null +++ b/htmlReport/ns-4/sources/source-f.html @@ -0,0 +1,113 @@ + + + + + + + + Coverage Report > UserType + + + + + + +
+ + +

Coverage Summary for Class: UserType (ng.cove.web.data.model)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
UserType + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (2/2) + +
+ +
+
+ + +
+ package ng.cove.web.data.model
+ 
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ 
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ enum class UserType { Member
+ , Guard
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-5/index.html b/htmlReport/ns-5/index.html new file mode 100644 index 0000000..e638c8a --- /dev/null +++ b/htmlReport/ns-5/index.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_BLOCK.html b/htmlReport/ns-5/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..ee30284 --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_BLOCK.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-5/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..51b69f0 --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_CLASS.html b/htmlReport/ns-5/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..23c19e3 --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_CLASS.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-5/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..ca7854e --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_LINE.html b/htmlReport/ns-5/index_SORT_BY_LINE.html new file mode 100644 index 0000000..ab3bade --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_LINE.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-5/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..71b0e5c --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_METHOD.html b/htmlReport/ns-5/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..f3f02d1 --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_METHOD.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-5/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..b81b40c --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-5/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..b75be1b --- /dev/null +++ b/htmlReport/ns-5/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,224 @@ + + + + + + Coverage Report > ng.cove.web.http.body + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.body

+ + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Line, % +
ng.cove.web.http.body + + 50% + + + (2/4) + + + + 55.6% + + + (5/9) + + + + 45.5% + + + (5/11) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Line, % +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-5/sources/source-1.html b/htmlReport/ns-5/sources/source-1.html new file mode 100644 index 0000000..1b503b8 --- /dev/null +++ b/htmlReport/ns-5/sources/source-1.html @@ -0,0 +1,134 @@ + + + + + + + + Coverage Report > AccessInfoBody + + + + + + +
+ + +

Coverage Summary for Class: AccessInfoBody (ng.cove.web.http.body)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
AccessInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/3) + + + + 0% + + + (0/4) + +
+ +
+
+ + +
+ package ng.cove.web.http.body
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import com.mongodb.lang.NonNull
+ import jakarta.validation.constraints.Future
+ import jakarta.validation.constraints.Max
+ import jakarta.validation.constraints.Min
+ import jakarta.validation.constraints.NotNull
+ import java.util.*
+ 
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class AccessInfoBody {
+ 
+     @field:Min(value = 1, message = "Head count must be at least 1")
+     @field:Max(value = 10000, message = "Head count must be at most 10000")
+     @field:NotNull
+     val headCount = 1
+ 
+     @field:NotNull
+     @field:NonNull
+     @field:Future(message = "ValidUntil must be in the future")
+     val validUntil: Date? = null
+ 
+     @field:NotNull
+     @field:NonNull
+     @field:JsonProperty("duration_of_visit_in_sec")
+     val durationOfVisitInSec: Long? = null
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-5/sources/source-2.html b/htmlReport/ns-5/sources/source-2.html new file mode 100644 index 0000000..e880384 --- /dev/null +++ b/htmlReport/ns-5/sources/source-2.html @@ -0,0 +1,111 @@ + + + + + + + + Coverage Report > GuardInfoBody + + + + + + +
+ + +

Coverage Summary for Class: GuardInfoBody (ng.cove.web.http.body)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
GuardInfoBody + + 0% + + + (0/1) + + + + 0% + + + (0/1) + + + + 0% + + + (0/2) + +
+ +
+
+ + +
+ package ng.cove.web.http.body
+ 
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ 
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class GuardInfoBody (val firstName: String, val lastName: String, val phone: String)
+
+
+
+ + + + + + diff --git a/htmlReport/ns-5/sources/source-3.html b/htmlReport/ns-5/sources/source-3.html new file mode 100644 index 0000000..fa3ef44 --- /dev/null +++ b/htmlReport/ns-5/sources/source-3.html @@ -0,0 +1,116 @@ + + + + + + + + Coverage Report > LoginBody + + + + + + +
+ + +

Coverage Summary for Class: LoginBody (ng.cove.web.http.body)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
LoginBody + + 100% + + + (1/1) + + + + 100% + + + (4/4) + + + + 100% + + + (4/4) + +
+ +
+
+ + +
+ package ng.cove.web.http.body
+ 
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import jakarta.validation.constraints.NotBlank
+ 
+ @JsonNaming(PropertyNamingStrategies.SnakeCaseStrategy::class)
+ class LoginBody {
+     @field:NotBlank(message = "Otp cannot be blank") lateinit var otp: String
+     @field:NotBlank(message = "ref cannot be blank") lateinit var ref: String
+     @field:NotBlank(message = "deviceName cannot be blank") lateinit var deviceName: String
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-5/sources/source-4.html b/htmlReport/ns-5/sources/source-4.html new file mode 100644 index 0000000..f21ddd9 --- /dev/null +++ b/htmlReport/ns-5/sources/source-4.html @@ -0,0 +1,113 @@ + + + + + + + + Coverage Report > OtpRefBody + + + + + + +
+ + +

Coverage Summary for Class: OtpRefBody (ng.cove.web.http.body)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
OtpRefBody + + 100% + + + (1/1) + + + + 100% + + + (1/1) + + + + 100% + + + (1/1) + +
+ +
+
+ + +
+ package ng.cove.web.http.body
+ 
+ import com.fasterxml.jackson.annotation.JsonProperty
+ import com.fasterxml.jackson.databind.PropertyNamingStrategies
+ import com.fasterxml.jackson.databind.annotation.JsonNaming
+ import java.util.Date
+ 
+ 
+ data class OtpRefBody(var ref: String, var phone: String, @field:JsonProperty("expire_at") var expireAt: Date, @field:JsonProperty("daily_trial_left") var dailyTrialLeft: Int?)
+
+
+
+ + + + + + diff --git a/htmlReport/ns-6/index.html b/htmlReport/ns-6/index.html new file mode 100644 index 0000000..115c88b --- /dev/null +++ b/htmlReport/ns-6/index.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_BLOCK.html b/htmlReport/ns-6/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..c2ea686 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_BLOCK.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-6/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..40e0c77 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_CLASS.html b/htmlReport/ns-6/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..a67c982 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_CLASS.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-6/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..a2a02e9 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_LINE.html b/htmlReport/ns-6/index_SORT_BY_LINE.html new file mode 100644 index 0000000..2d68548 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_LINE.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-6/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..556fc02 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_METHOD.html b/htmlReport/ns-6/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..19d00ef --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_METHOD.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-6/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..25a9d98 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-6/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..1ec5090 --- /dev/null +++ b/htmlReport/ns-6/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,279 @@ + + + + + + Coverage Report > ng.cove.web.http.controller + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.controller

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.controller + + 100% + + + (5/5) + + + + 38.5% + + + (10/26) + + + + 0% + + + (0/8) + + + + 32.3% + + + (10/31) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 11.1% + + + (1/9) + +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 50% + + + (1/2) + +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + + 66.7% + + + (4/6) + +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + + 37.5% + + + (3/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-6/sources/source-1.html b/htmlReport/ns-6/sources/source-1.html new file mode 100644 index 0000000..ba6e610 --- /dev/null +++ b/htmlReport/ns-6/sources/source-1.html @@ -0,0 +1,236 @@ + + + + + + + + Coverage Report > AdminController + + + + + + +
+ + +

Coverage Summary for Class: AdminController (ng.cove.web.http.controller)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
AdminController + + 100% + + + (1/1) + + + + 37.5% + + + (3/8) + + + + 37.5% + + + (3/8) + +
+ +
+
+ + +
+ package ng.cove.web.http.controller
+ 
+ import ng.cove.web.data.model.JoinRequestId
+ import ng.cove.web.data.model.Member
+ import ng.cove.web.http.body.GuardInfoBody
+ import ng.cove.web.service.CommunityService
+ import io.swagger.v3.oas.annotations.Operation
+ import io.swagger.v3.oas.annotations.media.Content
+ import io.swagger.v3.oas.annotations.media.Schema
+ import io.swagger.v3.oas.annotations.responses.ApiResponse
+ import jakarta.validation.Valid
+ import ng.cove.web.data.model.AssignedLevy
+ import ng.cove.web.data.model.Levy
+ import ng.cove.web.service.LevyService
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.http.MediaType
+ import org.springframework.http.ResponseEntity
+ import org.springframework.web.bind.annotation.*
+ @ApiResponse(
+     description = "Unauthorized",
+     responseCode = "401",
+     content = [Content(schema = Schema(implementation = String::class))]
+ )
+ @RestController
+ @RequestMapping("/admin")
+ class AdminController : BaseController() {
+ 
+     @Autowired
+     lateinit var communityService: CommunityService
+ 
+     @Autowired
+     lateinit var levyService: LevyService
+ 
+ 
+     @ApiResponse(
+         description = "Request updated",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @ApiResponse(
+         description = "Request already accepted",
+         responseCode = "202",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Handle community join request")
+     @PostMapping("/community/request/{accept}")
+     fun handleCommunityJoinRequest(
+         @RequestAttribute("user") user: Member,
+         @RequestBody @Valid requestId: JoinRequestId,
+         @PathVariable accept: Boolean
+     ): ResponseEntity<*> {
+         return communityService.handleCommunityJoinRequest(user.id!!, requestId, accept)
+     }
+ 
+     /**
+      * Add security guard to community
+      */
+     @ApiResponse(
+         description = "Security guard added",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @ApiResponse(
+         description = "User not Admin or Phone number already used",
+         responseCode = "400",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Add security guard to community")
+     @PostMapping("/guard")
+     fun addSecurityGuard(
+         @RequestAttribute("user") user: Member,
+         @RequestBody @Valid guardData: GuardInfoBody,
+     ): ResponseEntity<*> {
+         return communityService.addSecurityGuardToCommunity(guardData, user.id!!)
+     }
+ 
+     /**
+      * Remove Security Guard from a community
+      */
+     @ApiResponse(
+         description = "Security Guard removed",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Remove security guard from community")
+     @DeleteMapping("/guard/{guard_id}")
+     fun removeSecurityGuard(
+         @RequestAttribute("user") user: Member,
+         @PathVariable("guard_id") guardId: String,
+     ): ResponseEntity<*> {
+         return communityService.removeSecurityGuardFromCommunity(guardId, user.id!!)
+     }
+ 
+     @ApiResponse(
+         description = "Levy created",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = Levy::class),
+             mediaType = MediaType.APPLICATION_JSON_VALUE)]
+     )
+     @ApiResponse(
+         description = "Levy name already exists",
+         responseCode = "400",
+         content = [Content(schema = Schema(implementation = Levy::class),
+             mediaType = MediaType.APPLICATION_JSON_VALUE)]
+     )
+     @Operation(summary = "Create a new levy")
+     @PostMapping("/levy")
+     fun createLevy(@RequestAttribute("user") user: Member,
+                    @RequestBody levy: Levy): ResponseEntity<*> {
+         return levyService.createLevy(levy, user)
+     }
+ 
+     @ApiResponse(
+         description = "Levy assigned",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = AssignedLevy::class),
+             mediaType = MediaType.APPLICATION_JSON_VALUE)]
+     )
+     @ApiResponse(
+         description = "Levy already assigned",
+         responseCode = "400",
+         content = [Content(schema = Schema(implementation = Map::class),
+             mediaType = MediaType.APPLICATION_JSON_VALUE)]
+     )
+     @PostMapping("/levy/assign", consumes = [MediaType.APPLICATION_JSON_VALUE])
+     fun assignLevy(@RequestAttribute("user") user: Member,
+                    @Valid @RequestBody levy: AssignedLevy): ResponseEntity<*>{
+         return levyService.assignLevy(levy, user)
+     }
+ 
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-6/sources/source-2.html b/htmlReport/ns-6/sources/source-2.html new file mode 100644 index 0000000..e946aae --- /dev/null +++ b/htmlReport/ns-6/sources/source-2.html @@ -0,0 +1,156 @@ + + + + + + + + Coverage Report > BaseController + + + + + + +
+ + +

Coverage Summary for Class: BaseController (ng.cove.web.http.controller)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
BaseController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 0% + + + (0/8) + + + + 16.7% + + + (1/6) + +
+ +
+
+ + +
+ package ng.cove.web.http.controller
+ 
+ import com.mongodb.lang.NonNull
+ import jakarta.servlet.http.HttpServletRequest
+ import org.springframework.http.ResponseEntity
+ import org.springframework.http.converter.HttpMessageNotReadableException
+ import org.springframework.web.bind.MethodArgumentNotValidException
+ import org.springframework.web.bind.MissingRequestHeaderException
+ import org.springframework.web.bind.annotation.ControllerAdvice
+ import org.springframework.web.bind.annotation.ExceptionHandler
+ import org.springframework.web.servlet.NoHandlerFoundException
+ 
+ 
+ @ControllerAdvice
+ abstract class BaseController {
+ 
+     @ExceptionHandler(MissingRequestHeaderException::class)
+     fun handleMissingHeader(@NonNull ex: MissingRequestHeaderException): ResponseEntity<String> {
+         return ResponseEntity.badRequest().body("Required header '" + ex.headerName + "' is missing.")
+     }
+ 
+     @ExceptionHandler(MethodArgumentNotValidException::class)
+     fun handleMethodArgumentNotValid(@NonNull ex: MethodArgumentNotValidException): ResponseEntity<String> {
+         return ResponseEntity.badRequest()
+             .body(ex.bindingResult.fieldError?.defaultMessage ?: "Required argument is missing.")
+     }
+ 
+     @ExceptionHandler(HttpMessageNotReadableException::class)
+     fun handleHttpMessageNotReadable(@NonNull ex: HttpMessageNotReadableException): ResponseEntity<String> {
+         return ResponseEntity.badRequest().body(ex.cause?.localizedMessage ?: ex.localizedMessage)
+     }
+ 
+     @ExceptionHandler(NoHandlerFoundException::class)
+     fun handleNoHandlerFoundException(
+         ex: NoHandlerFoundException?, httpServletRequest: HttpServletRequest?
+     ): ResponseEntity<Any> {
+         return ResponseEntity.notFound().build()
+     }
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-6/sources/source-3.html b/htmlReport/ns-6/sources/source-3.html new file mode 100644 index 0000000..f560d8b --- /dev/null +++ b/htmlReport/ns-6/sources/source-3.html @@ -0,0 +1,201 @@ + + + + + + + + Coverage Report > DefaultController + + + + + + +
+ + +

Coverage Summary for Class: DefaultController (ng.cove.web.http.controller)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
DefaultController + + 100% + + + (1/1) + + + + 66.7% + + + (4/6) + + + + 66.7% + + + (4/6) + +
+ +
+
+ + +
+ package ng.cove.web.http.controller
+ 
+ import ng.cove.web.data.model.UserType
+ import ng.cove.web.http.body.LoginBody
+ import ng.cove.web.http.body.OtpRefBody
+ import ng.cove.web.service.UserService
+ import io.swagger.v3.oas.annotations.Operation
+ import io.swagger.v3.oas.annotations.media.Content
+ import io.swagger.v3.oas.annotations.media.Schema
+ import io.swagger.v3.oas.annotations.responses.ApiResponse
+ import jakarta.validation.Valid
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.http.MediaType
+ import org.springframework.http.ResponseEntity
+ import org.springframework.web.bind.annotation.*
+ 
+ @RestController
+ class DefaultController: BaseController() {
+ 
+     @Autowired
+     lateinit var userService: UserService
+ 
+     /**
+      * Enable the client app to request for OTP with user phone number
+      *
+      * @param phone user's phone number
+      * @return An OTP reference
+      */
+     @ApiResponse(
+         description = "OTP reference",
+         responseCode = "200",
+         content = [Content(schema = Schema(name = "message", implementation = OtpRefBody::class))]
+     )
+     @ApiResponse(
+         description = "No user found",
+         responseCode = "204",
+         content = [Content(schema = Schema(name = "message", implementation = String::class))]
+     )
+ 
+     @Operation(summary = "Log in User with phone number")
+     @GetMapping("/user/login")
+     fun loginUserWithPhone(
+         @RequestParam phone: String,
+     ): ResponseEntity<*> {
+         return userService.getOtpForLogin(phone, UserType.Member)
+     }
+ 
+     @ApiResponse(
+         description = "Verification successful",
+         responseCode = "200",
+         content = [Content(schema = Schema(name = "Custom token", implementation = String::class))]
+     )
+     @ApiResponse(
+         description = "Internal server error",
+         responseCode = "500"
+     )
+     @Operation(summary = "Verify login OTP for user")
+     @PostMapping("/user/login/verify")
+     fun verifyUserPhoneOtp(
+         @RequestBody @Valid body: LoginBody
+     ): ResponseEntity<*> {
+         return userService.verifyPhoneOtp(body, UserType.Member)
+     }
+ 
+     @ApiResponse(
+         description = "Otp reference",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = OtpRefBody::class),
+             mediaType = MediaType.APPLICATION_JSON_VALUE)]
+     )
+     @ApiResponse(
+         description = "Invalid OTP",
+         responseCode = "404",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Log in Security guard with phone number")
+     @GetMapping("/guard/login")
+     fun loginSecurityGuard(
+         @RequestParam phone: String,
+     ): ResponseEntity<*>? {
+         return userService.getOtpForLogin(phone, UserType.Guard)
+     }
+ 
+     @ApiResponse(
+         description = "New Firebase JWT",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Verify login OTP for Security guard")
+     @PostMapping("/guard/login/verify")
+     fun verifyGuardPhoneOtp(
+         @Valid @RequestBody body: LoginBody
+     ): ResponseEntity<*> {
+         return userService.verifyPhoneOtp(body, UserType.Guard)
+     }
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-6/sources/source-4.html b/htmlReport/ns-6/sources/source-4.html new file mode 100644 index 0000000..5d20c47 --- /dev/null +++ b/htmlReport/ns-6/sources/source-4.html @@ -0,0 +1,118 @@ + + + + + + + + Coverage Report > ErrorController + + + + + + +
+ + +

Coverage Summary for Class: ErrorController (ng.cove.web.http.controller)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
ErrorController + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + 50% + + + (1/2) + +
+ +
+
+ + +
+ package ng.cove.web.http.controller
+ 
+ import org.springframework.boot.web.servlet.error.ErrorController
+ import org.springframework.http.ResponseEntity
+ import org.springframework.stereotype.Controller
+ import org.springframework.web.bind.annotation.RequestMapping
+ 
+ @Controller
+ class ErrorController: ErrorController {
+     @RequestMapping("/error")
+     fun handleError(): ResponseEntity<Any> {
+         return ResponseEntity.notFound().build()
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-6/sources/source-5.html b/htmlReport/ns-6/sources/source-5.html new file mode 100644 index 0000000..3d78142 --- /dev/null +++ b/htmlReport/ns-6/sources/source-5.html @@ -0,0 +1,197 @@ + + + + + + + + Coverage Report > SecureController + + + + + + +
+ + +

Coverage Summary for Class: SecureController (ng.cove.web.http.controller)

+ + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Line, % +
SecureController + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + +
+ package ng.cove.web.http.controller
+ 
+ import io.swagger.v3.oas.annotations.Operation
+ import io.swagger.v3.oas.annotations.media.Content
+ import io.swagger.v3.oas.annotations.media.Schema
+ import io.swagger.v3.oas.annotations.responses.ApiResponse
+ import jakarta.validation.Valid
+ import ng.cove.web.data.model.Access
+ import ng.cove.web.data.model.JoinRequest
+ import ng.cove.web.data.model.Member
+ import ng.cove.web.data.model.SecurityGuard
+ import ng.cove.web.http.body.AccessInfoBody
+ import ng.cove.web.service.CommunityService
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.http.ResponseEntity
+ import org.springframework.web.bind.annotation.*
+ 
+ @ApiResponse(
+     description = "Unauthorized",
+     responseCode = "401",
+     content = [Content(schema = Schema(implementation = String::class))]
+ )
+ @RestController
+ @RequestMapping("/secure")
+ class SecureController : BaseController() {
+ 
+     @Autowired
+     lateinit var communityService: CommunityService
+ 
+     @ApiResponse(
+         description = "Request already sent",
+         responseCode = "208",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @ApiResponse(
+         description = "Request sent",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @PostMapping("/invite")
+     @Operation(summary = "Invite user to community")
+     fun inviteUserToCommunity(
+         @RequestAttribute("user") user: Member,
+         @Valid @RequestBody request: JoinRequest
+     ): ResponseEntity<*> {
+         return try {
+             communityService.inviteUserToCommunity(request, user)
+         } catch (e: Exception) {
+             ResponseEntity.internalServerError()
+                 .body(e.localizedMessage)
+         }
+     }
+ 
+     @ApiResponse(
+         description = "Success",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = Access::class))]
+     )
+     @ApiResponse(
+         description = "User does not have a community",
+         responseCode = "500",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Create access code for visitor")
+     @PostMapping("/access")
+     fun getAccessCode(
+         @RequestAttribute("user") user: Member,
+         @Valid @RequestBody info: AccessInfoBody
+     )
+             : ResponseEntity<*> {
+         return communityService.getAccessCodeForVisitor(info, user)
+     }
+ 
+     @ApiResponse(
+         description = "Check in successful",
+         responseCode = "200",
+         content = [Content(schema = Schema(implementation = Access::class))]
+     )
+     @ApiResponse(
+         description = "Access code not found",
+         responseCode = "204",
+         content = [Content(schema = Schema(implementation = String::class))]
+     )
+     @Operation(summary = "Check in visitor")
+     @GetMapping("/check-in/{code}")
+     fun checkIn(
+         @RequestAttribute("user") user: SecurityGuard,
+         @PathVariable code: String
+     )
+             : ResponseEntity<*> {
+         return communityService.checkInVisitor(code, user)
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-7/index.html b/htmlReport/ns-7/index.html new file mode 100644 index 0000000..ef4dbd0 --- /dev/null +++ b/htmlReport/ns-7/index.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_BLOCK.html b/htmlReport/ns-7/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..e801d2c --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_BLOCK.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-7/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..34b6f0f --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_CLASS.html b/htmlReport/ns-7/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..2c7c956 --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_CLASS.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-7/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..e61d159 --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_LINE.html b/htmlReport/ns-7/index_SORT_BY_LINE.html new file mode 100644 index 0000000..2a0b220 --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_LINE.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-7/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..48dab66 --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_METHOD.html b/htmlReport/ns-7/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..b13c340 --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_METHOD.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-7/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..78b1d2c --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-7/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..6bab281 --- /dev/null +++ b/htmlReport/ns-7/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,202 @@ + + + + + + Coverage Report > ng.cove.web.http.interceptor + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.http.interceptor

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.http.interceptor + + 100% + + + (2/2) + + + + 100% + + + (4/4) + + + + 66.7% + + + (8/12) + + + + 85.7% + + + (30/35) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-7/sources/source-1.html b/htmlReport/ns-7/sources/source-1.html new file mode 100644 index 0000000..21bbaaa --- /dev/null +++ b/htmlReport/ns-7/sources/source-1.html @@ -0,0 +1,140 @@ + + + + + + + + Coverage Report > AdminInterceptor + + + + + + +
+ + +

Coverage Summary for Class: AdminInterceptor (ng.cove.web.http.interceptor)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
AdminInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 100% + + + (8/8) + +
+ +
+
+ + +
+ package ng.cove.web.http.interceptor
+ 
+ import jakarta.servlet.http.HttpServletRequest
+ import jakarta.servlet.http.HttpServletResponse
+ import ng.cove.web.data.model.Member
+ import org.springframework.http.HttpStatus
+ import org.springframework.web.servlet.HandlerInterceptor
+ 
+ /***
+  * This interceptor runs after [SecureInterceptor] is successful.
+  * It checks if the user is an admin of their community.
+  * There is an assumption that [SecureInterceptor] sets a [Member] user as a request attribute.
+  */
+ class AdminInterceptor : HandlerInterceptor {
+     override fun preHandle(request: HttpServletRequest, response: HttpServletResponse, handler: Any): Boolean {
+         val member = request.getAttribute("user") as Member
+         member.community?.adminIds?.let {
+             if (it.contains(member.id))
+                 return true
+         }
+         response.status = HttpStatus.UNAUTHORIZED.value()
+         return false
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-7/sources/source-2.html b/htmlReport/ns-7/sources/source-2.html new file mode 100644 index 0000000..490eeb8 --- /dev/null +++ b/htmlReport/ns-7/sources/source-2.html @@ -0,0 +1,177 @@ + + + + + + + + Coverage Report > SecureInterceptor + + + + + + +
+ + +

Coverage Summary for Class: SecureInterceptor (ng.cove.web.http.interceptor)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
SecureInterceptor + + 100% + + + (1/1) + + + + 100% + + + (2/2) + + + + 66.7% + + + (4/6) + + + + 81.5% + + + (22/27) + +
+ +
+
+ + +
+ package ng.cove.web.http.interceptor
+ 
+ import com.google.firebase.auth.FirebaseAuth
+ import com.google.firebase.auth.FirebaseToken
+ import jakarta.servlet.http.HttpServletRequest
+ import jakarta.servlet.http.HttpServletResponse
+ import ng.cove.web.data.model.UserType
+ import ng.cove.web.service.CacheService
+ import org.slf4j.LoggerFactory
+ import org.springframework.http.HttpStatus
+ import org.springframework.web.context.WebApplicationContext
+ import org.springframework.web.servlet.HandlerInterceptor
+ import javax.annotation.Nonnull
+ 
+ class SecureInterceptor(private val context: WebApplicationContext) : HandlerInterceptor {
+ 
+     override fun preHandle(
+         request: HttpServletRequest,
+         @Nonnull response: HttpServletResponse,
+         @Nonnull handler: Any
+     ): Boolean {
+ 
+         request.getHeader("Authorization")?.let { it ->
+             try {
+                 val idToken =
+                     it.split(" ".toRegex()).dropLastWhile { it.isEmpty() }.toTypedArray()[1] //Remove Bearer prefix
+ 
+                 val userId: String
+                 val userType: UserType
+                 // Running on dev, skip token decode
+                 if (context.environment.activeProfiles[0] == "dev") {
+                     userType = UserType.Member // Manually update this to any type when running on dev
+                     userId = idToken
+                     println("Bearer is: $idToken")
+                 } else {
+                     val firebaseToken: FirebaseToken = FirebaseAuth.getInstance()
+                         .verifyIdToken(idToken, true)
+                     userType = UserType.valueOf(firebaseToken.claims["type"] as String)
+                     userId = firebaseToken.uid
+                 }
+ 
+                 val cacheService = context.getBean(CacheService::class.java)
+                 /** Get User model from DB and attach to request**/
+                 if (userType == UserType.Guard) {
+                     val guard = cacheService.getGuardById(userId)!!
+                     request.setAttribute("user", guard)
+                 } else {
+                     val member = cacheService.getMemberById(userId)!!
+                     request.setAttribute("user", member)
+                 }
+                 return true
+             } catch (e: Exception) {
+                 LoggerFactory.getLogger(this::class.java.simpleName).warn(e.localizedMessage)
+             }
+         }
+ 
+         response.status = HttpStatus.UNAUTHORIZED.value()
+         return false
+     }
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-8/index.html b/htmlReport/ns-8/index.html new file mode 100644 index 0000000..a559e00 --- /dev/null +++ b/htmlReport/ns-8/index.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_BLOCK.html b/htmlReport/ns-8/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..3e784dd --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_BLOCK.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-8/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..2f7e756 --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_CLASS.html b/htmlReport/ns-8/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..d4482ef --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_CLASS.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-8/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..6dcc82f --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_LINE.html b/htmlReport/ns-8/index_SORT_BY_LINE.html new file mode 100644 index 0000000..b7ec8c2 --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_LINE.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-8/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..41e6b16 --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_METHOD.html b/htmlReport/ns-8/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..57d55ab --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_METHOD.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-8/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..e275e4d --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-8/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..5be04d2 --- /dev/null +++ b/htmlReport/ns-8/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,293 @@ + + + + + + Coverage Report > ng.cove.web.service + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.service

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.service + + 100% + + + (5/5) + + + + 54.8% + + + (23/42) + + + + 32.9% + + + (25/76) + + + + 41.1% + + + (99/241) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
NotificationService + + 100% + + + (1/1) + + + + 50% + + + (1/2) + + + + + 11.1% + + + (1/9) + +
LevyService + + 100% + + + (1/1) + + + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
CacheService + + 100% + + + (1/1) + + + + 20% + + + (1/5) + + + + + 20% + + + (1/5) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-8/sources/source-1.html b/htmlReport/ns-8/sources/source-1.html new file mode 100644 index 0000000..506393f --- /dev/null +++ b/htmlReport/ns-8/sources/source-1.html @@ -0,0 +1,142 @@ + + + + + + + + Coverage Report > CacheService + + + + + + +
+ + +

Coverage Summary for Class: CacheService (ng.cove.web.service)

+ + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
CacheService + + 20% + + + (1/5) + + + + 20% + + + (1/5) + +
CacheService$$SpringCGLIB$$0
Total + + 20% + + + (1/5) + + + + 20% + + + (1/5) + +
+ +
+
+ + +
+ package ng.cove.web.service
+ 
+ import ng.cove.web.data.model.Member
+ import ng.cove.web.data.model.SecurityGuard
+ import ng.cove.web.data.repo.MemberRepo
+ import ng.cove.web.data.repo.SecurityGuardRepo
+ import ng.cove.web.util.CacheNames
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.cache.annotation.Cacheable
+ import org.springframework.stereotype.Service
+ 
+ @Service
+ class CacheService {
+     @Autowired
+     lateinit var memberRepo : MemberRepo
+     @Autowired
+     lateinit var guardRepo : SecurityGuardRepo
+ 
+     @Cacheable(value = [CacheNames.MEMBERS])
+     fun getMemberById(id: String): Member? {
+         return memberRepo.findById(id).orElse(null)
+     }
+ 
+     @Cacheable(value = [CacheNames.GUARDS])
+     fun getGuardById(id: String): SecurityGuard? {
+         return guardRepo.findById(id).orElse(null)
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-8/sources/source-2.html b/htmlReport/ns-8/sources/source-2.html new file mode 100644 index 0000000..3ea9fe9 --- /dev/null +++ b/htmlReport/ns-8/sources/source-2.html @@ -0,0 +1,358 @@ + + + + + + + + Coverage Report > CommunityService + + + + + + +
+ + +

Coverage Summary for Class: CommunityService (ng.cove.web.service)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
CommunityService + + 100% + + + (1/1) + + + + 33.3% + + + (5/15) + + + + 12.5% + + + (4/32) + + + + 19.8% + + + (22/111) + +
+ +
+
+ + +
+ package ng.cove.web.service
+ 
+ import com.google.firebase.auth.FirebaseAuth
+ import com.mongodb.DuplicateKeyException
+ import ng.cove.web.data.model.*
+ import ng.cove.web.data.repo.*
+ import ng.cove.web.http.body.AccessInfoBody
+ import ng.cove.web.http.body.GuardInfoBody
+ import ng.cove.web.util.ApiResponseMessage
+ import ng.cove.web.util.CodeGenerator
+ import org.slf4j.LoggerFactory
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.http.HttpStatus
+ import org.springframework.http.ResponseEntity
+ import org.springframework.stereotype.Service
+ import org.springframework.transaction.annotation.Transactional
+ import java.util.*
+ 
+ @Service
+ class CommunityService {
+ 
+     @Autowired
+     private lateinit var notificationService: NotificationService
+ 
+     @Autowired
+     lateinit var communityRepo: CommunityRepo
+ 
+     @Autowired
+     lateinit var memberRepo: MemberRepo
+ 
+     @Autowired
+     lateinit var joinRequestRepo: JoinRequestRepo
+ 
+     @Autowired
+     lateinit var securityGuardRepo: SecurityGuardRepo
+ 
+     @Autowired
+     lateinit var accessRepo: AccessRepo
+ 
+     @Autowired
+     lateinit var codeGenerator: CodeGenerator
+ 
+     @Autowired
+     lateinit var levyService: LevyService
+ 
+     @Autowired
+     lateinit var assignedLevyRepo: AssignedLevyRepo
+ 
+     private val logger = LoggerFactory.getLogger(CommunityService::class.java)
+ 
+ 
+     fun getAccessCodeForVisitor(info: AccessInfoBody, member: Member): ResponseEntity<*> {
+ 
+         try {
+ 
+             val communityId = member.community!!.id!!
+             var access: Access? = null
+             while (access == null) {
+                 access = Access()
+                 access.id = AccessId(communityId = communityId, code = codeGenerator.getCode())
+                 access.durationOfVisit = info.durationOfVisitInSec
+                 access.validUntil = info.validUntil
+                 access.headCount = info.headCount
+                 access.host = member
+                 access.createdAt = Date()
+ 
+                 access = try {
+                     accessRepo.save(access)
+                 } catch (e: DuplicateKeyException) {
+                     logger.warn("Access code duplication for community: ${member.community!!.id}")
+                     null // Set access to null to try again with another code
+                 }
+             }
+             return ResponseEntity.ok().body(access)
+ 
+         } catch (e: NullPointerException) {
+             logger.error("Access code generation failed: ${e.localizedMessage}")
+             return ResponseEntity.internalServerError().body("Community not found for user ${member.phone}")
+         }
+ 
+     }
+ 
+     fun checkInVisitor(code: String, guard: SecurityGuard): ResponseEntity<Any> {
+         try {
+             val accessId = AccessId(communityId = guard.communityId!!, code = code)
+             var access = accessRepo.findByIdAndValidUntilIsAfter(accessId, Date())
+                 ?: return ResponseEntity.noContent().build()
+ 
+             if (access.checkedInAt == null) {
+                 access.checkedInAt = Date()
+                 access.checkedInBy = guard
+             }
+             access = accessRepo.save(access) // Update access
+             return ResponseEntity.ok().body(access)
+ 
+         } catch (e: Exception) {
+             logger.error(e.localizedMessage)
+             return ResponseEntity.internalServerError().build()
+         }
+     }
+ 
+ 
+     /**
+      * @param adminUserId Identity of the admin of the community
+      * @param requestId
+      * @param accept      if the request was accepted or rejected
+      * @return status of the request
+      */
+     @Transactional
+     fun handleCommunityJoinRequest(
+         adminUserId: String,
+         requestId: JoinRequestId,
+         accept: Boolean
+     ): ResponseEntity<*> {
+ 
+         val community = communityRepo.findCommunityByIdAndAdminIdsContains(requestId.communityId, adminUserId)
+             ?: return ResponseEntity.badRequest()
+                 .body(ApiResponseMessage.USER_NOT_SUPER_ADMIN)
+ 
+         try {
+             val joinRequest = joinRequestRepo.findById(requestId).orElseThrow()!!
+ 
+             joinRequest.acceptedAt?.let {
+                 return ResponseEntity.accepted()
+                     .body(ApiResponseMessage.REQUEST_ALREADY_ACCEPTED)
+             }
+ 
+             // Admin rejects request
+             if (!accept) {
+                 joinRequestRepo.deleteById(requestId)
+                 //TODO:Notify referrer of rejection
+                 return ResponseEntity.ok().body("Request rejected")
+             }
+ 
+             // Update or create a Member
+             val memberPhone = joinRequest.id!!.phone
+             var member = memberRepo.findByPhone(memberPhone) ?: Member()
+ 
+             member.firstName = joinRequest.firstName
+             member.lastName = joinRequest.lastName
+             member.community = community
+             member.phone = memberPhone
+             member = memberRepo.save(member)
+ 
+             joinRequest.acceptedAt = Date()
+             joinRequest.approvedBy = adminUserId
+             joinRequestRepo.save(joinRequest)
+             //Delete all pending request with this phone number
+             joinRequestRepo.deleteAllByIdPhoneAndAcceptedAtIsNull(requestId.phone)
+ 
+             //TODO:Notify referrer of acceptance
+             return ResponseEntity.ok(member)
+         } catch (_: NoSuchElementException) {
+             return ResponseEntity.status(HttpStatus.NOT_FOUND)
+                 .body(ApiResponseMessage.REQUEST_CANT_BE_FOUND)
+         }
+     }
+ 
+ 
+     @Transactional
+     fun inviteUserToCommunity(request: JoinRequest, referrer: Member): ResponseEntity<*> {
+         try {
+ 
+             val community = referrer.community
+                 ?: return ResponseEntity.badRequest()
+                     .body("Referrer is not part of any community at this time")
+ 
+             val requestId = request.id!!
+             if (community.id != requestId.communityId) {
+                 return ResponseEntity.badRequest()
+                     .body("Referrer can only invite a user to their community")
+             }
+ 
+             if (community.superAdminId == null) {
+                 return ResponseEntity.badRequest().body("Community does not have an super admin")
+             }
+ 
+             if (memberRepo.existsByPhoneAndCommunityIsNotNull(requestId.phone)) {
+                 return ResponseEntity.badRequest().body("User is already part of a community")
+             }
+ 
+             if (joinRequestRepo.existsByIdAndAcceptedAtIsNull(requestId)) {
+                 return ResponseEntity.status(HttpStatus.ALREADY_REPORTED).body("User already has a pending request")
+             }
+ 
+             request.referrerId = referrer.id
+             joinRequestRepo.save(request)
+ 
+             //TODO: Notify community admin
+             return ResponseEntity.ok("Request to join community sent")
+ 
+         } catch (_: NoSuchElementException) {
+             return ResponseEntity.badRequest().body("Referrer not found")
+         }
+     }
+ 
+     fun addSecurityGuardToCommunity(guardData: GuardInfoBody, userId: String): ResponseEntity<*> {
+ 
+         val community = communityRepo.findByAdminIdsContains(userId)
+             ?: return ResponseEntity.badRequest().body("User not an Admin in a community")
+ 
+         val phone = guardData.phone
+         if (securityGuardRepo.existsByPhone(phone)) {
+             ResponseEntity.badRequest().body("$phone is already a Guard in a community")
+         }
+ 
+         val guard = SecurityGuard()
+         guard.phone = guardData.phone
+         guard.firstName = guardData.firstName
+         guard.lastName = guardData.lastName
+         guard.communityId = community.id
+         securityGuardRepo.save(guard)
+ 
+         return ResponseEntity.ok().body(guardData)
+     }
+ 
+     fun removeSecurityGuardFromCommunity(guardId: String, userId: String): ResponseEntity<*> {
+ 
+         val community = communityRepo.findByAdminIdsContains(userId)
+             ?: return ResponseEntity.badRequest().body("User not an Admin in a community")
+ 
+         try {
+             val guardCommunityId = securityGuardRepo.findById(guardId).orElseThrow()!!.communityId
+ 
+             if (guardCommunityId == community.id) {
+ 
+                 securityGuardRepo.deleteById(guardId)
+                 FirebaseAuth.getInstance().revokeRefreshTokens(guardId)
+                 return ResponseEntity.ok().body("Guard has been removed from community")
+             } else {
+                 return ResponseEntity.badRequest().body("User not an Admin in Guard's community")
+             }
+ 
+         } catch (e: NoSuchElementException) {
+             logger.warn(e.localizedMessage)
+             return ResponseEntity.badRequest().body(e.localizedMessage)
+         }
+ 
+ 
+     }
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-8/sources/source-3.html b/htmlReport/ns-8/sources/source-3.html new file mode 100644 index 0000000..f14bfcd --- /dev/null +++ b/htmlReport/ns-8/sources/source-3.html @@ -0,0 +1,246 @@ + + + + + + + + Coverage Report > LevyService + + + + + + +
+ + +

Coverage Summary for Class: LevyService (ng.cove.web.service)

+ + + + + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Branch, % + + Line, % +
LevyService + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
LevyService$WhenMappings
Total + + 66.7% + + + (6/9) + + + + 16.7% + + + (1/6) + + + + 53.3% + + + (24/45) + +
+ +
+
+ + +
+ package ng.cove.web.service
+ 
+ import com.mongodb.DuplicateKeyException
+ import ng.cove.web.data.model.*
+ import ng.cove.web.data.repo.AssignedLevyRepo
+ import ng.cove.web.data.repo.LevyPaymentRepo
+ import ng.cove.web.data.repo.LevyRepo
+ import ng.cove.web.data.repo.MemberRepo
+ import org.slf4j.LoggerFactory
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.http.ResponseEntity
+ import org.springframework.scheduling.annotation.Scheduled
+ import org.springframework.stereotype.Service
+ import org.springframework.transaction.annotation.Transactional
+ import org.springframework.web.context.WebApplicationContext
+ import java.util.*
+ 
+ @Service
+ class LevyService {
+ 
+     @Autowired
+     private lateinit var webApplicationContext: WebApplicationContext
+ 
+     @Autowired
+     lateinit var levyRepo: LevyRepo
+ 
+     @Autowired
+     lateinit var assignedLevyRepo: AssignedLevyRepo
+ 
+     @Autowired
+     lateinit var levyPaymentRepo: LevyPaymentRepo
+ 
+     @Autowired
+     lateinit var memberRepo: MemberRepo
+ 
+     private val logger = LoggerFactory.getLogger(this::class.simpleName)
+ 
+     fun createLevy(levy: Levy, admin: Member): ResponseEntity<*> {
+         levy.communityId = admin.community!!.id
+ 
+         return try {
+             ResponseEntity.ok(levyRepo.save(levy))
+         } catch (e: DuplicateKeyException) {
+             ResponseEntity.badRequest().body("Levy with this name already exists")
+         }
+     }
+ 
+     fun assignLevy(levyToAssign: AssignedLevy, admin: Member): ResponseEntity<*> {
+         try {
+             val communityLevy = levyRepo.findById(levyToAssign.levyId!!).orElseThrow()
+             if (communityLevy.communityId != admin.community!!.id) {
+                 return ResponseEntity.badRequest().body("You are not an admin of this community")
+             }
+ 
+             val member: Member = memberRepo.findById(levyToAssign.memberId!!).orElseThrow()
+             if (communityLevy.communityId != member.community!!.id) {
+                 return ResponseEntity.badRequest().body("Cannot assign levy from another community")
+             }
+         } catch (e: NoSuchElementException) {
+             return ResponseEntity.badRequest().body(mapOf("message" to "Levy or member not found"))
+         }
+ 
+         levyToAssign.assignedBy = admin.id
+         return try {
+             ResponseEntity.ok(assignedLevyRepo.save(levyToAssign))
+         } catch (e: Exception) {
+             ResponseEntity.badRequest().body(mapOf("message" to "levy already assigned to this member"))
+         }
+     }
+ 
+     /** Each payment creation runs in this transaction so that the failure of one
+     does not affect the creation of others from [createLevyPayments]*/
+     @Transactional
+     fun createPaymentForAssignedLevy(assignedLevy: AssignedLevy) {
+         logger.info("Creating payment for ${assignedLevy.memberId}")
+         val payment = LevyPayment()
+         payment.levyId = assignedLevy.levyId
+         payment.memberId = assignedLevy.memberId
+         payment.dueDate = assignedLevy.nextPaymentDue
+         levyPaymentRepo.save(payment)
+ 
+         //TODO: Notify member of payment
+ 
+         // Update next payment due of assigned levy
+         val levy = levyRepo.findById(assignedLevy.levyId!!).orElseThrow()
+         val cal = Calendar.getInstance()
+         logger.info("Levy ${levy.title} current due date:${assignedLevy.nextPaymentDue}")
+         cal.time = assignedLevy.nextPaymentDue
+         when (levy.type!!) {
+             LevyType.Month -> {
+                 cal.add(Calendar.MONTH, 1)
+             }
+ 
+             LevyType.Annual -> {
+                 cal.add(Calendar.YEAR, 1)
+             }
+         }
+         assignedLevy.nextPaymentDue = cal.time
+         assignedLevyRepo.save(assignedLevy)
+         logger.info("Levy ${levy.title} due date updated to: ${assignedLevy.nextPaymentDue}")
+     }
+ 
+     @Scheduled(fixedDelayString = "\${schedule.levy.duration.secs}")
+     fun createLevyPayments() {
+         val duePayments = assignedLevyRepo.findAllByNextPaymentDueIsBeforeOrderByNextPaymentDueAsc(Date())
+         duePayments.forEach {
+             // Use context to avoid Transaction self invocation
+             webApplicationContext.getBean(this::class.java).createPaymentForAssignedLevy(it)
+         }
+ 
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-8/sources/source-4.html b/htmlReport/ns-8/sources/source-4.html new file mode 100644 index 0000000..855babd --- /dev/null +++ b/htmlReport/ns-8/sources/source-4.html @@ -0,0 +1,140 @@ + + + + + + + + Coverage Report > NotificationService + + + + + + +
+ + +

Coverage Summary for Class: NotificationService (ng.cove.web.service)

+ + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
NotificationService + + 50% + + + (1/2) + + + + 11.1% + + + (1/9) + +
NotificationService$$SpringCGLIB$$0
Total + + 50% + + + (1/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + +
+ package ng.cove.web.service
+ 
+ import com.google.firebase.messaging.FirebaseMessaging
+ import com.google.firebase.messaging.Message
+ import com.google.firebase.messaging.Notification
+ import org.springframework.scheduling.annotation.Async
+ import org.springframework.stereotype.Service
+ 
+ @Service
+ class NotificationService {
+ 
+     @Async
+     fun notifyUserDeviceWhenPossible(title: String?, message: String?, token: String, data: Map<String, String>) {
+         //Check if user has granted notification permission on this device
+ 
+         val notification = Notification.builder()
+             .setTitle(title)
+             .setBody(message).build()
+         val fcmMessage = Message.builder()
+             .putAllData(data)
+             .setToken(token)
+             .setNotification(notification).build()
+ 
+         FirebaseMessaging.getInstance().send(fcmMessage)
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-8/sources/source-5.html b/htmlReport/ns-8/sources/source-5.html new file mode 100644 index 0000000..3584cd4 --- /dev/null +++ b/htmlReport/ns-8/sources/source-5.html @@ -0,0 +1,287 @@ + + + + + + + + Coverage Report > UserService + + + + + + +
+ + +

Coverage Summary for Class: UserService (ng.cove.web.service)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
UserService + + 100% + + + (1/1) + + + + 90.9% + + + (10/11) + + + + 52.6% + + + (20/38) + + + + 71.8% + + + (51/71) + +
+ +
+
+ + +
+ package ng.cove.web.service
+ 
+ import com.google.firebase.auth.FirebaseAuth
+ import ng.cove.web.component.SmsOtpService
+ import ng.cove.web.data.model.PhoneOtp
+ import ng.cove.web.data.model.UserType
+ import ng.cove.web.data.repo.MemberPhoneOtpRepo
+ import ng.cove.web.data.repo.MemberRepo
+ import ng.cove.web.data.repo.SecurityGuardRepo
+ import ng.cove.web.http.body.LoginBody
+ import ng.cove.web.http.body.OtpRefBody
+ import ng.cove.web.util.CacheNames
+ import org.slf4j.Logger
+ import org.slf4j.LoggerFactory
+ import org.springframework.beans.factory.annotation.Autowired
+ import org.springframework.beans.factory.annotation.Value
+ import org.springframework.cache.caffeine.CaffeineCacheManager
+ import org.springframework.http.ResponseEntity
+ import org.springframework.stereotype.Service
+ import java.time.Duration
+ import java.time.Instant
+ import java.util.*
+ 
+ 
+ @Service
+ class UserService {
+ 
+     private val logger: Logger = LoggerFactory.getLogger(this::class.java)
+ 
+     @Autowired
+     lateinit var memberRepo: MemberRepo
+ 
+     @Autowired
+     lateinit var guardRepo: SecurityGuardRepo
+ 
+     @Autowired
+     lateinit var smsOtp: SmsOtpService
+ 
+     @Autowired
+     lateinit var cacheManager: CaffeineCacheManager
+ 
+     @Autowired
+     lateinit var otpRepo: MemberPhoneOtpRepo
+ 
+     @Value("\${otp.trial-limit}")
+     var maxDailyOtpTrial: Int = 0
+ 
+ 
+     fun getOtpForLogin(phone: String, userType: UserType): ResponseEntity<*> {
+         if (userType == UserType.Member) {
+             memberRepo.findByPhoneAndCommunityIsNotNull(phone)
+                 ?: return ResponseEntity.badRequest().body("$phone is not a member of a community")
+         } else {
+             guardRepo.findByPhoneAndCommunityIdIsNotNull(phone)
+                 ?: return ResponseEntity.badRequest().body("$phone is not guard of a community")
+         }
+ 
+         // Check if user is a tester
+         getTesterId(phone, userType)?.let {
+             val future = Date.from(Instant.now().plus(Duration.ofDays(1)))
+             return ResponseEntity.ok().body(OtpRefBody(it, phone, future, 100))
+         }
+ 
+         val aDayAgo = Date.from(Instant.now().minus(Duration.ofDays(1)))
+ 
+         var trialCount = otpRepo.countByPhoneAndCreatedAtIsAfter(phone, aDayAgo).toInt()
+         if (trialCount >= maxDailyOtpTrial) {
+             return ResponseEntity.badRequest().body("Trial exceeded try again later")
+         }
+ 
+         val otpResult = smsOtp.sendOtp(phone)
+         return if (otpResult != null) {
+             trialCount++
+             otpResult.dailyTrialLeft = maxDailyOtpTrial - trialCount
+             val phoneOtp = PhoneOtp()
+             phoneOtp.phone = phone
+             phoneOtp.ref = otpResult.ref
+             phoneOtp.type = userType
+             phoneOtp.expireAt = otpResult.expireAt
+             otpRepo.save(phoneOtp)
+             ResponseEntity.ok().body(otpResult)
+         } else {
+             ResponseEntity.internalServerError().body("OTP provider error")
+         }
+ 
+     }
+ 
+     /**
+      * Verify phone OTP for all user types
+      */
+     fun verifyPhoneOtp(login: LoginBody, userType: UserType): ResponseEntity<*> {
+ 
+         try {
+             // If user is tester return the phone number, otherwise verify OTP
+             val phone = verifyTesterOtp(login, userType) ?: smsOtp.verifyOtp(login.otp, login.ref)
+             ?: return ResponseEntity.badRequest().body("Invalid code")
+ 
+             val userId: String
+ 
+             if (userType == UserType.Member) {
+                 val member = memberRepo.findByPhone(phone)!!
+                 //TODO: Redesign this if possible: Its only going to run on first user login, but check forever
+                 if (member.phoneVerifiedAt == null) {
+                     member.phoneVerifiedAt = Date()
+                 }
+ 
+                 member.deviceName = login.deviceName
+                 member.lastLoginAt = Date()
+                 memberRepo.save(member)
+ 
+                 userId = member.id!!
+                 // Update cache
+                 cacheManager.getCache(CacheNames.MEMBERS)?.put(userId, member)
+ 
+             } else {
+                 val guard = guardRepo.findByPhone(phone)!!
+                 //TODO: Redesign this if possible: Its only going to run on first user login, but check forever
+                 if (guard.phoneVerifiedAt == null) {
+                     guard.phoneVerifiedAt = Date()
+                 }
+ 
+                 userId = guard.id!!
+ 
+                 // Update device info
+                 guard.deviceName = login.deviceName
+                 guard.lastLoginAt = Date()
+                 guardRepo.save(guard)
+                 // Update cache
+                 cacheManager.getCache(CacheNames.GUARDS)?.put(userId, guard)
+             }
+ 
+             val firebaseAuth = FirebaseAuth.getInstance()
+ 
+             try {
+                 //Revoke refresh token for old devices if any
+                 firebaseAuth.revokeRefreshTokens(userId)
+                 // Clear OTP limit for this user
+                 otpRepo.deleteAllByPhone(phone)
+             } catch (_: Exception) {
+             }
+ 
+             // Set Admin claim for JWT
+             val claims = mapOf("type" to userType.name)
+ 
+             val customToken = firebaseAuth.createCustomToken(userId, claims)
+ 
+             return ResponseEntity.ok().body(customToken)
+         } catch (e: Exception) {
+             logger.error(e.localizedMessage)
+             return ResponseEntity.internalServerError().body(e.localizedMessage)
+         }
+     }
+ 
+     fun getTesterId(phone: String, userType: UserType): String? {
+         return if (userType == UserType.Member) {
+             memberRepo.findFirstByTestOtpIsNotNullAndPhone(phone)?.id
+         } else {
+             guardRepo.findFirstByTestOtpIsNotNullAndPhone(phone)?.id
+         }
+     }
+ 
+     // Return null if user is not a tester
+     fun verifyTesterOtp(login: LoginBody, userType: UserType): String? {
+         return if (userType == UserType.Member) {
+             memberRepo.findByIdAndTestOtp(login.ref, login.otp)?.phone
+         } else {
+             guardRepo.findByIdAndTestOtp(login.ref, login.otp)?.phone
+         }
+     }
+ 
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-9/index.html b/htmlReport/ns-9/index.html new file mode 100644 index 0000000..6e08f41 --- /dev/null +++ b/htmlReport/ns-9/index.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_BLOCK.html b/htmlReport/ns-9/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..77f094d --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_BLOCK.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-9/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..7047992 --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_CLASS.html b/htmlReport/ns-9/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..e079e4a --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_CLASS.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-9/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..6bdf2ac --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_LINE.html b/htmlReport/ns-9/index_SORT_BY_LINE.html new file mode 100644 index 0000000..bd4d1a8 --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_LINE.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-9/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..892c201 --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_METHOD.html b/htmlReport/ns-9/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..1aabe6f --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_METHOD.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-9/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..28d9ccd --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-9/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..25c9fa3 --- /dev/null +++ b/htmlReport/ns-9/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,195 @@ + + + + + + Coverage Report > ng.cove.web.util + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.util

+ + + + + + + + + + + + + + + +
Package + Class, % + + Method, % + + Branch, % + + Line, % +
ng.cove.web.util + + 33.3% + + + (1/3) + + + + 20% + + + (1/5) + + + + 0% + + + (0/2) + + + + 11.1% + + + (1/9) + +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + +
+Class + Class, % + + Method, % + + Branch, % + + Line, % +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
CacheNames + + 0% + + + (0/2) + + + + 0% + + + (0/2) + + + + + 0% + + + (0/2) + +
+ +
+ + + + + + diff --git a/htmlReport/ns-9/sources/source-1.html b/htmlReport/ns-9/sources/source-1.html new file mode 100644 index 0000000..6ee98d6 --- /dev/null +++ b/htmlReport/ns-9/sources/source-1.html @@ -0,0 +1,138 @@ + + + + + + + + Coverage Report > CacheNames + + + + + + +
+ + +

Coverage Summary for Class: CacheNames (ng.cove.web.util)

+ + + + + + + + + + + + + + + + + + + + + + + +
Class + Method, % + + Line, % +
CacheNames + + 0% + + + (0/1) + + + + 0% + + + (0/1) + +
CacheNames$Companion + + 0% + + + (0/1) + + + + 0% + + + (0/1) + +
Total + + 0% + + + (0/2) + + + + 0% + + + (0/2) + +
+ +
+
+ + +
+ package ng.cove.web.util
+ 
+ class CacheNames{
+     companion object {
+         const val MEMBERS = "members"
+         const val GUARDS  = "guards"
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-9/sources/source-2.html b/htmlReport/ns-9/sources/source-2.html new file mode 100644 index 0000000..20a44a5 --- /dev/null +++ b/htmlReport/ns-9/sources/source-2.html @@ -0,0 +1,137 @@ + + + + + + + + Coverage Report > CodeGenerator + + + + + + +
+ + +

Coverage Summary for Class: CodeGenerator (ng.cove.web.util)

+ + + + + + + + + + + + + + + + + +
Class + Class, % + + Method, % + + Branch, % + + Line, % +
CodeGenerator + + 100% + + + (1/1) + + + + 33.3% + + + (1/3) + + + + 0% + + + (0/2) + + + + 14.3% + + + (1/7) + +
+ +
+
+ + +
+ package ng.cove.web.util
+ 
+ import org.springframework.beans.factory.annotation.Value
+ import org.springframework.stereotype.Component
+ import java.security.SecureRandom
+ 
+ @Component
+ class CodeGenerator {
+ 
+     @Value("\${visitor.access-code.length}")
+     val accessCodeLength: Int? = null
+ 
+     fun getCode(): String {
+         val sb = StringBuilder()
+         val random = SecureRandom()
+         for (i in 0 until accessCodeLength!!) {
+             sb.append(random.nextInt(10)) // Append a random digit (0-9)
+         }
+         return sb.toString()
+     }
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-9/sources/source-3.html b/htmlReport/ns-9/sources/source-3.html new file mode 100644 index 0000000..3c79d92 --- /dev/null +++ b/htmlReport/ns-9/sources/source-3.html @@ -0,0 +1,78 @@ + + + + + + + + Coverage Report > ApiResponseMessage + + + + + + +
+ + +

Coverage Summary for Class: ApiResponseMessage (ng.cove.web.util)

+ + + + + + + + + +
Class
ApiResponseMessage
+ +
+
+ + +
+ package ng.cove.web.util
+ 
+ object ApiResponseMessage {
+     const val ACCESS_CODE_INVALID= "Access code is invalid"
+     const val MEMBER_IN_EXISTING_COMMUNITY = "User already part of a community"
+     const val REQUEST_CANT_BE_FOUND: String = "Request does not exist"
+     const val REQUEST_ALREADY_ACCEPTED: String = "Request has already been accepted"
+     const val USER_NOT_SUPER_ADMIN: String = "This user is not a super admin of this community"
+     const val PHOTO_IS_REQUIRED: String = "Upload profile photo before joining a community"
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/index.html b/htmlReport/ns-a/index.html new file mode 100644 index 0000000..f8427d0 --- /dev/null +++ b/htmlReport/ns-a/index.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_BLOCK.html b/htmlReport/ns-a/index_SORT_BY_BLOCK.html new file mode 100644 index 0000000..526928d --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_BLOCK.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_BLOCK_DESC.html b/htmlReport/ns-a/index_SORT_BY_BLOCK_DESC.html new file mode 100644 index 0000000..60035bf --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_BLOCK_DESC.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_CLASS.html b/htmlReport/ns-a/index_SORT_BY_CLASS.html new file mode 100644 index 0000000..a22ff7b --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_CLASS.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_CLASS_DESC.html b/htmlReport/ns-a/index_SORT_BY_CLASS_DESC.html new file mode 100644 index 0000000..f6f75fb --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_CLASS_DESC.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_LINE.html b/htmlReport/ns-a/index_SORT_BY_LINE.html new file mode 100644 index 0000000..c17044a --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_LINE.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_LINE_DESC.html b/htmlReport/ns-a/index_SORT_BY_LINE_DESC.html new file mode 100644 index 0000000..bf55aaa --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_LINE_DESC.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_METHOD.html b/htmlReport/ns-a/index_SORT_BY_METHOD.html new file mode 100644 index 0000000..ef3a886 --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_METHOD.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_METHOD_DESC.html b/htmlReport/ns-a/index_SORT_BY_METHOD_DESC.html new file mode 100644 index 0000000..05b0b17 --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_METHOD_DESC.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/index_SORT_BY_NAME_DESC.html b/htmlReport/ns-a/index_SORT_BY_NAME_DESC.html new file mode 100644 index 0000000..c1f08e5 --- /dev/null +++ b/htmlReport/ns-a/index_SORT_BY_NAME_DESC.html @@ -0,0 +1,68 @@ + + + + + + Coverage Report > ng.cove.web.data.repo + + + + + + +
+ + + +

Coverage Summary for Package: ng.cove.web.data.repo

+ + + + + + + +
Package
ng.cove.web.data.repo
+ +
+
+ + + + + +
+Class
+ +
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-1.html b/htmlReport/ns-a/sources/source-1.html new file mode 100644 index 0000000..2e6cf96 --- /dev/null +++ b/htmlReport/ns-a/sources/source-1.html @@ -0,0 +1,79 @@ + + + + + + + + Coverage Report > MemberPhoneOtpRepo + + + + + + +
+ + +

Coverage Summary for Class: MemberPhoneOtpRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
MemberPhoneOtpRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.PhoneOtp
+ import org.springframework.data.mongodb.repository.MongoRepository
+ import java.util.Date
+ 
+ interface MemberPhoneOtpRepo: MongoRepository<PhoneOtp, String> {
+     fun countByPhoneAndCreatedAtIsAfter(phone: String, createdAt: Date): Long
+     fun deleteAllByPhone(phone: String)
+     fun countByPhone(phone: String): Long
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-2.html b/htmlReport/ns-a/sources/source-2.html new file mode 100644 index 0000000..11484cb --- /dev/null +++ b/htmlReport/ns-a/sources/source-2.html @@ -0,0 +1,77 @@ + + + + + + + + Coverage Report > AssignedLevyRepo + + + + + + +
+ + +

Coverage Summary for Class: AssignedLevyRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
AssignedLevyRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.AssignedLevy
+ import org.springframework.data.mongodb.repository.MongoRepository
+ import java.util.*
+ 
+ interface AssignedLevyRepo : MongoRepository<AssignedLevy?,String?>{
+     fun findAllByNextPaymentDueIsBeforeOrderByNextPaymentDueAsc(today: Date): List<AssignedLevy>
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-3.html b/htmlReport/ns-a/sources/source-3.html new file mode 100644 index 0000000..18bfe4d --- /dev/null +++ b/htmlReport/ns-a/sources/source-3.html @@ -0,0 +1,74 @@ + + + + + + + + Coverage Report > LevyPaymentRepo + + + + + + +
+ + +

Coverage Summary for Class: LevyPaymentRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
LevyPaymentRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.LevyPayment
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface LevyPaymentRepo : MongoRepository<LevyPayment, String>
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-4.html b/htmlReport/ns-a/sources/source-4.html new file mode 100644 index 0000000..d45c376 --- /dev/null +++ b/htmlReport/ns-a/sources/source-4.html @@ -0,0 +1,74 @@ + + + + + + + + Coverage Report > LevyRepo + + + + + + +
+ + +

Coverage Summary for Class: LevyRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
LevyRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.Levy
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface LevyRepo : MongoRepository<Levy, String>
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-5.html b/htmlReport/ns-a/sources/source-5.html new file mode 100644 index 0000000..a3220a1 --- /dev/null +++ b/htmlReport/ns-a/sources/source-5.html @@ -0,0 +1,78 @@ + + + + + + + + Coverage Report > AccessRepo + + + + + + +
+ + +

Coverage Summary for Class: AccessRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
AccessRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.Access
+ import ng.cove.web.data.model.AccessId
+ import org.springframework.data.mongodb.repository.MongoRepository
+ import java.util.*
+ 
+ interface AccessRepo: MongoRepository<Access, AccessId>{
+     fun findByIdAndValidUntilIsAfter(id: AccessId, validUntil: Date): Access?
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-6.html b/htmlReport/ns-a/sources/source-6.html new file mode 100644 index 0000000..6fe53c0 --- /dev/null +++ b/htmlReport/ns-a/sources/source-6.html @@ -0,0 +1,77 @@ + + + + + + + + Coverage Report > CommunityRepo + + + + + + +
+ + +

Coverage Summary for Class: CommunityRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
CommunityRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.Community
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface CommunityRepo : MongoRepository<Community?, String?> {
+     fun findByAdminIdsContains(adminId: String): Community?
+     fun findCommunityByIdAndAdminIdsContains(id: String, adminId: String): Community?
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-7.html b/htmlReport/ns-a/sources/source-7.html new file mode 100644 index 0000000..5117d60 --- /dev/null +++ b/htmlReport/ns-a/sources/source-7.html @@ -0,0 +1,84 @@ + + + + + + + + Coverage Report > MemberRepo + + + + + + +
+ + +

Coverage Summary for Class: MemberRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
MemberRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.Member
+ import org.springframework.cache.annotation.Cacheable
+ import org.springframework.data.domain.Page
+ import org.springframework.data.domain.Pageable
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface MemberRepo : MongoRepository<Member, String?> {
+ 
+     fun findByPhoneAndCommunityIsNotNull(phone: String): Member?
+     fun findByPhone(phone: String): Member?
+     fun existsByPhoneAndCommunityIsNotNull(phone: String): Boolean
+     fun findFirstByTestOtpIsNotNullAndPhone(phone: String): Member?
+     fun findByIdAndTestOtp(id: String, testOtp: String): Member?
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-8.html b/htmlReport/ns-a/sources/source-8.html new file mode 100644 index 0000000..ef95d77 --- /dev/null +++ b/htmlReport/ns-a/sources/source-8.html @@ -0,0 +1,81 @@ + + + + + + + + Coverage Report > SecurityGuardRepo + + + + + + +
+ + +

Coverage Summary for Class: SecurityGuardRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
SecurityGuardRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.Member
+ import ng.cove.web.data.model.SecurityGuard
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface SecurityGuardRepo : MongoRepository<SecurityGuard?, String?>{
+     fun findByPhoneAndCommunityIdIsNotNull(phone: String): SecurityGuard?
+     fun existsByPhone(phone: String): Boolean
+     fun findByPhone(phone: String): SecurityGuard?
+     fun findFirstByTestOtpIsNotNullAndPhone(phone: String): SecurityGuard?
+     fun findByIdAndTestOtp(id: String, testOtp: String): SecurityGuard?
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-9.html b/htmlReport/ns-a/sources/source-9.html new file mode 100644 index 0000000..80e4691 --- /dev/null +++ b/htmlReport/ns-a/sources/source-9.html @@ -0,0 +1,80 @@ + + + + + + + + Coverage Report > OccupantRepo + + + + + + +
+ + +

Coverage Summary for Class: OccupantRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
OccupantRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.Occupant
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface OccupantRepo : MongoRepository<Occupant?, String?> {
+     fun findByGuardianCommunityIdAndFirstNameOrLastNameContainingIgnoreCase(
+         id: String?,
+         s1: String?,
+         s2: String?
+     ): Occupant?
+ }
+
+
+
+ + + + + + diff --git a/htmlReport/ns-a/sources/source-a.html b/htmlReport/ns-a/sources/source-a.html new file mode 100644 index 0000000..c9b7215 --- /dev/null +++ b/htmlReport/ns-a/sources/source-a.html @@ -0,0 +1,79 @@ + + + + + + + + Coverage Report > JoinRequestRepo + + + + + + +
+ + +

Coverage Summary for Class: JoinRequestRepo (ng.cove.web.data.repo)

+ + + + + + + + + +
Class
JoinRequestRepo
+ +
+
+ + +
+ package ng.cove.web.data.repo
+ 
+ import ng.cove.web.data.model.JoinRequest
+ import ng.cove.web.data.model.JoinRequestId
+ import org.springframework.data.mongodb.repository.MongoRepository
+ 
+ interface JoinRequestRepo : MongoRepository<JoinRequest, JoinRequestId?> {
+     fun existsByIdAndAcceptedAtIsNull(id: JoinRequestId): Boolean
+     fun findByIdPhone(phone: String): List<JoinRequest>?
+     fun deleteAllByIdPhoneAndAcceptedAtIsNull(phone: String)
+ }
+
+
+
+ + + + + + diff --git a/pom.xml b/pom.xml index e8ef037..4b0c506 100644 --- a/pom.xml +++ b/pom.xml @@ -132,12 +132,6 @@ de.flapdoodle.embed de.flapdoodle.embed.mongo.spring30x 4.11.0 - - - - org.junit.jupiter - junit-jupiter - 5.10.2 test @@ -152,12 +146,51 @@ 3.1.8 + + org.junit.platform + junit-platform-suite + 1.11.0-M2 + test + + src/main/kotlin src/test/kotlin + + org.jacoco + jacoco-maven-plugin + 0.8.12 + + + + prepare-agent + + + + report + prepare-package + + report + + + + + + + org.apache.maven.plugins + maven-surefire-plugin + 3.2.5 + + + org.apache.maven.surefire + surefire-junit-platform + 3.2.5 + + + com.google.cloud.tools diff --git a/src/main/kotlin/ng/cove/web/App.kt b/src/main/kotlin/ng/cove/web/App.kt index 6dc9e94..d3103d2 100644 --- a/src/main/kotlin/ng/cove/web/App.kt +++ b/src/main/kotlin/ng/cove/web/App.kt @@ -6,7 +6,6 @@ import com.google.cloud.secretmanager.v1.SecretManagerServiceClient import com.google.cloud.secretmanager.v1.SecretVersionName import com.google.firebase.FirebaseApp import com.google.firebase.FirebaseOptions -import de.flapdoodle.embed.mongo.spring.autoconfigure.EmbeddedMongoAutoConfiguration import ng.cove.web.component.SmsOtpService import org.springframework.boot.SpringApplication import org.springframework.boot.autoconfigure.SpringBootApplication @@ -16,16 +15,20 @@ import org.springframework.cache.annotation.EnableCaching import org.springframework.cache.caffeine.CaffeineCacheManager import org.springframework.context.ApplicationListener import org.springframework.context.annotation.Bean +import org.springframework.data.mongodb.repository.config.EnableMongoRepositories import org.springframework.scheduling.annotation.EnableAsync -import org.springframework.scheduling.annotation.EnableScheduling +import org.springframework.web.client.RestTemplate import java.util.concurrent.TimeUnit @EnableAsync @EnableCaching -@EnableScheduling -@SpringBootApplication(exclude = [MongoDataAutoConfiguration::class, EmbeddedMongoAutoConfiguration::class]) +@EnableMongoRepositories("ng.cove.web.data.repo") +@SpringBootApplication(exclude = [MongoDataAutoConfiguration::class]) class App { + @Bean + fun restTemplate() = RestTemplate() + @Bean fun caffeineConfig(): Caffeine { return Caffeine.newBuilder().expireAfterWrite(60, TimeUnit.MINUTES) diff --git a/src/main/kotlin/ng/cove/web/component/SmsOtpService.kt b/src/main/kotlin/ng/cove/web/component/SmsOtpService.kt index 94da7c5..72b745e 100644 --- a/src/main/kotlin/ng/cove/web/component/SmsOtpService.kt +++ b/src/main/kotlin/ng/cove/web/component/SmsOtpService.kt @@ -4,16 +4,28 @@ import com.google.gson.JsonObject import ng.cove.web.http.body.OtpRefBody import org.slf4j.Logger import org.slf4j.LoggerFactory +import org.springframework.beans.factory.annotation.Autowired import org.springframework.beans.factory.annotation.Value +import org.springframework.context.annotation.Bean +import org.springframework.http.HttpEntity +import org.springframework.http.HttpHeaders import org.springframework.http.HttpStatus import org.springframework.http.MediaType import org.springframework.stereotype.Service -import org.springframework.web.client.RestClient +import org.springframework.web.client.RestTemplate +import java.time.Instant import java.util.* +const val SEND_URL: String = "https://api.ng.termii.com/api/sms/otp/send" +const val VERIFY_URL = "https://api.ng.termii.com/api/sms/otp/verify" + @Service -class SmsOtpService { +class SmsOtpService( + @Autowired + val template: RestTemplate +) { + val logger: Logger = LoggerFactory.getLogger(this::class.java) // Set in ApplicationStartup @@ -25,36 +37,36 @@ class SmsOtpService { fun sendOtp(phone: String): OtpRefBody? { - val requestBody = JsonObject() - requestBody.addProperty("api_key", termiiApiKey) - requestBody.addProperty("from", "N-Alert") - requestBody.addProperty("to", phone) - requestBody.addProperty("message_type", "NUMERIC") - requestBody.addProperty("channel", "dnd") - requestBody.addProperty("pin_attempts", 5) - requestBody.addProperty("pin_time_to_live", otpExpiryMins) - requestBody.addProperty("pin_length", 6) - requestBody.addProperty("pin_placeholder", "") - requestBody.addProperty("message_text", "Your login OTP is: ") - - val url = "https://api.ng.termii.com/api/sms/otp/send" - val client = RestClient.builder().baseUrl(url).defaultHeaders { h -> - h.contentType = MediaType.APPLICATION_JSON - }.build() + val requestBody = JsonObject().apply { + addProperty("api_key", termiiApiKey) + addProperty("from", "N-Alert") + addProperty("to", phone) + addProperty("message_type", "NUMERIC") + addProperty("channel", "dnd") + addProperty("pin_attempts", 5) + addProperty("pin_time_to_live", otpExpiryMins) + addProperty("pin_length", 6) + addProperty("pin_placeholder", "") + addProperty("message_text", "Your login OTP is: ") + } + val headers: HttpHeaders = HttpHeaders().apply { + contentType = MediaType.APPLICATION_JSON + } + val entity = HttpEntity(requestBody.toString(), headers) try { - val result = client.post() - .body(requestBody.toString()) - .retrieve().toEntity(Map::class.java).body!! + val response = template + .postForEntity(SEND_URL, entity, Map::class.java) + val result = response.body!! // Log error when Termii returns any other status result["status"].takeIf { it != 200 }?.let { throw Exception("SMS OTP provide error code: $it") } val ref = result["pinId"] as String - val futureDateTime = Date().toInstant() + val futureDateTime = Instant.now() .plusSeconds(otpExpiryMins.toLong() * 60) val expiry = Date.from(futureDateTime) @@ -67,20 +79,19 @@ class SmsOtpService { } fun verifyOtp(otp: String, ref: String): String? { - val requestBody = JsonObject() - requestBody.addProperty("api_key", termiiApiKey) - requestBody.addProperty("pin_id", ref) - requestBody.addProperty("pin", otp) - - val url = "https://api.ng.termii.com/api/sms/otp/verify" - val client = RestClient.builder().baseUrl(url).defaultHeaders { h -> - run { - h.contentType = MediaType.APPLICATION_JSON - } - }.build() + val requestBody = JsonObject().apply { + addProperty("api_key", termiiApiKey) + addProperty("pin_id", ref) + addProperty("pin", otp) + } + + val headers: HttpHeaders = HttpHeaders().apply { + contentType = MediaType.APPLICATION_JSON + } + val entity = HttpEntity(requestBody.toString(), headers) try { - val result = client.post().body(requestBody.toString()) - .retrieve().toEntity(Map::class.java) + val result = template + .postForEntity(VERIFY_URL, entity, Map::class.java) if (result.statusCode != HttpStatus.OK) return null diff --git a/src/main/kotlin/ng/cove/web/config/MongoConfig.kt b/src/main/kotlin/ng/cove/web/config/MongoConfig.kt index 2ba93d8..06eb76a 100644 --- a/src/main/kotlin/ng/cove/web/config/MongoConfig.kt +++ b/src/main/kotlin/ng/cove/web/config/MongoConfig.kt @@ -4,32 +4,16 @@ import com.google.cloud.secretmanager.v1.SecretManagerServiceClient import com.google.cloud.secretmanager.v1.SecretVersionName import com.mongodb.client.MongoClient import com.mongodb.client.MongoClients -import de.flapdoodle.embed.mongo.commands.ServerAddress -import de.flapdoodle.embed.mongo.distribution.Version -import de.flapdoodle.embed.mongo.transitions.Mongod -import de.flapdoodle.embed.mongo.transitions.RunningMongodProcess -import de.flapdoodle.reverse.TransitionWalker import org.springframework.context.annotation.Configuration +import org.springframework.context.annotation.Profile import org.springframework.data.mongodb.config.AbstractMongoClientConfiguration -import org.springframework.data.mongodb.repository.config.EnableMongoRepositories import org.springframework.web.context.WebApplicationContext -import javax.annotation.PreDestroy -//@Profile("!test") +@Profile("!test") @Configuration -@EnableMongoRepositories("ng.cove.web.data.repo") class MongoConfig(val context: WebApplicationContext) : AbstractMongoClientConfiguration() { - var embeddedMongo: TransitionWalker.ReachedState? = null - - override fun getDatabaseName(): String { - val profiles = context.environment.activeProfiles - return if (profiles.getOrNull(0) == "test"){ - "test" - }else { - "dev" - } - } + override fun getDatabaseName(): String = "dev" override fun mongoClient(): MongoClient { val profiles = context.environment.activeProfiles @@ -50,24 +34,10 @@ class MongoConfig(val context: WebApplicationContext) : AbstractMongoClientConfi return MongoClients.create(String(dbSecretPayload.readAllBytes())) } } - else -> { - // Embedded Mongo instance for testing - embeddedMongo = Mongod.instance().start(Version.Main.V7_0) - val serverAddress: ServerAddress = embeddedMongo!!.current().serverAddress - val host = serverAddress.host - val port = serverAddress.port - return MongoClients.create("mongodb://$host:$port/test") - } + else -> throw Exception("This profile has no db configuration") } } - - @PreDestroy - fun onShutDown() { - embeddedMongo?.close() - } - override fun autoIndexCreation(): Boolean = true - } \ No newline at end of file diff --git a/src/main/kotlin/ng/cove/web/config/ScheduleConfig.kt b/src/main/kotlin/ng/cove/web/config/ScheduleConfig.kt new file mode 100644 index 0000000..6ddaf26 --- /dev/null +++ b/src/main/kotlin/ng/cove/web/config/ScheduleConfig.kt @@ -0,0 +1,8 @@ +package ng.cove.web.config + +import org.springframework.context.annotation.Configuration +import org.springframework.scheduling.annotation.EnableScheduling + +@EnableScheduling +@Configuration +class ScheduleConfig \ No newline at end of file diff --git a/src/main/kotlin/ng/cove/web/data/model/Member.kt b/src/main/kotlin/ng/cove/web/data/model/Member.kt index e3fc109..54ee2b7 100644 --- a/src/main/kotlin/ng/cove/web/data/model/Member.kt +++ b/src/main/kotlin/ng/cove/web/data/model/Member.kt @@ -57,11 +57,6 @@ open class Member { @Field("device_name") var deviceName: String? = null - @JsonIgnore - @Indexed(unique = true, sparse = true) - @Field("device_id") - var deviceId: String? = null - @JsonIgnore @Field("fcm_token") var fcmToken: String? = null diff --git a/src/main/kotlin/ng/cove/web/data/model/SecurityGuard.kt b/src/main/kotlin/ng/cove/web/data/model/SecurityGuard.kt index 41ebbc4..be83b43 100644 --- a/src/main/kotlin/ng/cove/web/data/model/SecurityGuard.kt +++ b/src/main/kotlin/ng/cove/web/data/model/SecurityGuard.kt @@ -45,11 +45,6 @@ class SecurityGuard { @Field("device_name") var deviceName: String? = null - @JsonIgnore - @Indexed(unique = true, sparse = true) - @Field("device_id") - var deviceId: String? = null - @JsonIgnore @Field("fcm_token") var fcmToken: String? = null diff --git a/src/main/kotlin/ng/cove/web/data/repo/MemberRepo.kt b/src/main/kotlin/ng/cove/web/data/repo/MemberRepo.kt index 7e5662d..ad444dc 100644 --- a/src/main/kotlin/ng/cove/web/data/repo/MemberRepo.kt +++ b/src/main/kotlin/ng/cove/web/data/repo/MemberRepo.kt @@ -1,6 +1,7 @@ package ng.cove.web.data.repo import ng.cove.web.data.model.Member +import ng.cove.web.util.CacheNames import org.springframework.cache.annotation.Cacheable import org.springframework.data.domain.Page import org.springframework.data.domain.Pageable @@ -8,10 +9,11 @@ import org.springframework.data.mongodb.repository.MongoRepository interface MemberRepo : MongoRepository { + @Cacheable(value = [CacheNames.MEMBERS]) + fun findMemberById(id: String): Member? fun findByPhoneAndCommunityIsNotNull(phone: String): Member? fun findByPhone(phone: String): Member? fun existsByPhoneAndCommunityIsNotNull(phone: String): Boolean - fun findByFirstNameEquals(name: String, pageable: Pageable): Page fun findFirstByTestOtpIsNotNullAndPhone(phone: String): Member? fun findByIdAndTestOtp(id: String, testOtp: String): Member? } diff --git a/src/main/kotlin/ng/cove/web/data/repo/SecurityGuardRepo.kt b/src/main/kotlin/ng/cove/web/data/repo/SecurityGuardRepo.kt index c1212f3..073ac30 100644 --- a/src/main/kotlin/ng/cove/web/data/repo/SecurityGuardRepo.kt +++ b/src/main/kotlin/ng/cove/web/data/repo/SecurityGuardRepo.kt @@ -2,9 +2,13 @@ package ng.cove.web.data.repo import ng.cove.web.data.model.Member import ng.cove.web.data.model.SecurityGuard +import ng.cove.web.util.CacheNames +import org.springframework.cache.annotation.Cacheable import org.springframework.data.mongodb.repository.MongoRepository interface SecurityGuardRepo : MongoRepository{ + @Cacheable(value = [CacheNames.GUARDS]) + fun findSecurityGuardById(id: String): SecurityGuard? fun findByPhoneAndCommunityIdIsNotNull(phone: String): SecurityGuard? fun existsByPhone(phone: String): Boolean fun findByPhone(phone: String): SecurityGuard? diff --git a/src/main/kotlin/ng/cove/web/http/body/LoginBody.kt b/src/main/kotlin/ng/cove/web/http/body/LoginBody.kt index 76023b5..dfc4f86 100644 --- a/src/main/kotlin/ng/cove/web/http/body/LoginBody.kt +++ b/src/main/kotlin/ng/cove/web/http/body/LoginBody.kt @@ -8,6 +8,5 @@ import jakarta.validation.constraints.NotBlank class LoginBody { @field:NotBlank(message = "Otp cannot be blank") lateinit var otp: String @field:NotBlank(message = "ref cannot be blank") lateinit var ref: String - @field:NotBlank(message = "deviceId cannot be blank") lateinit var deviceId: String @field:NotBlank(message = "deviceName cannot be blank") lateinit var deviceName: String } \ No newline at end of file diff --git a/src/main/kotlin/ng/cove/web/http/controller/DefaultController.kt b/src/main/kotlin/ng/cove/web/http/controller/DefaultController.kt index 228cb72..f5f84db 100644 --- a/src/main/kotlin/ng/cove/web/http/controller/DefaultController.kt +++ b/src/main/kotlin/ng/cove/web/http/controller/DefaultController.kt @@ -8,6 +8,8 @@ import io.swagger.v3.oas.annotations.Operation import io.swagger.v3.oas.annotations.media.Content import io.swagger.v3.oas.annotations.media.Schema import io.swagger.v3.oas.annotations.responses.ApiResponse +import io.swagger.v3.oas.annotations.security.SecurityRequirement +import io.swagger.v3.oas.annotations.security.SecurityRequirements import jakarta.validation.Valid import org.springframework.beans.factory.annotation.Autowired import org.springframework.http.MediaType @@ -15,6 +17,7 @@ import org.springframework.http.ResponseEntity import org.springframework.web.bind.annotation.* @RestController +@SecurityRequirements class DefaultController: BaseController() { @Autowired diff --git a/src/main/kotlin/ng/cove/web/http/interceptor/SecureInterceptor.kt b/src/main/kotlin/ng/cove/web/http/interceptor/SecureInterceptor.kt index 36805d6..6ab3ee4 100644 --- a/src/main/kotlin/ng/cove/web/http/interceptor/SecureInterceptor.kt +++ b/src/main/kotlin/ng/cove/web/http/interceptor/SecureInterceptor.kt @@ -5,7 +5,8 @@ import com.google.firebase.auth.FirebaseToken import jakarta.servlet.http.HttpServletRequest import jakarta.servlet.http.HttpServletResponse import ng.cove.web.data.model.UserType -import ng.cove.web.service.CacheService +import ng.cove.web.data.repo.MemberRepo +import ng.cove.web.data.repo.SecurityGuardRepo import org.slf4j.LoggerFactory import org.springframework.http.HttpStatus import org.springframework.web.context.WebApplicationContext @@ -39,13 +40,14 @@ class SecureInterceptor(private val context: WebApplicationContext) : HandlerInt userId = firebaseToken.uid } - val cacheService = context.getBean(CacheService::class.java) /** Get User model from DB and attach to request**/ if (userType == UserType.Guard) { - val guard = cacheService.getGuardById(userId)!! + val repo = context.getBean(SecurityGuardRepo::class.java) + val guard = repo.findSecurityGuardById(userId)!! request.setAttribute("user", guard) } else { - val member = cacheService.getMemberById(userId)!! + val repo = context.getBean(MemberRepo::class.java) + val member = repo.findMemberById(userId)!! request.setAttribute("user", member) } return true diff --git a/src/main/kotlin/ng/cove/web/service/CacheService.kt b/src/main/kotlin/ng/cove/web/service/CacheService.kt deleted file mode 100644 index 633aa9e..0000000 --- a/src/main/kotlin/ng/cove/web/service/CacheService.kt +++ /dev/null @@ -1,28 +0,0 @@ -package ng.cove.web.service - -import ng.cove.web.data.model.Member -import ng.cove.web.data.model.SecurityGuard -import ng.cove.web.data.repo.MemberRepo -import ng.cove.web.data.repo.SecurityGuardRepo -import ng.cove.web.util.CacheNames -import org.springframework.beans.factory.annotation.Autowired -import org.springframework.cache.annotation.Cacheable -import org.springframework.stereotype.Service - -@Service -class CacheService { - @Autowired - lateinit var memberRepo : MemberRepo - @Autowired - lateinit var guardRepo : SecurityGuardRepo - - @Cacheable(value = [CacheNames.MEMBERS]) - fun getMemberById(id: String): Member? { - return memberRepo.findById(id).orElse(null) - } - - @Cacheable(value = [CacheNames.GUARDS]) - fun getGuardById(id: String): SecurityGuard? { - return guardRepo.findById(id).orElse(null) - } -} \ No newline at end of file diff --git a/src/main/kotlin/ng/cove/web/service/CommunityService.kt b/src/main/kotlin/ng/cove/web/service/CommunityService.kt index 62837b5..f06569f 100644 --- a/src/main/kotlin/ng/cove/web/service/CommunityService.kt +++ b/src/main/kotlin/ng/cove/web/service/CommunityService.kt @@ -12,12 +12,9 @@ import org.slf4j.LoggerFactory import org.springframework.beans.factory.annotation.Autowired import org.springframework.http.HttpStatus import org.springframework.http.ResponseEntity -import org.springframework.scheduling.annotation.Scheduled -import org.springframework.stereotype.Component import org.springframework.stereotype.Service import org.springframework.transaction.annotation.Transactional import java.util.* -import java.util.concurrent.TimeUnit @Service class CommunityService { @@ -242,14 +239,4 @@ class CommunityService { } - - @Scheduled(fixedRate = 12, timeUnit = TimeUnit.HOURS) - fun createLevyPayments() { - val duePayments = assignedLevyRepo.findAllByNextPaymentDueIsBeforeOrderByNextPaymentDueAsc(Date()) - duePayments.forEach { - levyService.createPaymentForAssignedLevy(it) - } - - } - } diff --git a/src/main/kotlin/ng/cove/web/service/LevyService.kt b/src/main/kotlin/ng/cove/web/service/LevyService.kt index 5a4555e..31eacd9 100644 --- a/src/main/kotlin/ng/cove/web/service/LevyService.kt +++ b/src/main/kotlin/ng/cove/web/service/LevyService.kt @@ -9,13 +9,18 @@ import ng.cove.web.data.repo.MemberRepo import org.slf4j.LoggerFactory import org.springframework.beans.factory.annotation.Autowired import org.springframework.http.ResponseEntity +import org.springframework.scheduling.annotation.Scheduled import org.springframework.stereotype.Service import org.springframework.transaction.annotation.Transactional +import org.springframework.web.context.WebApplicationContext import java.util.* @Service class LevyService { + @Autowired + private lateinit var webApplicationContext: WebApplicationContext + @Autowired lateinit var levyRepo: LevyRepo @@ -63,7 +68,8 @@ class LevyService { } } - + /** Each payment creation runs in this transaction so that the failure of one + does not affect the creation of others from [createLevyPayments]*/ @Transactional fun createPaymentForAssignedLevy(assignedLevy: AssignedLevy) { logger.info("Creating payment for ${assignedLevy.memberId}") @@ -93,4 +99,14 @@ class LevyService { assignedLevyRepo.save(assignedLevy) logger.info("Levy ${levy.title} due date updated to: ${assignedLevy.nextPaymentDue}") } + + @Scheduled(fixedDelayString = "\${schedule.levy.duration.secs}") + fun createLevyPayments() { + val duePayments = assignedLevyRepo.findAllByNextPaymentDueIsBeforeOrderByNextPaymentDueAsc(Date()) + duePayments.forEach { + // Use context to avoid Transaction self invocation + webApplicationContext.getBean(this::class.java).createPaymentForAssignedLevy(it) + } + + } } \ No newline at end of file diff --git a/src/main/kotlin/ng/cove/web/service/UserService.kt b/src/main/kotlin/ng/cove/web/service/UserService.kt index 5fec402..52345cd 100644 --- a/src/main/kotlin/ng/cove/web/service/UserService.kt +++ b/src/main/kotlin/ng/cove/web/service/UserService.kt @@ -92,7 +92,7 @@ class UserService { try { // If user is tester return the phone number, otherwise verify OTP - val phone = getTesterPhone(login.ref, login.otp, userType) ?: smsOtp.verifyOtp(login.otp, login.ref) + val phone = verifyTesterOtp(login, userType) ?: smsOtp.verifyOtp(login.otp, login.ref) ?: return ResponseEntity.badRequest().body("Invalid code") val userId: String @@ -104,14 +104,13 @@ class UserService { member.phoneVerifiedAt = Date() } - member.deviceId = login.deviceId member.deviceName = login.deviceName member.lastLoginAt = Date() memberRepo.save(member) userId = member.id!! // Update cache - cacheManager.getCache(CacheNames.MEMBERS)?.put(userId, member) + cacheManager.getCache(CacheNames.MEMBERS)?.evict(userId) } else { val guard = guardRepo.findByPhone(phone)!! @@ -123,12 +122,11 @@ class UserService { userId = guard.id!! // Update device info - guard.deviceId = login.deviceId guard.deviceName = login.deviceName guard.lastLoginAt = Date() guardRepo.save(guard) // Update cache - cacheManager.getCache(CacheNames.GUARDS)?.put(userId, guard) + cacheManager.getCache(CacheNames.GUARDS)?.evict(userId) } val firebaseAuth = FirebaseAuth.getInstance() @@ -161,11 +159,12 @@ class UserService { } } - fun getTesterPhone(id: String, otp: String, userType: UserType): String? { + // Return null if user is not a tester + fun verifyTesterOtp(login: LoginBody, userType: UserType): String? { return if (userType == UserType.Member) { - memberRepo.findByIdAndTestOtp(id, otp)?.phone + memberRepo.findByIdAndTestOtp(login.ref, login.otp)?.phone } else { - guardRepo.findByIdAndTestOtp(id, otp)?.phone + guardRepo.findByIdAndTestOtp(login.ref, login.otp)?.phone } } diff --git a/src/main/resources/application-dev.properties b/src/main/resources/application-dev.properties index cb35b1e..44b09b3 100644 --- a/src/main/resources/application-dev.properties +++ b/src/main/resources/application-dev.properties @@ -1,5 +1,4 @@ # API Request timeout spring.mvc.async.request-timeout=30000 # swagger-ui config -springdoc.swagger-ui.path=/swagger-ui.html springdoc.api-docs.enabled=true \ No newline at end of file diff --git a/src/main/resources/application.properties b/src/main/resources/application.properties index ce4cb49..48fa461 100644 --- a/src/main/resources/application.properties +++ b/src/main/resources/application.properties @@ -10,4 +10,6 @@ visitor.access-code.length=8 #This resolves CORS issues on Swagger UI server.forward-headers-strategy=framework -#de.flapdoodle.mongodb.embedded.version=6.0.5 \ No newline at end of file +schedule.levy.duration.secs = 43200 + +springdoc.swagger-ui.path=/swagger-ui.html \ No newline at end of file diff --git a/src/test/kotlin/ng/cove/web/AppTest.kt b/src/test/kotlin/ng/cove/web/AppTest.kt index ff3ae16..cb71eb6 100644 --- a/src/test/kotlin/ng/cove/web/AppTest.kt +++ b/src/test/kotlin/ng/cove/web/AppTest.kt @@ -3,8 +3,15 @@ package ng.cove.web import com.fasterxml.jackson.databind.ObjectMapper import com.fasterxml.jackson.databind.PropertyNamingStrategies import com.google.firebase.auth.FirebaseAuth +import com.mongodb.client.MongoClient +import com.mongodb.client.MongoClients +import de.flapdoodle.embed.mongo.commands.ServerAddress +import de.flapdoodle.embed.mongo.distribution.Version +import de.flapdoodle.embed.mongo.spring.autoconfigure.EmbeddedMongoAutoConfiguration +import de.flapdoodle.embed.mongo.transitions.Mongod +import de.flapdoodle.embed.mongo.transitions.RunningMongodProcess +import de.flapdoodle.reverse.TransitionWalker import net.datafaker.Faker -import ng.cove.web.component.SmsOtpService import ng.cove.web.data.model.Community import ng.cove.web.data.model.Member import ng.cove.web.data.repo.CommunityRepo @@ -19,21 +26,25 @@ import org.mockito.MockedStatic import org.mockito.Mockito import org.mockito.Mockito.mockStatic import org.springframework.beans.factory.annotation.Autowired -import org.springframework.beans.factory.annotation.Value +import org.springframework.boot.autoconfigure.EnableAutoConfiguration import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc import org.springframework.boot.test.context.SpringBootTest -import org.springframework.boot.test.mock.mockito.MockBean -import org.springframework.data.mongodb.core.MongoTemplate +import org.springframework.boot.test.context.TestConfiguration +import org.springframework.context.annotation.Import +import org.springframework.data.mongodb.config.AbstractMongoClientConfiguration import org.springframework.test.context.ActiveProfiles import org.springframework.test.context.TestPropertySource import org.springframework.test.web.servlet.MockMvc +import javax.annotation.PreDestroy @SpringBootTest(classes = [App::class]) @ActiveProfiles("test") @AutoConfigureMockMvc @TestInstance(TestInstance.Lifecycle.PER_CLASS) -@TestPropertySource("/application–test.properties") +@TestPropertySource("/application.properties") +@Import(value = [EmbeddedMongoConfig::class]) +@EnableAutoConfiguration(exclude = [EmbeddedMongoAutoConfiguration::class]) class AppTest { @Autowired @@ -48,46 +59,38 @@ class AppTest { @Autowired lateinit var memberPhoneOtpRepo: MemberPhoneOtpRepo - @MockBean - lateinit var smsOtpService: SmsOtpService - @Autowired lateinit var mockMvc: MockMvc - lateinit var staticFirebaseAuth: MockedStatic + lateinit var authMockedStatic: MockedStatic // Mocked FirebaseAuth for testing val auth: FirebaseAuth = Mockito.mock(FirebaseAuth::class.java) - @Value("\${otp.trial-limit}") - var maxDailyOtpTrial: Int = 0 - final val faker = Faker() - val mapper = ObjectMapper().apply { propertyNamingStrategy = PropertyNamingStrategies.SNAKE_CASE } + val mapper = ObjectMapper() + .apply { propertyNamingStrategy = PropertyNamingStrategies.SNAKE_CASE } lateinit var member: Member lateinit var community: Community - @Autowired - lateinit var mongoTemplate: MongoTemplate - @BeforeEach fun setUp() { + member = Member().apply { + id = faker.random().hex(20) + firstName = faker.name().firstName() + lastName = faker.name().lastName() + phone = faker.phoneNumber().cellPhone() + } - community = Community() - community.id = faker.random().hex(20) - community.name = "${faker.address().state()} Community" - community.address = faker.address().streetAddress() - - member = Member() - member.id = faker.random().hex(20) - member.firstName = faker.name().firstName() - member.lastName = faker.name().lastName() - member.phone = faker.phoneNumber().cellPhone() - - community.superAdminId = member.id - community.adminIds = setOf(member.id!!) + community = Community().apply { + id = faker.random().hex(20) + name = "${faker.address().state()} Community" + address = faker.address().streetAddress() + superAdminId = member.id + adminIds = setOf(member.id!!) + } member.community = community } @@ -95,19 +98,35 @@ class AppTest { @BeforeAll fun setupAll() { // Mock FirebaseAuth - staticFirebaseAuth = mockStatic(FirebaseAuth::class.java) - staticFirebaseAuth.`when`(FirebaseAuth::getInstance).thenReturn(auth) - - //TODO: Investigate why 'phone_otp' collection index is always duplicate in Embedded db - mongoTemplate.collectionNames.forEach { - mongoTemplate.getCollection(it).dropIndexes() - mongoTemplate.getCollection(it).drop() - } + authMockedStatic = mockStatic(FirebaseAuth::class.java) + authMockedStatic.`when`(FirebaseAuth::getInstance).thenReturn(auth) } @AfterAll fun tearDownAll() { - staticFirebaseAuth.close() + authMockedStatic.close() } } + +@TestConfiguration +class EmbeddedMongoConfig : AbstractMongoClientConfiguration() { + + var embeddedMongo: TransitionWalker.ReachedState? = null + + override fun getDatabaseName(): String = "test" + + override fun mongoClient(): MongoClient { + // Embedded Mongo instance for testing + embeddedMongo = Mongod.instance()./**/start(Version.Main.V7_0) + val serverAddress: ServerAddress = embeddedMongo!!.current().serverAddress + val host = serverAddress.host + val port = serverAddress.port + return MongoClients.create("mongodb://$host:$port") + } + + @PreDestroy + fun onShutDown() { + embeddedMongo?.close() + } +} diff --git a/src/test/kotlin/ng/cove/web/IntegrationTest.kt b/src/test/kotlin/ng/cove/web/IntegrationTest.kt new file mode 100644 index 0000000..56ded72 --- /dev/null +++ b/src/test/kotlin/ng/cove/web/IntegrationTest.kt @@ -0,0 +1,11 @@ +//package ng.cove.web +// +// +//import org.junit.platform.suite.api.SelectPackages +//import org.junit.platform.suite.api.Suite +//import org.junit.platform.suite.api.SuiteDisplayName +// +//@Suite +//@SuiteDisplayName("Integration suite") +//@SelectPackages(value = ["ng.cove.web.http.controller"]) +//class IntegrationTest \ No newline at end of file diff --git a/src/test/kotlin/ng/cove/web/components/SmsOtpServiceTest.kt b/src/test/kotlin/ng/cove/web/components/SmsOtpServiceTest.kt new file mode 100644 index 0000000..f47156a --- /dev/null +++ b/src/test/kotlin/ng/cove/web/components/SmsOtpServiceTest.kt @@ -0,0 +1,70 @@ +package ng.cove.web.components + +import ng.cove.web.component.SmsOtpService +import org.junit.jupiter.api.BeforeEach +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.extension.ExtendWith +import org.mockito.Mock +import org.mockito.Mockito.verify +import org.mockito.Mockito.`when` +import org.mockito.kotlin.any +import org.mockito.kotlin.times +import org.mockito.kotlin.verifyNoMoreInteractions +import org.springframework.http.ResponseEntity +import org.springframework.test.context.junit.jupiter.SpringExtension +import org.springframework.web.client.RestTemplate +import java.time.Instant +import kotlin.test.assertEquals +import kotlin.test.assertNotNull +import kotlin.test.assertTrue + +@ExtendWith(SpringExtension::class) +class SmsOtpServiceTest { + + @Mock + lateinit var restTemplate: RestTemplate + + private lateinit var service: SmsOtpService + + private val phone = "23458689595" + + @BeforeEach + fun setUp() { + service = SmsOtpService(restTemplate) + } + + @Test + fun givenValidPhone_whenSendOtp_ReturnSuccess() { + + val ref = "12234344" + val apiResult = mapOf("status" to 200, "pinId" to ref) + + val mockResponse = ResponseEntity.ok().body>(apiResult) + `when`(restTemplate.postForEntity(any(), any(), any>())).thenReturn(mockResponse) + + val result = service.sendOtp(phone) + + assertNotNull(result) + assertEquals(ref, result.ref, "Otp reference from API is return") + assertTrue(result.expireAt.toInstant().isAfter(Instant.now()), "ExpireAt is in the future") + verify(restTemplate, times(1)).postForEntity(any(), any(), any>()) + verifyNoMoreInteractions(restTemplate) + } + + @Test + fun givenValidOtpRef_whenVerifyOtp_ReturnSuccess() { + + val apiResult = mapOf("status" to 200, "msisdn" to phone) + + val mockResponse = ResponseEntity.ok(apiResult) + `when`(restTemplate.postForEntity(any(), any(), any>())).thenReturn(mockResponse) + + val ref = "12234344" + val result = service.verifyOtp("", ref) + + assertNotNull(result) + assertEquals(phone, result, "Correct phone number is return") + verify(restTemplate, times(1)).postForEntity(any(), any(), any>()) + verifyNoMoreInteractions(restTemplate) + } +} \ No newline at end of file diff --git a/src/test/kotlin/ng/cove/web/http/controller/AdminControllerTest.kt b/src/test/kotlin/ng/cove/web/http/controller/AdminControllerTest.kt index c80fbcd..fb93055 100644 --- a/src/test/kotlin/ng/cove/web/http/controller/AdminControllerTest.kt +++ b/src/test/kotlin/ng/cove/web/http/controller/AdminControllerTest.kt @@ -4,7 +4,6 @@ package ng.cove.web.http.controller import com.google.firebase.auth.FirebaseToken import ng.cove.web.AppTest import ng.cove.web.data.model.* -import ng.cove.web.service.CacheService import org.junit.jupiter.api.Assertions.assertEquals import org.junit.jupiter.api.BeforeEach import org.junit.jupiter.api.Nested @@ -12,8 +11,10 @@ import org.junit.jupiter.api.Test import org.mockito.Mockito import org.mockito.Mockito.verifyNoInteractions import org.mockito.Mockito.`when` -import org.mockito.kotlin.* -import org.springframework.boot.test.mock.mockito.MockBean +import org.mockito.kotlin.any +import org.mockito.kotlin.reset +import org.mockito.kotlin.times +import org.mockito.kotlin.verify import org.springframework.http.MediaType import org.springframework.test.context.ActiveProfiles import org.springframework.test.web.servlet.delete @@ -21,15 +22,12 @@ import org.springframework.test.web.servlet.post @ActiveProfiles("test") class AdminControllerTest : AppTest() { - @MockBean - lateinit var cacheService: CacheService @Test fun givenUserNoJWT_WhenDeleteGuard_ThenReturns401() { val result = mockMvc.delete("/admin/guard/{guard_id}", 1).andReturn().response assertEquals(401, result.status) verifyNoInteractions(auth) - verify(cacheService, never()).getMemberById(member.id!!) } @Test @@ -44,7 +42,6 @@ class AdminControllerTest : AppTest() { assertEquals(401, result.status) verify(auth, times(1)).verifyIdToken(unknownToken, true) - verify(cacheService, never()).getMemberById(member.id!!) } @Nested @@ -57,11 +54,9 @@ class AdminControllerTest : AppTest() { fun setUp() { //auth user reset(auth) - reset(cacheService) `when`(auth.verifyIdToken(idToken, true)).thenReturn(firebaseToken) `when`(firebaseToken.claims).thenReturn(mapOf("type" to UserType.Member.name)) `when`(firebaseToken.uid).thenReturn(member.id) - `when`(cacheService.getMemberById(member.id!!)).thenReturn(member) } @@ -76,7 +71,6 @@ class AdminControllerTest : AppTest() { assertEquals(401, result.status) verify(firebaseToken, times(1)).uid verify(firebaseToken, times(1)).claims - verify(cacheService, times(1)).getMemberById(member.id!!) } @Test @@ -105,6 +99,7 @@ class AdminControllerTest : AppTest() { } joinRequestRepo.save(request) + memberRepo.save(member) val result = mockMvc.post("/admin/community/request/{accept}", true) { header("Authorization", "Bearer $idToken") diff --git a/src/test/kotlin/ng/cove/web/http/controller/DefaultControllerTest.kt b/src/test/kotlin/ng/cove/web/http/controller/DefaultControllerTest.kt index 076a40d..b452593 100644 --- a/src/test/kotlin/ng/cove/web/http/controller/DefaultControllerTest.kt +++ b/src/test/kotlin/ng/cove/web/http/controller/DefaultControllerTest.kt @@ -1,6 +1,7 @@ package ng.cove.web.http.controller import ng.cove.web.AppTest +import ng.cove.web.component.SmsOtpService import ng.cove.web.data.model.PhoneOtp import ng.cove.web.data.model.UserType import ng.cove.web.http.body.OtpRefBody @@ -11,12 +12,10 @@ import org.junit.jupiter.api.BeforeEach import org.junit.jupiter.api.Nested import org.junit.jupiter.api.Test import org.mockito.Mockito -import org.mockito.kotlin.any -import org.mockito.kotlin.times -import org.mockito.kotlin.verify -import org.mockito.kotlin.verifyNoInteractions +import org.mockito.kotlin.* +import org.springframework.beans.factory.annotation.Value +import org.springframework.boot.test.mock.mockito.MockBean import org.springframework.http.MediaType -import org.springframework.test.context.event.annotation.AfterTestClass import org.springframework.test.web.servlet.get import org.springframework.test.web.servlet.post import org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get @@ -28,6 +27,19 @@ import java.util.* //@WebAppConfiguration("") class DefaultControllerTest : AppTest() { + @MockBean + lateinit var smsOtpService: SmsOtpService + + @Value("\${otp.trial-limit}") + var maxDailyOtpTrial: Int = 0 + + @BeforeEach + override fun setUp() { + super.setUp() + reset(smsOtpService) + reset(auth) + } + @Test fun givenPhoneRegistered_whenUserPhoneGetOtp_thenSuccess() { val phone = member.phone!! @@ -61,16 +73,27 @@ class DefaultControllerTest : AppTest() { val otpRefBody = OtpRefBody(ref, phone, Date(), 2) Mockito.`when`(smsOtpService.sendOtp(member.phone!!)).thenReturn(otpRefBody) - val result = mockMvc.perform( get("/user/login").param("phone", phone) .contentType(MediaType.APPLICATION_JSON_VALUE) ).andReturn().response - println("Test result: ${result.contentAsString}") assertTrue(result.status == 400, "Should return 400") } + @Test + fun givenUserIsTester_whenLoginWithPhone_ThenDoNotSendOtp(){ + member.testOtp = faker.random().nextInt(6).toString() + memberRepo.save(member) + + val result = mockMvc.perform( + get("/user/login").param("phone", member.phone!!) + .contentType(MediaType.APPLICATION_JSON_VALUE) + ).andReturn().response + assertEquals(200, result.status) + verifyNoInteractions(smsOtpService) + } + @Nested inner class OTPVerificationTest{ @@ -134,6 +157,38 @@ class DefaultControllerTest : AppTest() { verify(smsOtpService, times(1)).verifyOtp(otp, ref) verify(auth, times(1)).createCustomToken(any(), any()) } + + @Test + fun givenUserIsTester_whenVerifyOtp_ThenDoNotCallVerificationService(){ + val otp = faker.random().nextInt(6).toString() + member.testOtp = otp + memberRepo.save(member) + + val customJWT = faker.random().hex(55) + Mockito.`when`(auth.createCustomToken(member.id!!, mapOf("type" to "Member"))) + .thenReturn(customJWT) + + val login = mapOf( + "otp" to otp, + "ref" to member.id, + "device_id" to faker.random().hex(30), + "device_name" to faker.device().modelName() + ) + + val result = mockMvc.post("/user/login/verify") { + contentType = MediaType.APPLICATION_JSON + content = mapper.writeValueAsString(login) + }.andReturn().response + + assertEquals(200, result.status) + verifyNoInteractions(smsOtpService) + } + + @Test + fun givenRouteIsNotConfigured_whenRouteIsCalled_thenReturn404(){ + val result = mockMvc.post("/").andReturn().response + assertTrue(result.status == 404) + } } diff --git a/src/test/kotlin/ng/cove/web/service/CacheTest.kt b/src/test/kotlin/ng/cove/web/service/CacheTest.kt new file mode 100644 index 0000000..ab8b23f --- /dev/null +++ b/src/test/kotlin/ng/cove/web/service/CacheTest.kt @@ -0,0 +1,35 @@ +package ng.cove.web.service + +import ng.cove.web.AppTest +import ng.cove.web.data.model.Member +import ng.cove.web.util.CacheNames +import org.junit.jupiter.api.Assertions.assertNull +import org.junit.jupiter.api.Test +import org.springframework.beans.factory.annotation.Autowired +import org.springframework.cache.caffeine.CaffeineCacheManager +import kotlin.test.assertEquals +import kotlin.test.assertNotNull + + +class CacheTest : AppTest() { + + @Autowired + lateinit var cacheManager: CaffeineCacheManager + + @Test + fun givenMemberThatShouldBeCached_whenMemberIsQueriedById_thenMemberIsAddedToCache() { + memberRepo.save(member) + val id = member.id!! + val cacheName = CacheNames.MEMBERS + cacheManager.getCache(cacheName)?.clear() + var cachedMember = cacheManager.getCache(cacheName)?.get(id, Member::class.java) + + assertNull(cachedMember, "Cache is empty before query") + + val queriedMember = memberRepo.findMemberById(id)!! + cachedMember = cacheManager.getCache(cacheName)!!.get(id, Member::class.java) + + assertNotNull(cachedMember, "Cache is not empty before query") + assertEquals(cachedMember, queriedMember, "Member is cached after query") + } +} \ No newline at end of file diff --git a/src/test/kotlin/ng/cove/web/service/LevyServiceTest.kt b/src/test/kotlin/ng/cove/web/service/LevyServiceTest.kt index ec17293..65d0964 100644 --- a/src/test/kotlin/ng/cove/web/service/LevyServiceTest.kt +++ b/src/test/kotlin/ng/cove/web/service/LevyServiceTest.kt @@ -1,53 +1,90 @@ package ng.cove.web.service import ng.cove.web.AppTest -import ng.cove.web.data.model.Community +import ng.cove.web.data.model.AssignedLevy import ng.cove.web.data.model.Levy import ng.cove.web.data.model.LevyType -import ng.cove.web.data.model.Member +import ng.cove.web.data.repo.AssignedLevyRepo +import ng.cove.web.data.repo.LevyPaymentRepo +import ng.cove.web.data.repo.LevyRepo +import org.junit.jupiter.api.AfterAll import org.junit.jupiter.api.BeforeEach +import org.junit.jupiter.api.Test import org.springframework.beans.factory.annotation.Autowired +import org.springframework.web.context.WebApplicationContext +import java.time.Duration +import java.time.Instant import java.util.* +import kotlin.test.assertTrue class LevyServiceTest : AppTest() { @Autowired - private lateinit var levyService: LevyService + lateinit var levyRepo: LevyRepo - private lateinit var admin: Member - private lateinit var createdLevy: Levy - private lateinit var levyData: Levy + @Autowired + lateinit var assignedLevyRepo: AssignedLevyRepo + + @Autowired + lateinit var levyPaymentRepo: LevyPaymentRepo + + lateinit var assignedLevyNotDueForPayment: AssignedLevy + + @Autowired + lateinit var webApplicationContext: WebApplicationContext @BeforeEach override fun setUp() { super.setUp() - community = Community() - community.id = "2133232323423" - community.name = "Test Community" - community.address = "123 Test Street" - - admin = Member() - admin.id = "admin_user_id" - admin.firstName = "Janet" - admin.lastName = "Doe" - admin.phone = "234234234234" - - community.superAdminId = admin.id - community.adminIds = setOf(admin.id!!) - - admin.community = community - - levyData = Levy() - levyData.title = "Apartments" - levyData.communityId = community.id - levyData.type = LevyType.Annual - levyData.amount = 1200000.0 - - createdLevy = levyData - createdLevy.createdAt = Calendar.getInstance().time - createdLevy.updatedAt = Calendar.getInstance().time - createdLevy.id = "levy_id" + var levy = Levy().apply { + title = "Refuse dump fee" + communityId = community.id + amount = 23000.0 + type = LevyType.Month + } + levy = levyRepo.save(levy) + assignedLevyNotDueForPayment = AssignedLevy().apply { + levyId = levy.id + memberId = member.id + assignedBy = member.id + nextPaymentDue = Date.from(Instant.now().plus(Duration.ofDays(1))) + } + assignedLevyNotDueForPayment = assignedLevyRepo.save(assignedLevyNotDueForPayment) + } + + @AfterAll + fun teardown() { + levyRepo.deleteAll() + assignedLevyRepo.deleteAll() } + + @Test + fun givenThatAssignedLevyPaymentIsDue_whenSchedulerRuns_PaymentIsCreated() { + + assertTrue(levyPaymentRepo.count().toInt() == 0, "No payment yet") + + // Backdating nextPaymentDue + val aHourAgo = Date.from(Instant.now().minus(Duration.ofHours(1))) + assignedLevyNotDueForPayment.nextPaymentDue = aHourAgo + assignedLevyRepo.save(assignedLevyNotDueForPayment) + + /** This [delayInSecs] value is updated by [AppTest]'s @SpringBootTest annotation to fast track + the execution of this test. The production [delayInSecs] value is not affect*/ + val delayInSecs = webApplicationContext.environment.getProperty("schedule.levy.duration.secs") + + /** Waiting for [LevyService.createLevyPayments]'s [@Scheduled] annotation + to take effect */ + Thread.sleep(delayInSecs!!.toLong() * 1000) + + assertTrue(levyPaymentRepo.count().toInt() > 0, "Payment created by Scheduled function") + val createdPayment = assignedLevyRepo.findAll().first()!! + val now = Instant.now() + assertTrue( + createdPayment.nextPaymentDue!!.toInstant().isAfter(now), + "Next payment updated to future" + ) + } + } \ No newline at end of file diff --git "a/src/test/resources/application\342\200\223test.properties" b/src/test/resources/application.properties similarity index 61% rename from "src/test/resources/application\342\200\223test.properties" rename to src/test/resources/application.properties index ace41f7..a0b1120 100644 --- "a/src/test/resources/application\342\200\223test.properties" +++ b/src/test/resources/application.properties @@ -1,6 +1,12 @@ +schedule.levy.duration.secs=2 + +otp.trial-limit=3 +otp.expiry-mins=10 +visitor.access-code.length=8 + +#Disable GCP components autoconfiguration in test spring.cloud.gcp.core.enabled=false spring.cloud.gcp.config.enabled=false - spring.cloud.gcp.firestore.enabled=false spring.cloud.gcp.storage.enabled=false spring.cloud.gcp.logging.enabled=false