Compare commits
57 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fe8e766670 | ||
|
|
b63863e283 | ||
|
|
516f9fb951 | ||
|
|
c2234ec647 | ||
|
|
02cbee5c8c | ||
|
|
518436ffd2 | ||
|
|
311aa2b1ac | ||
|
|
8ad44ddc23 | ||
|
|
874aad87a8 | ||
|
|
f2e875d03d | ||
|
|
45fd32c8ca | ||
|
|
08b36ba69f | ||
|
|
d5cfe577a0 | ||
|
|
c131cfef27 | ||
|
|
526d4cd133 | ||
|
|
8e09871b44 | ||
|
|
6aa3ce0b32 | ||
|
|
23ea8e493e | ||
|
|
22d996bed7 | ||
|
|
9df864787c | ||
|
|
b7a6ac9ec0 | ||
|
|
38d8d416e1 | ||
|
|
592613a99a | ||
|
|
e9f43c6a53 | ||
|
|
89c6800913 | ||
|
|
db9c807289 | ||
|
|
6e48a3eac7 | ||
|
|
ba9a716913 | ||
|
|
6d7292a60a | ||
|
|
c0d02732e7 | ||
|
|
d071a91917 | ||
|
|
d67e14b08f | ||
|
|
174a386f54 | ||
|
|
ce23cfc5b5 | ||
|
|
8be690c417 | ||
|
|
2f778dc4b4 | ||
|
|
e0c165313b | ||
|
|
c86192d301 | ||
|
|
775c105d1d | ||
|
|
be85c0a560 | ||
|
|
70e1af15cc | ||
|
|
6afea91c5f | ||
|
|
7a4c892804 | ||
|
|
9b50b1ea42 | ||
|
|
52a18d21f1 | ||
|
|
8aaf312c67 | ||
|
|
ede06900e5 | ||
|
|
a42455c9dc | ||
|
|
547a84f2fc | ||
|
|
f56ea99dc2 | ||
|
|
4cea5d2310 | ||
|
|
e4f09b4e68 | ||
|
|
cc094fe04e | ||
|
|
ac5c743390 | ||
|
|
963db18642 | ||
|
|
748fb76eaf | ||
|
|
d93a698e87 |
5
.gitignore
vendored
5
.gitignore
vendored
@@ -138,4 +138,7 @@ dmypy.json
|
|||||||
secrets.yml
|
secrets.yml
|
||||||
.vscode
|
.vscode
|
||||||
.DS_Store
|
.DS_Store
|
||||||
compiledstatic/
|
compiledstatic/
|
||||||
|
|
||||||
|
# Pycharm
|
||||||
|
.idea
|
||||||
|
|||||||
9
Pipfile
9
Pipfile
@@ -4,13 +4,13 @@ url = "https://pypi.org/simple"
|
|||||||
verify_ssl = true
|
verify_ssl = true
|
||||||
|
|
||||||
[packages]
|
[packages]
|
||||||
django = "~=3.0"
|
django = "~=3.1"
|
||||||
django-allauth = "~=0.42.0"
|
django-allauth = "~=0.42.0"
|
||||||
geoip2 = "~=3.0.0"
|
geoip2 = "~=3.0.0"
|
||||||
whitenoise = "~=5.1.0"
|
whitenoise = "~=5.1.0"
|
||||||
celery = "~=4.4.6"
|
celery = "~=4.4.6"
|
||||||
django-ipware = "~=2.1.0"
|
django-ipware = "~=2.1.0"
|
||||||
pyyaml = "~=5.3.1"
|
pyyaml = "~=5.4"
|
||||||
ua-parser = "~=0.10.0"
|
ua-parser = "~=0.10.0"
|
||||||
user-agents = "~=2.1"
|
user-agents = "~=2.1"
|
||||||
emoji-country-flag = "~=1.2.1"
|
emoji-country-flag = "~=1.2.1"
|
||||||
@@ -18,8 +18,9 @@ rules = "~=2.2"
|
|||||||
gunicorn = "~=20.0.4"
|
gunicorn = "~=20.0.4"
|
||||||
psycopg2-binary = "~=2.8.5"
|
psycopg2-binary = "~=2.8.5"
|
||||||
redis = "~=3.5.3"
|
redis = "~=3.5.3"
|
||||||
django-redis-cache = "~=2.1.1"
|
django-redis-cache = "~=3.0.0"
|
||||||
pycountry = "~=19.8.18"
|
pycountry = "~=19.8.18"
|
||||||
html2text = "~=2020.1.16"
|
html2text = "~=2020.1.16"
|
||||||
django-health-check = "~=3.12.1"
|
django-health-check = "~=3.12.1"
|
||||||
django-npm = "~=1.0.0"
|
django-npm = "~=1.0.0"
|
||||||
|
django-debug-toolbar = "*"
|
||||||
|
|||||||
135
Pipfile.lock
generated
135
Pipfile.lock
generated
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"_meta": {
|
"_meta": {
|
||||||
"hash": {
|
"hash": {
|
||||||
"sha256": "f9ab7bf92fe2342e3d221ab34ac957bf217638062da0554b2328502650515741"
|
"sha256": "f8c76565a776f1bd36364077a86d6c16fccc522d9d2024bb9b51be5cb9f8b4b5"
|
||||||
},
|
},
|
||||||
"pipfile-spec": 6,
|
"pipfile-spec": 6,
|
||||||
"requires": {},
|
"requires": {},
|
||||||
@@ -19,14 +19,16 @@
|
|||||||
"sha256:70cdb10628468ff14e57ec2f751c7aa9e48e7e3651cfd62d431213c0c4e58f21",
|
"sha256:70cdb10628468ff14e57ec2f751c7aa9e48e7e3651cfd62d431213c0c4e58f21",
|
||||||
"sha256:aa7f313fb887c91f15474c1229907a04dac0b8135822d6603437803424c0aa59"
|
"sha256:aa7f313fb887c91f15474c1229907a04dac0b8135822d6603437803424c0aa59"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'",
|
||||||
"version": "==2.6.1"
|
"version": "==2.6.1"
|
||||||
},
|
},
|
||||||
"asgiref": {
|
"asgiref": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:7e51911ee147dd685c3c8b805c0ad0cb58d360987b56953878f8c06d2d1c6f1a",
|
"sha256:5ee950735509d04eb673bd7f7120f8fa1c9e2df495394992c73234d526907e17",
|
||||||
"sha256:9fc6fb5d39b8af147ba40765234fa822b39818b12cc80b35ad9b0cef3a476aed"
|
"sha256:7162a3cb30ab0609f1a4c95938fd73e8604f63bdba516a7f7d64b83ff09478f0"
|
||||||
],
|
],
|
||||||
"version": "==3.2.10"
|
"markers": "python_version >= '3.5'",
|
||||||
|
"version": "==3.3.1"
|
||||||
},
|
},
|
||||||
"billiard": {
|
"billiard": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
@@ -45,32 +47,34 @@
|
|||||||
},
|
},
|
||||||
"certifi": {
|
"certifi": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:5930595817496dd21bb8dc35dad090f1c2cd0adfaf21204bf6732ca5d8ee34d3",
|
"sha256:1a4995114262bffbc2413b159f2a1a480c969de6e6eb13ee966d470af86af59c",
|
||||||
"sha256:8fc0819f1f30ba15bdb34cceffb9ef04d99f420f68eb75d901e9560b8749fc41"
|
"sha256:719a74fb9e33b9bd44cc7f3a8d94bc35e4049deebe19ba7d8e108280cfd59830"
|
||||||
],
|
],
|
||||||
"version": "==2020.6.20"
|
"version": "==2020.12.5"
|
||||||
},
|
},
|
||||||
"chardet": {
|
"chardet": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae",
|
"sha256:0d6f53a15db4120f2b08c94f11e7d93d2c911ee118b6b30a04ec3ee8310179fa",
|
||||||
"sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691"
|
"sha256:f864054d66fd9118f2e67044ac8981a54775ec5b67aed0441892edb553d21da5"
|
||||||
],
|
],
|
||||||
"version": "==3.0.4"
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'",
|
||||||
|
"version": "==4.0.0"
|
||||||
},
|
},
|
||||||
"defusedxml": {
|
"defusedxml": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:6687150770438374ab581bb7a1b327a847dd9c5749e396102de3fad4e8a3ef93",
|
"sha256:1bb3032db185915b62d7c6209c5a8792be6a32ab2fedacc84e01b52c51aa3e69",
|
||||||
"sha256:f684034d135af4c6cbb949b8a4d2ed61634515257a67299e5f940fbaa34377f5"
|
"sha256:a352e7e428770286cc899e2542b6cdaedb2b4953ff269a210103ec58f6198a61"
|
||||||
],
|
],
|
||||||
"version": "==0.6.0"
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'",
|
||||||
|
"version": "==0.7.1"
|
||||||
},
|
},
|
||||||
"django": {
|
"django": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:a2127ad0150ec6966655bedf15dbbff9697cc86d61653db2da1afa506c0b04cc",
|
"sha256:32ce792ee9b6a0cbbec340123e229ac9f765dff8c2a4ae9247a14b2ba3a365a7",
|
||||||
"sha256:c93c28ccf1d094cbd00d860e83128a39e45d2c571d3b54361713aaaf9a94cac4"
|
"sha256:baf099db36ad31f970775d0be5587cc58a6256a6771a44eb795b554d45f211b8"
|
||||||
],
|
],
|
||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==3.1.2"
|
"version": "==3.1.7"
|
||||||
},
|
},
|
||||||
"django-allauth": {
|
"django-allauth": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
@@ -79,6 +83,14 @@
|
|||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==0.42.0"
|
"version": "==0.42.0"
|
||||||
},
|
},
|
||||||
|
"django-debug-toolbar": {
|
||||||
|
"hashes": [
|
||||||
|
"sha256:84e2607d900dbd571df0a2acf380b47c088efb787dce9805aefeb407341961d2",
|
||||||
|
"sha256:9e5a25d0c965f7e686f6a8ba23613ca9ca30184daa26487706d4829f5cfb697a"
|
||||||
|
],
|
||||||
|
"index": "pypi",
|
||||||
|
"version": "==3.2"
|
||||||
|
},
|
||||||
"django-health-check": {
|
"django-health-check": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:2667b89b8f85ad9b2a24c90581b376016d22ea912fedf37f9866413a3c2e0a5d",
|
"sha256:2667b89b8f85ad9b2a24c90581b376016d22ea912fedf37f9866413a3c2e0a5d",
|
||||||
@@ -103,19 +115,18 @@
|
|||||||
},
|
},
|
||||||
"django-redis-cache": {
|
"django-redis-cache": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:9b2c45a1bc0f295bccd56c2542d937665ae98f3325f20b3d82fc620e14395d52",
|
"sha256:9a2eebef421d996a82098a19d17ff6b321265cd73178fa398913019764e8394a"
|
||||||
"sha256:e72691539be99c0b2dd64ac380e26f4d2be5c53c1b2d26845dd279ae38b47477"
|
|
||||||
],
|
],
|
||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==2.1.3"
|
"version": "==3.0.0"
|
||||||
},
|
},
|
||||||
"emoji-country-flag": {
|
"emoji-country-flag": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:67c0cb6a3765fb53f31b34160d6b1c8a5f44b297bc278d1835c6f2e5b0a9a592",
|
"sha256:338f5e374119dcde093cfeaa8ca3af372d4b8d984d89a7fb2fb0db0011662560",
|
||||||
"sha256:ae7edb38077b0840210fa9e37673f481f2b9c032446e13ad6dab2b1108cd7ad6"
|
"sha256:a3a068191294294143d8ef294fdfe9792c5c243753eac130798bf2fa5de38185"
|
||||||
],
|
],
|
||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==1.2.1"
|
"version": "==1.2.4"
|
||||||
},
|
},
|
||||||
"geoip2": {
|
"geoip2": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
@@ -146,6 +157,7 @@
|
|||||||
"sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6",
|
"sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6",
|
||||||
"sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"
|
"sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||||
"version": "==2.10"
|
"version": "==2.10"
|
||||||
},
|
},
|
||||||
"kombu": {
|
"kombu": {
|
||||||
@@ -153,12 +165,14 @@
|
|||||||
"sha256:be48cdffb54a2194d93ad6533d73f69408486483d189fe9f5990ee24255b0e0a",
|
"sha256:be48cdffb54a2194d93ad6533d73f69408486483d189fe9f5990ee24255b0e0a",
|
||||||
"sha256:ca1b45faac8c0b18493d02a8571792f3c40291cf2bcf1f55afed3d8f3aa7ba74"
|
"sha256:ca1b45faac8c0b18493d02a8571792f3c40291cf2bcf1f55afed3d8f3aa7ba74"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'",
|
||||||
"version": "==4.6.11"
|
"version": "==4.6.11"
|
||||||
},
|
},
|
||||||
"maxminddb": {
|
"maxminddb": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:47e86a084dd814fac88c99ea34ba3278a74bc9de5a25f4b815b608798747c7dc"
|
"sha256:47e86a084dd814fac88c99ea34ba3278a74bc9de5a25f4b815b608798747c7dc"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '3.6'",
|
||||||
"version": "==2.0.3"
|
"version": "==2.0.3"
|
||||||
},
|
},
|
||||||
"oauthlib": {
|
"oauthlib": {
|
||||||
@@ -166,6 +180,7 @@
|
|||||||
"sha256:bee41cc35fcca6e988463cacc3bcb8a96224f470ca547e697b604cc697b2f889",
|
"sha256:bee41cc35fcca6e988463cacc3bcb8a96224f470ca547e697b604cc697b2f889",
|
||||||
"sha256:df884cd6cbe20e32633f1db1072e9356f53638e4361bef4e8b03c9127c9328ea"
|
"sha256:df884cd6cbe20e32633f1db1072e9356f53638e4361bef4e8b03c9127c9328ea"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||||
"version": "==3.1.0"
|
"version": "==3.1.0"
|
||||||
},
|
},
|
||||||
"psycopg2-binary": {
|
"psycopg2-binary": {
|
||||||
@@ -173,9 +188,11 @@
|
|||||||
"sha256:0deac2af1a587ae12836aa07970f5cb91964f05a7c6cdb69d8425ff4c15d4e2c",
|
"sha256:0deac2af1a587ae12836aa07970f5cb91964f05a7c6cdb69d8425ff4c15d4e2c",
|
||||||
"sha256:0e4dc3d5996760104746e6cfcdb519d9d2cd27c738296525d5867ea695774e67",
|
"sha256:0e4dc3d5996760104746e6cfcdb519d9d2cd27c738296525d5867ea695774e67",
|
||||||
"sha256:11b9c0ebce097180129e422379b824ae21c8f2a6596b159c7659e2e5a00e1aa0",
|
"sha256:11b9c0ebce097180129e422379b824ae21c8f2a6596b159c7659e2e5a00e1aa0",
|
||||||
|
"sha256:15978a1fbd225583dd8cdaf37e67ccc278b5abecb4caf6b2d6b8e2b948e953f6",
|
||||||
"sha256:1fabed9ea2acc4efe4671b92c669a213db744d2af8a9fc5d69a8e9bc14b7a9db",
|
"sha256:1fabed9ea2acc4efe4671b92c669a213db744d2af8a9fc5d69a8e9bc14b7a9db",
|
||||||
"sha256:2dac98e85565d5688e8ab7bdea5446674a83a3945a8f416ad0110018d1501b94",
|
"sha256:2dac98e85565d5688e8ab7bdea5446674a83a3945a8f416ad0110018d1501b94",
|
||||||
"sha256:42ec1035841b389e8cc3692277a0bd81cdfe0b65d575a2c8862cec7a80e62e52",
|
"sha256:42ec1035841b389e8cc3692277a0bd81cdfe0b65d575a2c8862cec7a80e62e52",
|
||||||
|
"sha256:6422f2ff0919fd720195f64ffd8f924c1395d30f9a495f31e2392c2efafb5056",
|
||||||
"sha256:6a32f3a4cb2f6e1a0b15215f448e8ce2da192fd4ff35084d80d5e39da683e79b",
|
"sha256:6a32f3a4cb2f6e1a0b15215f448e8ce2da192fd4ff35084d80d5e39da683e79b",
|
||||||
"sha256:7312e931b90fe14f925729cde58022f5d034241918a5c4f9797cac62f6b3a9dd",
|
"sha256:7312e931b90fe14f925729cde58022f5d034241918a5c4f9797cac62f6b3a9dd",
|
||||||
"sha256:7d92a09b788cbb1aec325af5fcba9fed7203897bbd9269d5691bb1e3bce29550",
|
"sha256:7d92a09b788cbb1aec325af5fcba9fed7203897bbd9269d5691bb1e3bce29550",
|
||||||
@@ -223,27 +240,45 @@
|
|||||||
},
|
},
|
||||||
"pytz": {
|
"pytz": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:a494d53b6d39c3c6e44c3bec237336e14305e4f29bbf800b599253057fbb79ed",
|
"sha256:83a4a90894bf38e243cf052c8b58f381bfe9a7a483f6a9cab140bc7f702ac4da",
|
||||||
"sha256:c35965d010ce31b23eeb663ed3cc8c906275d6be1a34393a1d73a41febf4a048"
|
"sha256:eb10ce3e7736052ed3623d49975ce333bcd712c7bb19a58b9e2089d4057d0798"
|
||||||
],
|
],
|
||||||
"version": "==2020.1"
|
"version": "==2021.1"
|
||||||
},
|
},
|
||||||
"pyyaml": {
|
"pyyaml": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:06a0d7ba600ce0b2d2fe2e78453a470b5a6e000a985dd4a4e54e436cc36b0e97",
|
"sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf",
|
||||||
"sha256:240097ff019d7c70a4922b6869d8a86407758333f02203e0fc6ff79c5dcede76",
|
"sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696",
|
||||||
"sha256:4f4b913ca1a7319b33cfb1369e91e50354d6f07a135f3b901aca02aa95940bd2",
|
"sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393",
|
||||||
"sha256:69f00dca373f240f842b2931fb2c7e14ddbacd1397d57157a9b005a6a9942648",
|
"sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77",
|
||||||
"sha256:73f099454b799e05e5ab51423c7bcf361c58d3206fa7b0d555426b1f4d9a3eaf",
|
"sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922",
|
||||||
"sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f",
|
"sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5",
|
||||||
"sha256:7739fc0fa8205b3ee8808aea45e968bc90082c10aef6ea95e855e10abf4a37b2",
|
"sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8",
|
||||||
"sha256:95f71d2af0ff4227885f7a6605c37fd53d3a106fcab511b8860ecca9fcf400ee",
|
"sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10",
|
||||||
"sha256:b8eac752c5e14d3eca0e6dd9199cd627518cb5ec06add0de9d32baeee6fe645d",
|
"sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc",
|
||||||
"sha256:cc8955cfbfc7a115fa81d85284ee61147059a753344bc51098f3ccd69b0d7e0c",
|
"sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018",
|
||||||
"sha256:d13155f591e6fcc1ec3b30685d50bf0711574e2c0dfffd7644babf8b5102ca1a"
|
"sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e",
|
||||||
|
"sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253",
|
||||||
|
"sha256:72a01f726a9c7851ca9bfad6fd09ca4e090a023c00945ea05ba1638c09dc3347",
|
||||||
|
"sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183",
|
||||||
|
"sha256:895f61ef02e8fed38159bb70f7e100e00f471eae2bc838cd0f4ebb21e28f8541",
|
||||||
|
"sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb",
|
||||||
|
"sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185",
|
||||||
|
"sha256:bfb51918d4ff3d77c1c856a9699f8492c612cde32fd3bcd344af9be34999bfdc",
|
||||||
|
"sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db",
|
||||||
|
"sha256:cb333c16912324fd5f769fff6bc5de372e9e7a202247b48870bc251ed40239aa",
|
||||||
|
"sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46",
|
||||||
|
"sha256:d483ad4e639292c90170eb6f7783ad19490e7a8defb3e46f97dfe4bacae89122",
|
||||||
|
"sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b",
|
||||||
|
"sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63",
|
||||||
|
"sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df",
|
||||||
|
"sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc",
|
||||||
|
"sha256:fd7f6999a8070df521b6384004ef42833b9bd62cfee11a09bda1079b4b704247",
|
||||||
|
"sha256:fdc842473cd33f45ff6bce46aea678a54e3d21f1b61a7750ce3c498eedfe25d6",
|
||||||
|
"sha256:fe69978f3f768926cfa37b867e3843918e012cf83f680806599ddce33c2c68b0"
|
||||||
],
|
],
|
||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==5.3.1"
|
"version": "==5.4.1"
|
||||||
},
|
},
|
||||||
"redis": {
|
"redis": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
@@ -255,15 +290,17 @@
|
|||||||
},
|
},
|
||||||
"requests": {
|
"requests": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:b3559a131db72c33ee969480840fff4bb6dd111de7dd27c8ee1f820f4f00231b",
|
"sha256:27973dd4a904a4f13b263a19c866c13b92a39ed1c964655f025f3f8d3d75b804",
|
||||||
"sha256:fe75cc94a9443b9246fc7049224f75604b113c36acb93f87b80ed42c44cbb898"
|
"sha256:c210084e36a42ae6b9219e00e48287def368a26d03a048ddad7bfee44f75871e"
|
||||||
],
|
],
|
||||||
"version": "==2.24.0"
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'",
|
||||||
|
"version": "==2.25.1"
|
||||||
},
|
},
|
||||||
"requests-oauthlib": {
|
"requests-oauthlib": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:7f71572defaecd16372f9006f33c2ec8c077c3cfa6f5911a9a90202beb513f3d",
|
"sha256:7f71572defaecd16372f9006f33c2ec8c077c3cfa6f5911a9a90202beb513f3d",
|
||||||
"sha256:b4261601a71fd721a8bd6d7aa1cc1d6a8a93b4a9f5e96626f8e4d91e8beeaa6a"
|
"sha256:b4261601a71fd721a8bd6d7aa1cc1d6a8a93b4a9f5e96626f8e4d91e8beeaa6a",
|
||||||
|
"sha256:fa6c47b933f01060936d87ae9327fead68768b69c6c9ea2109c48be30f2d4dbc"
|
||||||
],
|
],
|
||||||
"version": "==1.3.0"
|
"version": "==1.3.0"
|
||||||
},
|
},
|
||||||
@@ -274,18 +311,12 @@
|
|||||||
"index": "pypi",
|
"index": "pypi",
|
||||||
"version": "==2.2"
|
"version": "==2.2"
|
||||||
},
|
},
|
||||||
"six": {
|
|
||||||
"hashes": [
|
|
||||||
"sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259",
|
|
||||||
"sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"
|
|
||||||
],
|
|
||||||
"version": "==1.15.0"
|
|
||||||
},
|
|
||||||
"sqlparse": {
|
"sqlparse": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:017cde379adbd6a1f15a61873f43e8274179378e95ef3fede90b5aa64d304ed0",
|
"sha256:017cde379adbd6a1f15a61873f43e8274179378e95ef3fede90b5aa64d304ed0",
|
||||||
"sha256:0f91fd2e829c44362cbcfab3e9ae12e22badaa8a29ad5ff599f9ec109f0454e8"
|
"sha256:0f91fd2e829c44362cbcfab3e9ae12e22badaa8a29ad5ff599f9ec109f0454e8"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '3.5'",
|
||||||
"version": "==0.4.1"
|
"version": "==0.4.1"
|
||||||
},
|
},
|
||||||
"ua-parser": {
|
"ua-parser": {
|
||||||
@@ -298,10 +329,11 @@
|
|||||||
},
|
},
|
||||||
"urllib3": {
|
"urllib3": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
"sha256:91056c15fa70756691db97756772bb1eb9678fa585d9184f24534b100dc60f4a",
|
"sha256:2f4da4594db7e1e110a944bb1b551fdf4e6c136ad42e4234131391e21eb5b0df",
|
||||||
"sha256:e7983572181f5e1522d9c98453462384ee92a0be7fac5f1413a1e35c56cc0461"
|
"sha256:e7b021f7241115872f92f43c6508082facffbd1c048e3c6e2bb9c2a157e28937"
|
||||||
],
|
],
|
||||||
"version": "==1.25.10"
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'",
|
||||||
|
"version": "==1.26.4"
|
||||||
},
|
},
|
||||||
"user-agents": {
|
"user-agents": {
|
||||||
"hashes": [
|
"hashes": [
|
||||||
@@ -316,6 +348,7 @@
|
|||||||
"sha256:133ee6d7a9016f177ddeaf191c1f58421a1dcc6ee9a42c58b34bed40e1d2cd87",
|
"sha256:133ee6d7a9016f177ddeaf191c1f58421a1dcc6ee9a42c58b34bed40e1d2cd87",
|
||||||
"sha256:ea4947cc56d1fd6f2095c8d543ee25dad966f78692528e68b4fada11ba3f98af"
|
"sha256:ea4947cc56d1fd6f2095c8d543ee25dad966f78692528e68b4fada11ba3f98af"
|
||||||
],
|
],
|
||||||
|
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'",
|
||||||
"version": "==1.3.0"
|
"version": "==1.3.0"
|
||||||
},
|
},
|
||||||
"whitenoise": {
|
"whitenoise": {
|
||||||
|
|||||||
13
TEMPLATE.env
13
TEMPLATE.env
@@ -70,3 +70,16 @@ SHOW_SHYNET_VERSION=True
|
|||||||
# that you have a separate queue consumer running somewhere via `celeryworker.sh`.
|
# that you have a separate queue consumer running somewhere via `celeryworker.sh`.
|
||||||
# CELERY_TASK_ALWAYS_EAGER=False
|
# CELERY_TASK_ALWAYS_EAGER=False
|
||||||
# CELERY_BROKER_URL=redis://redis.default.svc.cluster.local/1
|
# CELERY_BROKER_URL=redis://redis.default.svc.cluster.local/1
|
||||||
|
|
||||||
|
# Should Shynet show third-party icons in the dashboard?
|
||||||
|
SHOW_THIRD_PARTY_ICONS=True
|
||||||
|
|
||||||
|
# Should Shynet block collection of IP addresses globally?
|
||||||
|
BLOCK_ALL_IPS=False
|
||||||
|
|
||||||
|
# Should Shynet include the date and site ID when hashing users?
|
||||||
|
# This will prevent any possibility of cross-site tracking provided
|
||||||
|
# that IP collection is also disabled, and external keys (primary
|
||||||
|
# keys) aren't supplied. It will also prevent sessions from spanning
|
||||||
|
# one day to another.
|
||||||
|
AGGRESSIVE_HASH_SALTING=True
|
||||||
Binary file not shown.
|
Before Width: | Height: | Size: 215 KiB After Width: | Height: | Size: 263 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 360 KiB After Width: | Height: | Size: 240 KiB |
@@ -17,7 +17,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: "shynet-webserver"
|
- name: "shynet-webserver"
|
||||||
image: "milesmcc/shynet:latest"
|
image: "milesmcc/shynet:dev"
|
||||||
imagePullPolicy: Always
|
imagePullPolicy: Always
|
||||||
envFrom:
|
envFrom:
|
||||||
- secretRef:
|
- secretRef:
|
||||||
@@ -42,7 +42,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: "shynet-celeryworker"
|
- name: "shynet-celeryworker"
|
||||||
image: "milesmcc/shynet:latest"
|
image: "milesmcc/shynet:dev"
|
||||||
command: ["./celeryworker.sh"]
|
command: ["./celeryworker.sh"]
|
||||||
imagePullPolicy: Always
|
imagePullPolicy: Always
|
||||||
envFrom:
|
envFrom:
|
||||||
@@ -61,7 +61,7 @@ spec:
|
|||||||
selector:
|
selector:
|
||||||
app: shynet-redis
|
app: shynet-redis
|
||||||
---
|
---
|
||||||
apiVersion: apps/v1beta2
|
apiVersion: apps/v1
|
||||||
kind: StatefulSet
|
kind: StatefulSet
|
||||||
metadata:
|
metadata:
|
||||||
name: shynet-redis
|
name: shynet-redis
|
||||||
@@ -83,3 +83,37 @@ spec:
|
|||||||
ports:
|
ports:
|
||||||
- containerPort: 6379
|
- containerPort: 6379
|
||||||
name: redis
|
name: redis
|
||||||
|
---
|
||||||
|
apiVersion: v1
|
||||||
|
kind: Service
|
||||||
|
metadata:
|
||||||
|
name: shynet-webserver-service
|
||||||
|
spec:
|
||||||
|
type: ClusterIP
|
||||||
|
ports:
|
||||||
|
- port: 8080
|
||||||
|
selector:
|
||||||
|
app: shynet-webserver
|
||||||
|
---
|
||||||
|
apiVersion: networking.k8s.io/v1beta1
|
||||||
|
kind: Ingress
|
||||||
|
metadata:
|
||||||
|
name: shynet-webserver-ingress
|
||||||
|
annotations:
|
||||||
|
kubernetes.io/ingress.class: addon-http-application-routing
|
||||||
|
spec:
|
||||||
|
rules:
|
||||||
|
- host: shynet.rmrm.io
|
||||||
|
http:
|
||||||
|
paths:
|
||||||
|
- backend:
|
||||||
|
serviceName: shynet-webserver-service
|
||||||
|
servicePort: 8080
|
||||||
|
path: /
|
||||||
|
- host: shynet-beta.rmrm.io
|
||||||
|
http:
|
||||||
|
paths:
|
||||||
|
- backend:
|
||||||
|
serviceName: shynet-webserver-service
|
||||||
|
servicePort: 8080
|
||||||
|
path: /
|
||||||
17
package-lock.json
generated
17
package-lock.json
generated
@@ -9,14 +9,14 @@
|
|||||||
"integrity": "sha512-OEdH7SyC1suTdhBGW91/zBfR6qaIhThbcN8PUXtXilY4GYnSBbVqOntdHbC1vXwsDnX0Qix2m2+DSU1J51ybOQ=="
|
"integrity": "sha512-OEdH7SyC1suTdhBGW91/zBfR6qaIhThbcN8PUXtXilY4GYnSBbVqOntdHbC1vXwsDnX0Qix2m2+DSU1J51ybOQ=="
|
||||||
},
|
},
|
||||||
"a17t": {
|
"a17t": {
|
||||||
"version": "0.2.9",
|
"version": "0.5.1",
|
||||||
"resolved": "https://registry.npmjs.org/a17t/-/a17t-0.2.9.tgz",
|
"resolved": "https://registry.npmjs.org/a17t/-/a17t-0.5.1.tgz",
|
||||||
"integrity": "sha512-PdkeIVJYTfKCrYkx64c8HEPvbiVo2Prx8NWMCsiXHbsvPLbai64FwydXnNSzq/hRBQ3Toi5qU8DNCxeX1AXBCw=="
|
"integrity": "sha512-peIPrH9eDiu49LLzLlSTFFrXj6WLlEX3TRsUkqyyOHi/i58ilJ/eERnu7AcswXhuCBx+/2W9EUuHM+8iAq4ipg=="
|
||||||
},
|
},
|
||||||
"apexcharts": {
|
"apexcharts": {
|
||||||
"version": "3.22.0",
|
"version": "3.24.0",
|
||||||
"resolved": "https://registry.npmjs.org/apexcharts/-/apexcharts-3.22.0.tgz",
|
"resolved": "https://registry.npmjs.org/apexcharts/-/apexcharts-3.24.0.tgz",
|
||||||
"integrity": "sha512-DDh2eXnAEA8GoKU/hdicOaS2jzGehXwv8Bj1djYYudkeQzEdglFoWsVyIxff+Ds7+aUtVAJzd/9ythZuyyIbXQ==",
|
"integrity": "sha512-iT6czJCIVrmAtrcO90MZTQCvC+xi6R6Acf0jNH/d40FVTtCfcqECuKIh5iAMyOTtgUb7+fQ8rbadH2bm1kbL9Q==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"svg.draggable.js": "^2.2.2",
|
"svg.draggable.js": "^2.2.2",
|
||||||
"svg.easing.js": "^2.0.0",
|
"svg.easing.js": "^2.0.0",
|
||||||
@@ -99,11 +99,6 @@
|
|||||||
"requires": {
|
"requires": {
|
||||||
"svg.js": "^2.6.5"
|
"svg.js": "^2.6.5"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"turbolinks": {
|
|
||||||
"version": "5.2.0",
|
|
||||||
"resolved": "https://registry.npmjs.org/turbolinks/-/turbolinks-5.2.0.tgz",
|
|
||||||
"integrity": "sha512-pMiez3tyBo6uRHFNNZoYMmrES/IaGgMhQQM+VFF36keryjb5ms0XkVpmKHkfW/4Vy96qiGW3K9bz0tF5sK9bBw=="
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,10 +18,9 @@
|
|||||||
"homepage": "https://github.com/milesmcc/shynet#readme",
|
"homepage": "https://github.com/milesmcc/shynet#readme",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fortawesome/fontawesome-free": "^5.15.1",
|
"@fortawesome/fontawesome-free": "^5.15.1",
|
||||||
"a17t": "^0.2.9",
|
"a17t": "^0.5.1",
|
||||||
"apexcharts": "^3.22.0",
|
"apexcharts": "^3.24.0",
|
||||||
"inter-ui": "^3.15.0",
|
"inter-ui": "^3.15.0",
|
||||||
"litepicker": "^1.5.7",
|
"litepicker": "^1.5.7"
|
||||||
"turbolinks": "^5.2.0"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<nav class="flex w-full flex-wrap items-center justify-between" role="navigation" aria-label="pagination">
|
<nav class="flex w-full flex-wrap items-center justify-between" role="navigation" aria-label="pagination">
|
||||||
<div class="w-full md:w-auto mb-2">
|
<div class="w-full md:w-auto mb-2">
|
||||||
{% if page.has_previous %}
|
{% if page.has_previous %}
|
||||||
<a href="?page={{ page.previous_page_number }}{{url_parameters}}" class="button field bg-neutral-000 w-auto mr-1">Previous</a>
|
<a href="?page={{ page.previous_page_number }}&{{url_parameters}}" class="button field bg-neutral-000 w-auto mr-1">Previous</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a class="button field bg-neutral-000 w-auto mr-1" disabled>Previous</a>
|
<a class="button field bg-neutral-000 w-auto mr-1" disabled>Previous</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if page.has_next %}
|
{% if page.has_next %}
|
||||||
<a href="?page={{ page.next_page_number }}{{url_parameters}}" class="button field bg-neutral-000 w-auto">Next</a>
|
<a href="?page={{ page.next_page_number }}&{{url_parameters}}" class="button field bg-neutral-000 w-auto">Next</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a class="button field bg-neutral-000 w-auto" disabled>Next</a>
|
<a class="button field bg-neutral-000 w-auto" disabled>Next</a>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
{% ifequal page.number pnum %}
|
{% ifequal page.number pnum %}
|
||||||
<li><a class="button field w-auto mx-1 text-white bg-neutral-700">{{ pnum }}</a></li>
|
<li><a class="button field w-auto mx-1 text-white bg-neutral-700">{{ pnum }}</a></li>
|
||||||
{% else %}
|
{% else %}
|
||||||
<li><a class="button field bg-neutral-000 w-auto mx-1" href="?page={{ pnum }}{{url_parameters}}">{{ pnum }}</a></li>
|
<li><a class="button field bg-neutral-000 w-auto mx-1" href="?page={{ pnum }}&{{url_parameters}}">{{ pnum }}</a></li>
|
||||||
{% endifequal %}
|
{% endifequal %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
@@ -27,7 +27,7 @@
|
|||||||
{% ifequal page.number pnum %}
|
{% ifequal page.number pnum %}
|
||||||
<li><a class="button field w-auto mx-1 text-white bg-neutral-700">{{ pnum }}</a></li>
|
<li><a class="button field w-auto mx-1 text-white bg-neutral-700">{{ pnum }}</a></li>
|
||||||
{% else %}
|
{% else %}
|
||||||
<li><a class="button field bg-neutral-000 w-auto mx-1" href="?page={{ pnum }}{{url_parameters}}">{{ pnum }}</a></li>
|
<li><a class="button field bg-neutral-000 w-auto mx-1" href="?page={{ pnum }}&{{url_parameters}}">{{ pnum }}</a></li>
|
||||||
{% endifequal %}
|
{% endifequal %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@@ -38,7 +38,7 @@
|
|||||||
{% ifequal page.number pnum %}
|
{% ifequal page.number pnum %}
|
||||||
<li><a class="button field w-auto mx-1 text-white bg-neutral-700">{{ pnum }}</a></li>
|
<li><a class="button field w-auto mx-1 text-white bg-neutral-700">{{ pnum }}</a></li>
|
||||||
{% else %}
|
{% else %}
|
||||||
<li><a class="button field bg-neutral-000 w-auto mx-1" href="?page={{ pnum }}{{url_parameters}}">{{ pnum }}</a></li>
|
<li><a class="button field bg-neutral-000 w-auto mx-1" href="?page={{ pnum }}&{{url_parameters}}">{{ pnum }}</a></li>
|
||||||
{% endifequal %}
|
{% endifequal %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|||||||
@@ -15,12 +15,8 @@ def pagination(
|
|||||||
before_current_pages=4,
|
before_current_pages=4,
|
||||||
after_current_pages=4,
|
after_current_pages=4,
|
||||||
):
|
):
|
||||||
url_parameters = "".join(
|
url_parameters = urlencode(
|
||||||
[
|
[(key, value) for key, value in request.GET.items() if key != "page"]
|
||||||
f"&{urlencode(key)}={urlencode(value)}"
|
|
||||||
for key, value in request.GET.items()
|
|
||||||
if key != "page"
|
|
||||||
]
|
|
||||||
)
|
)
|
||||||
|
|
||||||
before = max(page.number - before_current_pages - 1, 0)
|
before = max(page.number - before_current_pages - 1, 0)
|
||||||
|
|||||||
@@ -60,7 +60,9 @@ class Migration(migrations.Migration):
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
options={"ordering": ["-start_time"],},
|
options={
|
||||||
|
"ordering": ["-start_time"],
|
||||||
|
},
|
||||||
),
|
),
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name="Hit",
|
name="Hit",
|
||||||
@@ -90,7 +92,9 @@ class Migration(migrations.Migration):
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
options={"ordering": ["-start_time"],},
|
options={
|
||||||
|
"ordering": ["-start_time"],
|
||||||
|
},
|
||||||
),
|
),
|
||||||
migrations.AddIndex(
|
migrations.AddIndex(
|
||||||
model_name="session",
|
model_name="session",
|
||||||
|
|||||||
46
shynet/analytics/migrations/0004_auto_20210328_1514.py
Normal file
46
shynet/analytics/migrations/0004_auto_20210328_1514.py
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
# Generated by Django 3.1.7 on 2021-03-28 19:14
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.utils.timezone
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("analytics", "0003_auto_20200502_1227"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="hit",
|
||||||
|
name="last_seen",
|
||||||
|
field=models.DateTimeField(default=django.utils.timezone.now),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="hit",
|
||||||
|
name="start_time",
|
||||||
|
field=models.DateTimeField(
|
||||||
|
db_index=True, default=django.utils.timezone.now
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="session",
|
||||||
|
name="last_seen",
|
||||||
|
field=models.DateTimeField(
|
||||||
|
db_index=True, default=django.utils.timezone.now
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="session",
|
||||||
|
name="start_time",
|
||||||
|
field=models.DateTimeField(
|
||||||
|
db_index=True, default=django.utils.timezone.now
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddIndex(
|
||||||
|
model_name="session",
|
||||||
|
index=models.Index(
|
||||||
|
fields=["service", "-last_seen"], name="analytics_s_service_10bb96_idx"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
||||||
26
shynet/analytics/migrations/0005_auto_20210328_1518.py
Normal file
26
shynet/analytics/migrations/0005_auto_20210328_1518.py
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
# Generated by Django 3.1.7 on 2021-03-28 19:18
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.utils.timezone
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("analytics", "0004_auto_20210328_1514"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="hit",
|
||||||
|
name="last_seen",
|
||||||
|
field=models.DateTimeField(
|
||||||
|
db_index=True, default=django.utils.timezone.now
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="hit",
|
||||||
|
name="load_time",
|
||||||
|
field=models.FloatField(db_index=True, null=True),
|
||||||
|
),
|
||||||
|
]
|
||||||
40
shynet/analytics/migrations/0006_hit_service.py
Normal file
40
shynet/analytics/migrations/0006_hit_service.py
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
# Generated by Django 3.1.7 on 2021-03-28 19:36
|
||||||
|
|
||||||
|
from ..models import Hit, Session
|
||||||
|
from django.db import migrations, models
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db.models import Subquery, OuterRef
|
||||||
|
|
||||||
|
|
||||||
|
def add_service_to_hits(_a, _b):
|
||||||
|
service = Session.objects.filter(pk=OuterRef("session")).values_list("service")[:1]
|
||||||
|
|
||||||
|
Hit.objects.update(service=Subquery(service))
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("core", "0008_auto_20200628_1403"),
|
||||||
|
("analytics", "0005_auto_20210328_1518"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="hit",
|
||||||
|
name="service",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
null=True,
|
||||||
|
on_delete=django.db.models.deletion.CASCADE,
|
||||||
|
to="core.service",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.RunPython(add_service_to_hits, lambda: ()),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="hit",
|
||||||
|
name="service",
|
||||||
|
field=models.ForeignKey(
|
||||||
|
on_delete=django.db.models.deletion.CASCADE, to="core.service"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
||||||
19
shynet/analytics/migrations/0007_auto_20210328_1634.py
Normal file
19
shynet/analytics/migrations/0007_auto_20210328_1634.py
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# Generated by Django 3.1.7 on 2021-03-28 20:34
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("analytics", "0006_hit_service"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddIndex(
|
||||||
|
model_name="hit",
|
||||||
|
index=models.Index(
|
||||||
|
fields=["service", "-start_time"], name="analytics_h_service_f4f41e_idx"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
||||||
28
shynet/analytics/migrations/0008_session_is_bounce.py
Normal file
28
shynet/analytics/migrations/0008_session_is_bounce.py
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
# Generated by Django 3.1.7 on 2021-03-28 21:38
|
||||||
|
|
||||||
|
from django.db.models.expressions import F
|
||||||
|
from ..models import Session, Hit
|
||||||
|
from django.db import migrations, models
|
||||||
|
from django.db.models import Subquery, OuterRef
|
||||||
|
|
||||||
|
|
||||||
|
def update_bounce_stats(_a, _b):
|
||||||
|
Session.objects.all().annotate(hit_count=models.Count("hit")).filter(
|
||||||
|
hit_count__gt=1
|
||||||
|
).update(is_bounce=False)
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("analytics", "0007_auto_20210328_1634"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="session",
|
||||||
|
name="is_bounce",
|
||||||
|
field=models.BooleanField(db_index=True, default=True),
|
||||||
|
),
|
||||||
|
migrations.RunPython(update_bounce_stats, lambda: ()),
|
||||||
|
]
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import json
|
import json
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.shortcuts import reverse
|
from django.shortcuts import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
@@ -20,8 +21,8 @@ class Session(models.Model):
|
|||||||
identifier = models.TextField(blank=True, db_index=True)
|
identifier = models.TextField(blank=True, db_index=True)
|
||||||
|
|
||||||
# Time
|
# Time
|
||||||
start_time = models.DateTimeField(auto_now_add=True, db_index=True)
|
start_time = models.DateTimeField(default=timezone.now, db_index=True)
|
||||||
last_seen = models.DateTimeField(auto_now_add=True)
|
last_seen = models.DateTimeField(default=timezone.now, db_index=True)
|
||||||
|
|
||||||
# Core request information
|
# Core request information
|
||||||
user_agent = models.TextField()
|
user_agent = models.TextField()
|
||||||
@@ -48,16 +49,21 @@ class Session(models.Model):
|
|||||||
latitude = models.FloatField(null=True)
|
latitude = models.FloatField(null=True)
|
||||||
time_zone = models.TextField(blank=True)
|
time_zone = models.TextField(blank=True)
|
||||||
|
|
||||||
|
is_bounce = models.BooleanField(default=True, db_index=True)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ["-start_time"]
|
ordering = ["-start_time"]
|
||||||
indexes = [
|
indexes = [
|
||||||
models.Index(fields=["service", "-start_time"]),
|
models.Index(fields=["service", "-start_time"]),
|
||||||
|
models.Index(fields=["service", "-last_seen"]),
|
||||||
models.Index(fields=["service", "identifier"]),
|
models.Index(fields=["service", "identifier"]),
|
||||||
]
|
]
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_currently_active(self):
|
def is_currently_active(self):
|
||||||
return timezone.now() - self.last_seen < timezone.timedelta(seconds=10)
|
return timezone.now() - self.last_seen < timezone.timedelta(
|
||||||
|
milliseconds=settings.SCRIPT_HEARTBEAT_FREQUENCY * 2
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def duration(self):
|
def duration(self):
|
||||||
@@ -72,14 +78,20 @@ class Session(models.Model):
|
|||||||
kwargs={"pk": self.service.pk, "session_pk": self.uuid},
|
kwargs={"pk": self.service.pk, "session_pk": self.uuid},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def recalculate_bounce(self):
|
||||||
|
bounce = self.hit_set.count() == 1
|
||||||
|
if bounce != self.is_bounce:
|
||||||
|
self.is_bounce = bounce
|
||||||
|
self.save()
|
||||||
|
|
||||||
|
|
||||||
class Hit(models.Model):
|
class Hit(models.Model):
|
||||||
session = models.ForeignKey(Session, on_delete=models.CASCADE, db_index=True)
|
session = models.ForeignKey(Session, on_delete=models.CASCADE, db_index=True)
|
||||||
initial = models.BooleanField(default=True, db_index=True)
|
initial = models.BooleanField(default=True, db_index=True)
|
||||||
|
|
||||||
# Base request information
|
# Base request information
|
||||||
start_time = models.DateTimeField(auto_now_add=True, db_index=True)
|
start_time = models.DateTimeField(default=timezone.now, db_index=True)
|
||||||
last_seen = models.DateTimeField(auto_now_add=True)
|
last_seen = models.DateTimeField(default=timezone.now, db_index=True)
|
||||||
heartbeats = models.IntegerField(default=0)
|
heartbeats = models.IntegerField(default=0)
|
||||||
tracker = models.TextField(
|
tracker = models.TextField(
|
||||||
choices=[("JS", "JavaScript"), ("PIXEL", "Pixel (noscript)")]
|
choices=[("JS", "JavaScript"), ("PIXEL", "Pixel (noscript)")]
|
||||||
@@ -88,12 +100,17 @@ class Hit(models.Model):
|
|||||||
# Advanced page information
|
# Advanced page information
|
||||||
location = models.TextField(blank=True, db_index=True)
|
location = models.TextField(blank=True, db_index=True)
|
||||||
referrer = models.TextField(blank=True, db_index=True)
|
referrer = models.TextField(blank=True, db_index=True)
|
||||||
load_time = models.FloatField(null=True)
|
load_time = models.FloatField(null=True, db_index=True)
|
||||||
|
|
||||||
|
# While not necessary, we store the root service directly for performance.
|
||||||
|
# It makes querying much easier; no need for inner joins.
|
||||||
|
service = models.ForeignKey(Service, on_delete=models.CASCADE, db_index=True)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ["-start_time"]
|
ordering = ["-start_time"]
|
||||||
indexes = [
|
indexes = [
|
||||||
models.Index(fields=["session", "-start_time"]),
|
models.Index(fields=["session", "-start_time"]),
|
||||||
|
models.Index(fields=["service", "-start_time"]),
|
||||||
models.Index(fields=["session", "location"]),
|
models.Index(fields=["session", "location"]),
|
||||||
models.Index(fields=["session", "referrer"]),
|
models.Index(fields=["session", "referrer"]),
|
||||||
]
|
]
|
||||||
@@ -105,5 +122,5 @@ class Hit(models.Model):
|
|||||||
def get_absolute_url(self):
|
def get_absolute_url(self):
|
||||||
return reverse(
|
return reverse(
|
||||||
"dashboard:service_session",
|
"dashboard:service_session",
|
||||||
kwargs={"pk": self.session.service.pk, "session_pk": self.session.pk},
|
kwargs={"pk": self.service.pk, "session_pk": self.session.pk},
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import ipaddress
|
import ipaddress
|
||||||
import json
|
|
||||||
import logging
|
import logging
|
||||||
from hashlib import sha256
|
from hashlib import sha256
|
||||||
|
|
||||||
@@ -79,6 +78,11 @@ def ingress_request(
|
|||||||
association_id_hash = sha256()
|
association_id_hash = sha256()
|
||||||
association_id_hash.update(str(ip).encode("utf-8"))
|
association_id_hash.update(str(ip).encode("utf-8"))
|
||||||
association_id_hash.update(str(user_agent).encode("utf-8"))
|
association_id_hash.update(str(user_agent).encode("utf-8"))
|
||||||
|
if settings.AGGRESSIVE_HASH_SALTING:
|
||||||
|
association_id_hash.update(str(service.pk).encode("utf-8"))
|
||||||
|
association_id_hash.update(
|
||||||
|
str(timezone.now().date().isoformat()).encode("utf-8")
|
||||||
|
)
|
||||||
session_cache_path = (
|
session_cache_path = (
|
||||||
f"session_association_{service.pk}_{association_id_hash.hexdigest()}"
|
f"session_association_{service.pk}_{association_id_hash.hexdigest()}"
|
||||||
)
|
)
|
||||||
@@ -117,12 +121,14 @@ def ingress_request(
|
|||||||
return
|
return
|
||||||
session = Session.objects.create(
|
session = Session.objects.create(
|
||||||
service=service,
|
service=service,
|
||||||
ip=ip if service.collect_ips else None,
|
ip=ip if service.collect_ips and not settings.BLOCK_ALL_IPS else None,
|
||||||
user_agent=user_agent,
|
user_agent=user_agent,
|
||||||
identifier=identifier.strip(),
|
identifier=identifier.strip(),
|
||||||
browser=ua.browser.family or "",
|
browser=ua.browser.family or "",
|
||||||
device=ua.device.family or ua.device.model or "",
|
device=ua.device.family or ua.device.model or "",
|
||||||
device_type=device_type,
|
device_type=device_type,
|
||||||
|
start_time=time,
|
||||||
|
last_seen=time,
|
||||||
os=ua.os.family or "",
|
os=ua.os.family or "",
|
||||||
asn=ip_data.get("asn") or "",
|
asn=ip_data.get("asn") or "",
|
||||||
country=ip_data.get("country") or "",
|
country=ip_data.get("country") or "",
|
||||||
@@ -139,7 +145,7 @@ def ingress_request(
|
|||||||
log.debug("Updating old session with new data...")
|
log.debug("Updating old session with new data...")
|
||||||
|
|
||||||
# Update last seen time
|
# Update last seen time
|
||||||
session.last_seen = timezone.now()
|
session.last_seen = time
|
||||||
if session.identifier == "" and identifier.strip() != "":
|
if session.identifier == "" and identifier.strip() != "":
|
||||||
session.identifier = identifier.strip()
|
session.identifier = identifier.strip()
|
||||||
session.save()
|
session.save()
|
||||||
@@ -160,7 +166,7 @@ def ingress_request(
|
|||||||
# this is a heartbeat.
|
# this is a heartbeat.
|
||||||
log.debug("Hit is a heartbeat; updating old hit with new data...")
|
log.debug("Hit is a heartbeat; updating old hit with new data...")
|
||||||
hit.heartbeats += 1
|
hit.heartbeats += 1
|
||||||
hit.last_seen = timezone.now()
|
hit.last_seen = time
|
||||||
hit.save()
|
hit.save()
|
||||||
|
|
||||||
if hit is None:
|
if hit is None:
|
||||||
@@ -176,7 +182,14 @@ def ingress_request(
|
|||||||
location=payload.get("location", location),
|
location=payload.get("location", location),
|
||||||
referrer=payload.get("referrer", ""),
|
referrer=payload.get("referrer", ""),
|
||||||
load_time=payload.get("loadTime"),
|
load_time=payload.get("loadTime"),
|
||||||
|
start_time=time,
|
||||||
|
last_seen=time,
|
||||||
|
service=service,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Recalculate whether the session is a bounce
|
||||||
|
session.recalculate_bounce()
|
||||||
|
|
||||||
# Set idempotency (if applicable)
|
# Set idempotency (if applicable)
|
||||||
if idempotency is not None:
|
if idempotency is not None:
|
||||||
cache.set(
|
cache.set(
|
||||||
|
|||||||
@@ -5,7 +5,12 @@ from urllib.parse import urlparse
|
|||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.cache import cache
|
from django.core.cache import cache
|
||||||
from django.core.exceptions import ValidationError
|
from django.core.exceptions import ValidationError
|
||||||
from django.http import Http404, HttpResponse, HttpResponseBadRequest, HttpResponseForbidden
|
from django.http import (
|
||||||
|
Http404,
|
||||||
|
HttpResponse,
|
||||||
|
HttpResponseBadRequest,
|
||||||
|
HttpResponseForbidden,
|
||||||
|
)
|
||||||
from django.shortcuts import render, reverse
|
from django.shortcuts import render, reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
@@ -53,7 +58,10 @@ class ValidateServiceOriginsMixin:
|
|||||||
|
|
||||||
if origins != "*":
|
if origins != "*":
|
||||||
remote_origin = request.META.get("HTTP_ORIGIN")
|
remote_origin = request.META.get("HTTP_ORIGIN")
|
||||||
if remote_origin is None and request.META.get("HTTP_REFERER") is not None:
|
if (
|
||||||
|
remote_origin is None
|
||||||
|
and request.META.get("HTTP_REFERER") is not None
|
||||||
|
):
|
||||||
parsed = urlparse(request.META.get("HTTP_REFERER"))
|
parsed = urlparse(request.META.get("HTTP_REFERER"))
|
||||||
remote_origin = f"{parsed.scheme}://{parsed.netloc}".lower()
|
remote_origin = f"{parsed.scheme}://{parsed.netloc}".lower()
|
||||||
origins = [origin.strip().lower() for origin in origins.split(",")]
|
origins = [origin.strip().lower() for origin in origins.split(",")]
|
||||||
@@ -104,7 +112,9 @@ class ScriptView(ValidateServiceOriginsMixin, View):
|
|||||||
endpoint = (
|
endpoint = (
|
||||||
reverse(
|
reverse(
|
||||||
"ingress:endpoint_script",
|
"ingress:endpoint_script",
|
||||||
kwargs={"service_uuid": self.kwargs.get("service_uuid"),},
|
kwargs={
|
||||||
|
"service_uuid": self.kwargs.get("service_uuid"),
|
||||||
|
},
|
||||||
)
|
)
|
||||||
if self.kwargs.get("identifier") == None
|
if self.kwargs.get("identifier") == None
|
||||||
else reverse(
|
else reverse(
|
||||||
|
|||||||
117
shynet/core/management/commands/demo.py
Normal file
117
shynet/core/management/commands/demo.py
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
import traceback
|
||||||
|
from django.utils.timezone import now
|
||||||
|
from django.utils.timezone import timedelta
|
||||||
|
import random
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.contrib.sites.models import Site
|
||||||
|
from django.core.management.base import BaseCommand, CommandError
|
||||||
|
from django.utils.crypto import get_random_string
|
||||||
|
import user_agents
|
||||||
|
from logging import info
|
||||||
|
|
||||||
|
from core.models import User, Service
|
||||||
|
from analytics.models import Session, Hit
|
||||||
|
from analytics.tasks import ingress_request
|
||||||
|
|
||||||
|
LOCATIONS = [
|
||||||
|
"/",
|
||||||
|
"/post/{rand}",
|
||||||
|
"/login",
|
||||||
|
"/me",
|
||||||
|
]
|
||||||
|
|
||||||
|
REFERRERS = [
|
||||||
|
"https://news.ycombinator.com/item?id=11116274",
|
||||||
|
"https://news.ycombinator.com/item?id=24872911",
|
||||||
|
"https://reddit.com",
|
||||||
|
"https://facebook.com",
|
||||||
|
"https://twitter.com/milesmccain",
|
||||||
|
"https://twitter.com",
|
||||||
|
"https://stanford.edu/~mccain/",
|
||||||
|
"https://tiktok.com",
|
||||||
|
"https://io.stanford.edu",
|
||||||
|
"https://en.wikipedia.org",
|
||||||
|
"https://stackoverflow.com",
|
||||||
|
"",
|
||||||
|
"",
|
||||||
|
"",
|
||||||
|
"",
|
||||||
|
]
|
||||||
|
|
||||||
|
USER_AGENTS = [
|
||||||
|
"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.90 Safari/537.36",
|
||||||
|
"Mozilla/5.0 (Macintosh; Intel Mac OS X x.y; rv:42.0) Gecko/20100101 Firefox/43.4",
|
||||||
|
"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/77.0.3865.90 Safari/537.36",
|
||||||
|
"Mozilla/5.0 (iPhone; CPU iPhone OS 11_3_1 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko)",
|
||||||
|
"Version/10.0 Mobile/14E304 Safari/602.1",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class Command(BaseCommand):
|
||||||
|
help = "Configures a Shynet demo service"
|
||||||
|
|
||||||
|
def add_arguments(self, parser):
|
||||||
|
parser.add_argument(
|
||||||
|
"name",
|
||||||
|
type=str,
|
||||||
|
)
|
||||||
|
parser.add_argument("owner_email", type=str)
|
||||||
|
parser.add_argument(
|
||||||
|
"avg",
|
||||||
|
type=int,
|
||||||
|
)
|
||||||
|
parser.add_argument("deviation", type=float, default=0.4)
|
||||||
|
parser.add_argument(
|
||||||
|
"days",
|
||||||
|
type=int,
|
||||||
|
)
|
||||||
|
parser.add_argument("load_time", type=float, default=1000)
|
||||||
|
|
||||||
|
def handle(self, *args, **options):
|
||||||
|
owner = User.objects.get(email=options.get("owner_email"))
|
||||||
|
service = Service.objects.create(name=options.get("name"), owner=owner)
|
||||||
|
|
||||||
|
print(
|
||||||
|
f"Created demo service `{service.name}` (uuid: `{service.uuid}`, owner: {owner})"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Go through each day requested, creating sessions and hits
|
||||||
|
for days in range(options.get("days")):
|
||||||
|
day = (now() - timedelta(days=days)).replace(hour=0, minute=0, second=0)
|
||||||
|
print(f"Populating info for {day}...")
|
||||||
|
avg = options.get("avg")
|
||||||
|
deviation = options.get("deviation")
|
||||||
|
ips = [
|
||||||
|
".".join(map(str, (random.randint(0, 255) for _ in range(4))))
|
||||||
|
for _ in range(avg)
|
||||||
|
]
|
||||||
|
|
||||||
|
n = avg + random.randrange(-1 * deviation * avg, deviation * avg)
|
||||||
|
for _ in range(n):
|
||||||
|
time = day + timedelta(
|
||||||
|
hours=random.randrange(0, 23),
|
||||||
|
minutes=random.randrange(0, 59),
|
||||||
|
seconds=random.randrange(0, 59),
|
||||||
|
)
|
||||||
|
ip = random.choice(ips)
|
||||||
|
load_time = random.normalvariate(options.get("load_time"), 500)
|
||||||
|
referrer = random.choice(REFERRERS)
|
||||||
|
location = "https://example.com" + random.choice(LOCATIONS).replace(
|
||||||
|
"{rand}", str(random.randint(0, n))
|
||||||
|
)
|
||||||
|
user_agent = random.choice(USER_AGENTS)
|
||||||
|
ingress_request(
|
||||||
|
service.uuid,
|
||||||
|
"JS",
|
||||||
|
time,
|
||||||
|
{"loadTime": load_time, "referrer": referrer},
|
||||||
|
ip,
|
||||||
|
location,
|
||||||
|
user_agent,
|
||||||
|
)
|
||||||
|
|
||||||
|
print(f"Created {n} demo hits on {day}!")
|
||||||
|
|
||||||
|
self.stdout.write(self.style.SUCCESS(f"Successfully created demo data!"))
|
||||||
@@ -14,7 +14,8 @@ class Command(BaseCommand):
|
|||||||
|
|
||||||
def add_arguments(self, parser):
|
def add_arguments(self, parser):
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"hostname", type=str,
|
"hostname",
|
||||||
|
type=str,
|
||||||
)
|
)
|
||||||
|
|
||||||
def handle(self, *args, **options):
|
def handle(self, *args, **options):
|
||||||
|
|||||||
@@ -14,7 +14,8 @@ class Command(BaseCommand):
|
|||||||
|
|
||||||
def add_arguments(self, parser):
|
def add_arguments(self, parser):
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"email", type=str,
|
"email",
|
||||||
|
type=str,
|
||||||
)
|
)
|
||||||
|
|
||||||
def handle(self, *args, **options):
|
def handle(self, *args, **options):
|
||||||
|
|||||||
@@ -14,7 +14,8 @@ class Command(BaseCommand):
|
|||||||
|
|
||||||
def add_arguments(self, parser):
|
def add_arguments(self, parser):
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"name", type=str,
|
"name",
|
||||||
|
type=str,
|
||||||
)
|
)
|
||||||
|
|
||||||
def handle(self, *args, **options):
|
def handle(self, *args, **options):
|
||||||
|
|||||||
@@ -112,7 +112,9 @@ class Migration(migrations.Migration):
|
|||||||
"verbose_name_plural": "users",
|
"verbose_name_plural": "users",
|
||||||
"abstract": False,
|
"abstract": False,
|
||||||
},
|
},
|
||||||
managers=[("objects", django.contrib.auth.models.UserManager()),],
|
managers=[
|
||||||
|
("objects", django.contrib.auth.models.UserManager()),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name="Service",
|
name="Service",
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ class Migration(migrations.Migration):
|
|||||||
|
|
||||||
operations = [
|
operations = [
|
||||||
migrations.AlterModelOptions(
|
migrations.AlterModelOptions(
|
||||||
name="service", options={"ordering": ["name", "uuid"]},
|
name="service",
|
||||||
|
options={"ordering": ["name", "uuid"]},
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -6,18 +6,20 @@ from django.db import migrations, models
|
|||||||
class Migration(migrations.Migration):
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
dependencies = [
|
dependencies = [
|
||||||
('core', '0007_service_ignore_robots'),
|
("core", "0007_service_ignore_robots"),
|
||||||
]
|
]
|
||||||
|
|
||||||
operations = [
|
operations = [
|
||||||
migrations.AddField(
|
migrations.AddField(
|
||||||
model_name='service',
|
model_name="service",
|
||||||
name='script_inject',
|
name="script_inject",
|
||||||
field=models.TextField(blank=True, default=''),
|
field=models.TextField(blank=True, default=""),
|
||||||
),
|
),
|
||||||
migrations.AlterField(
|
migrations.AlterField(
|
||||||
model_name='user',
|
model_name="user",
|
||||||
name='first_name',
|
name="first_name",
|
||||||
field=models.CharField(blank=True, max_length=150, verbose_name='first name'),
|
field=models.CharField(
|
||||||
|
blank=True, max_length=150, verbose_name="first name"
|
||||||
|
),
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -129,11 +129,11 @@ class Service(models.Model):
|
|||||||
session_count = sessions.count()
|
session_count = sessions.count()
|
||||||
|
|
||||||
hits = Hit.objects.filter(
|
hits = Hit.objects.filter(
|
||||||
session__service=self, start_time__lt=end_time, start_time__gt=start_time
|
service=self, start_time__lt=end_time, start_time__gt=start_time
|
||||||
)
|
)
|
||||||
hit_count = hits.count()
|
hit_count = hits.count()
|
||||||
|
|
||||||
bounces = sessions.annotate(hit_count=models.Count("hit")).filter(hit_count=1)
|
bounces = sessions.filter(is_bounce=True)
|
||||||
bounce_count = bounces.count()
|
bounce_count = bounces.count()
|
||||||
|
|
||||||
locations = (
|
locations = (
|
||||||
@@ -244,4 +244,7 @@ class Service(models.Model):
|
|||||||
}
|
}
|
||||||
|
|
||||||
def get_absolute_url(self):
|
def get_absolute_url(self):
|
||||||
return reverse("dashboard:service", kwargs={"pk": self.pk},)
|
return reverse(
|
||||||
|
"dashboard:service",
|
||||||
|
kwargs={"pk": self.pk},
|
||||||
|
)
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
from allauth.account.admin import EmailAddress
|
from allauth.account.admin import EmailAddress
|
||||||
from django import forms
|
from django import forms
|
||||||
|
from django.conf import settings
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from core.models import Service, User
|
from core.models import Service, User
|
||||||
@@ -18,7 +19,7 @@ class ServiceForm(forms.ModelForm):
|
|||||||
"hide_referrer_regex",
|
"hide_referrer_regex",
|
||||||
"origins",
|
"origins",
|
||||||
"collaborators",
|
"collaborators",
|
||||||
"script_inject"
|
"script_inject",
|
||||||
]
|
]
|
||||||
widgets = {
|
widgets = {
|
||||||
"name": forms.TextInput(),
|
"name": forms.TextInput(),
|
||||||
@@ -28,12 +29,11 @@ class ServiceForm(forms.ModelForm):
|
|||||||
"collect_ips": forms.RadioSelect(choices=[(True, "Yes"), (False, "No")]),
|
"collect_ips": forms.RadioSelect(choices=[(True, "Yes"), (False, "No")]),
|
||||||
"ignore_robots": forms.RadioSelect(choices=[(True, "Yes"), (False, "No")]),
|
"ignore_robots": forms.RadioSelect(choices=[(True, "Yes"), (False, "No")]),
|
||||||
"hide_referrer_regex": forms.TextInput(),
|
"hide_referrer_regex": forms.TextInput(),
|
||||||
"script_inject": forms.Textarea(attrs={'class':'font-mono', 'rows': 5})
|
"script_inject": forms.Textarea(attrs={"class": "font-mono", "rows": 5}),
|
||||||
}
|
}
|
||||||
labels = {
|
labels = {
|
||||||
"origins": "Allowed origins",
|
"origins": "Allowed origins",
|
||||||
"respect_dnt": "Respect DNT",
|
"respect_dnt": "Respect DNT",
|
||||||
"collect_ips": "Collect IP addresses",
|
|
||||||
"ignored_ips": "Ignored IP addresses",
|
"ignored_ips": "Ignored IP addresses",
|
||||||
"ignore_robots": "Ignore robots",
|
"ignore_robots": "Ignore robots",
|
||||||
"hide_referrer_regex": "Hide specific referrers",
|
"hide_referrer_regex": "Hide specific referrers",
|
||||||
@@ -46,13 +46,27 @@ class ServiceForm(forms.ModelForm):
|
|||||||
"At what origins does the service operate? Use commas to separate multiple values. This sets CORS headers, so use '*' if you're not sure (or don't care)."
|
"At what origins does the service operate? Use commas to separate multiple values. This sets CORS headers, so use '*' if you're not sure (or don't care)."
|
||||||
),
|
),
|
||||||
"respect_dnt": "Should visitors who have enabled <a href='https://en.wikipedia.org/wiki/Do_Not_Track'>Do Not Track</a> be excluded from all data?",
|
"respect_dnt": "Should visitors who have enabled <a href='https://en.wikipedia.org/wiki/Do_Not_Track'>Do Not Track</a> be excluded from all data?",
|
||||||
"collect_ips": "Should individual IP addresses be collected? IP metadata (location, host, etc) will still be collected.",
|
|
||||||
"ignored_ips": "A comma-separated list of IP addresses or IP ranges (IPv4 and IPv6) to exclude from tracking (e.g., '192.168.0.2, 127.0.0.1/32').",
|
"ignored_ips": "A comma-separated list of IP addresses or IP ranges (IPv4 and IPv6) to exclude from tracking (e.g., '192.168.0.2, 127.0.0.1/32').",
|
||||||
"ignore_robots": "Should sessions generated by bots be excluded from tracking?",
|
"ignore_robots": "Should sessions generated by bots be excluded from tracking?",
|
||||||
"hide_referrer_regex": "Any referrers that match this <a href='https://regexr.com/'>RegEx</a> will not be listed in the referrer summary. Sessions will still be tracked normally. No effect if left blank.",
|
"hide_referrer_regex": "Any referrers that match this <a href='https://regexr.com/'>RegEx</a> will not be listed in the referrer summary. Sessions will still be tracked normally. No effect if left blank.",
|
||||||
"script_inject": "Optional additional JavaScript to inject at the end of the Shynet script. This code will be injected on every page where this service is installed.",
|
"script_inject": "Optional additional JavaScript to inject at the end of the Shynet script. This code will be injected on every page where this service is installed.",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
collect_ips = forms.BooleanField(
|
||||||
|
help_text="IP address collection is disabled globally by your administrator."
|
||||||
|
if settings.BLOCK_ALL_IPS
|
||||||
|
else "Should individual IP addresses be collected? IP metadata (location, host, etc) will still be collected.",
|
||||||
|
widget=forms.RadioSelect(choices=[(True, "Yes"), (False, "No")]),
|
||||||
|
initial=False if settings.BLOCK_ALL_IPS else True,
|
||||||
|
required=False,
|
||||||
|
disabled=settings.BLOCK_ALL_IPS,
|
||||||
|
)
|
||||||
|
|
||||||
|
def clean_collect_ips(self):
|
||||||
|
collect_ips = self.cleaned_data["collect_ips"]
|
||||||
|
# Forces collect IPs to be false if it is disabled globally
|
||||||
|
return False if settings.BLOCK_ALL_IPS else collect_ips
|
||||||
|
|
||||||
collaborators = forms.CharField(
|
collaborators = forms.CharField(
|
||||||
help_text="Which users on this Shynet instance should have read-only access to this service? (Comma separated list of emails.)",
|
help_text="Which users on this Shynet instance should have read-only access to this service? (Comma separated list of emails.)",
|
||||||
required=False,
|
required=False,
|
||||||
@@ -60,7 +74,9 @@ class ServiceForm(forms.ModelForm):
|
|||||||
|
|
||||||
def clean_collaborators(self):
|
def clean_collaborators(self):
|
||||||
collaborators = []
|
collaborators = []
|
||||||
users_to_emails = {} # maps users to the email they are listed under as a collaborator
|
users_to_emails = (
|
||||||
|
{}
|
||||||
|
) # maps users to the email they are listed under as a collaborator
|
||||||
for collaborator_email in self.cleaned_data["collaborators"].split(","):
|
for collaborator_email in self.cleaned_data["collaborators"].split(","):
|
||||||
email = collaborator_email.strip()
|
email = collaborator_email.strip()
|
||||||
if email == "":
|
if email == "":
|
||||||
@@ -72,7 +88,9 @@ class ServiceForm(forms.ModelForm):
|
|||||||
raise forms.ValidationError(f"Email '{email}' is not registered")
|
raise forms.ValidationError(f"Email '{email}' is not registered")
|
||||||
user = collaborator_email_linked.user
|
user = collaborator_email_linked.user
|
||||||
if user in collaborators:
|
if user in collaborators:
|
||||||
raise forms.ValidationError(f"The emails '{email}' and '{users_to_emails[user]}' both correspond to the same user")
|
raise forms.ValidationError(
|
||||||
|
f"The emails '{email}' and '{users_to_emails[user]}' both correspond to the same user"
|
||||||
|
)
|
||||||
users_to_emails[user] = email
|
users_to_emails[user] = email
|
||||||
collaborators.append(collaborator_email_linked.user)
|
collaborators.append(collaborator_email_linked.user)
|
||||||
return collaborators
|
return collaborators
|
||||||
|
|||||||
@@ -13,4 +13,18 @@
|
|||||||
|
|
||||||
.rf {
|
.rf {
|
||||||
text-align: right !important;
|
text-align: right !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--color-neutral-000: white;
|
||||||
|
--color-neutral-50: #F8FAFC;
|
||||||
|
--color-neutral-100: #F1F5F9;
|
||||||
|
--color-neutral-200: #E2E8F0;
|
||||||
|
--color-neutral-300: #CBD5E1;
|
||||||
|
--color-neutral-400: #94A3B8;
|
||||||
|
--color-neutral-500: #64748B;
|
||||||
|
--color-neutral-600: #475569;
|
||||||
|
--color-neutral-700: #334155;
|
||||||
|
--color-neutral-800: #1E293B;
|
||||||
|
--color-neutral-900: #0F172A;
|
||||||
}
|
}
|
||||||
BIN
shynet/dashboard/static/dashboard/images/icon.png
Normal file
BIN
shynet/dashboard/static/dashboard/images/icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.2 KiB |
@@ -8,20 +8,21 @@
|
|||||||
<meta name="robots" content="noindex">
|
<meta name="robots" content="noindex">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
{% include 'a17t/includes/head.html' %}
|
{% include 'a17t/includes/head.html' %}
|
||||||
|
<link rel="icon" type="image/png" href="{% static 'dashboard/images/icon.png' %}">
|
||||||
<script src="{% static 'apexcharts/dist/apexcharts.min.js'%}"></script>
|
<script src="{% static 'apexcharts/dist/apexcharts.min.js'%}"></script>
|
||||||
<script src="{% static 'litepicker/dist/js/main.js' %}"></script>
|
<script src="{% static 'litepicker/dist/js/main.js' %}"></script>
|
||||||
<script src="{% static 'turbolinks/dist/turbolinks.js' %}"></script>
|
|
||||||
<script src="{% static 'dashboard/js/base.js' %}"></script>
|
<script src="{% static 'dashboard/js/base.js' %}"></script>
|
||||||
<link rel="stylesheet" href="{% static 'dashboard/css/global.css' %}">
|
<link rel="stylesheet" href="{% static 'dashboard/css/global.css' %}">
|
||||||
{% block extra_head %}
|
{% block extra_head %}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body class="bg-neutral-200 min-h-full">
|
<body class="bg-neutral-100 min-h-full">
|
||||||
{% block body %}
|
{% block body %}
|
||||||
|
|
||||||
<section class="max-w-screen-xl mx-auto px-4 py-4 md:py-12 md:flex">
|
<section class="max-w-screen-xl mx-auto px-4 py-4 md:py-12 md:flex">
|
||||||
<aside class="mb-8 md:w-2/12 md:pr-6 relative flex flex-wrap md:block justify-between items-center overflow-x-hidden">
|
<aside
|
||||||
|
class="mb-8 md:w-2/12 md:pr-6 relative flex flex-wrap md:block justify-between items-center overflow-x-hidden">
|
||||||
<a class="icon ~urge ml-2 md:ml-6 md:mb-8 md:mt-3" href="{% url 'dashboard:dashboard' %}">
|
<a class="icon ~urge ml-2 md:ml-6 md:mb-8 md:mt-3" href="{% url 'dashboard:dashboard' %}">
|
||||||
<i class="fas fa-binoculars fa-3x text-urge-600 hidden md:block"></i>
|
<i class="fas fa-binoculars fa-3x text-urge-600 hidden md:block"></i>
|
||||||
<i class="fas fa-binoculars fa-2x text-urge-600 md:hidden"></i>
|
<i class="fas fa-binoculars fa-2x text-urge-600 md:hidden"></i>
|
||||||
@@ -41,7 +42,7 @@
|
|||||||
|
|
||||||
{% for service in user.owning_services.all %}
|
{% for service in user.owning_services.all %}
|
||||||
{% url 'dashboard:service' service.uuid as url %}
|
{% url 'dashboard:service' service.uuid as url %}
|
||||||
{% include 'dashboard/includes/sidebar_portal.html' with label=service.name|truncatechars:20 url=url %}
|
{% include 'dashboard/includes/sidebar_portal.html' with label=service.name|truncatechars:16 url=url icon=service.link|iconify %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|||||||
@@ -4,8 +4,9 @@
|
|||||||
{% with stats=object.stats %}
|
{% with stats=object.stats %}
|
||||||
<div class="p-4 md:flex justify-between">
|
<div class="p-4 md:flex justify-between">
|
||||||
<div class="flex items-center mb-4 md:mb-0">
|
<div class="flex items-center mb-4 md:mb-0">
|
||||||
<h3 class="heading text-xl md:text-2xl mr-2 mb-1 text-urge-600">
|
<h3 class="heading text-xl md:text-2xl mr-2 mb-1 text-urge-600 flex items-center">
|
||||||
{{object.name}}
|
{{object.link|iconify}}
|
||||||
|
<span>{{object.name}}</span>
|
||||||
</h3>
|
</h3>
|
||||||
{% include 'dashboard/includes/stats_status_chip.html' %}
|
{% include 'dashboard/includes/stats_status_chip.html' %}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{% load helpers %}
|
{% load helpers %}
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<a class="portal !low {% if request.get_full_path|startswith:url %}~urge active bg-neutral-100{% endif %}"
|
<a class="portal !low {% if request.get_full_path|startswith:url %}~urge active bg-neutral-100{% endif %} flex items-center"
|
||||||
{% if disable_turbolinks %}data-turbolinks="false"{% endif %} href="{{url}}">{{label}}</a>
|
{% if disable_turbolinks %}data-turbolinks="false"{% endif %} href="{{url}}">{{icon}} {{label}}</a>
|
||||||
</div>
|
</div>
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{% extends "base.html" %}
|
{% extends "base.html" %}
|
||||||
|
|
||||||
{% load rules %}
|
{% load rules pagination %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="md:flex justify-between items-center">
|
<div class="md:flex justify-between items-center">
|
||||||
@@ -18,9 +18,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<hr class="sep">
|
<hr class="sep">
|
||||||
{% for object in services|dictsortreversed:"stats.session_count" %}
|
{% for object in object_list|dictsortreversed:"stats.session_count" %}
|
||||||
{% include 'dashboard/includes/service_overview.html' %}
|
{% include 'dashboard/includes/service_overview.html' %}
|
||||||
{% empty %}
|
{% empty %}
|
||||||
<p>You don't have any services on {{request.site.name|default:"Shynet"}} yet.</p>
|
<p>You don't have any services on {{request.site.name|default:"Shynet"}} yet.</p>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
|
{% pagination page_obj request %}
|
||||||
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@@ -17,18 +17,18 @@
|
|||||||
<p class="label text-gray-400">Sessions</p>
|
<p class="label text-gray-400">Sessions</p>
|
||||||
<p class="heading">
|
<p class="heading">
|
||||||
{{stats.session_count|intcomma}}
|
{{stats.session_count|intcomma}}
|
||||||
<div>
|
<div>
|
||||||
{% compare stats.compare.session_count stats.session_count "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
{% compare stats.compare.session_count stats.session_count "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
||||||
</div>
|
</div>
|
||||||
</p>
|
</p>
|
||||||
</article>
|
</article>
|
||||||
<article class="">
|
<article class="">
|
||||||
<p class="label text-gray-400">Hits</p>
|
<p class="label text-gray-400">Hits</p>
|
||||||
<p class="heading">
|
<p class="heading">
|
||||||
{{stats.hit_count|intcomma}}
|
{{stats.hit_count|intcomma}}
|
||||||
<div>
|
<div>
|
||||||
{% compare stats.compare.hit_count stats.hit_count "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
{% compare stats.compare.hit_count stats.hit_count "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
||||||
</div>
|
</div>
|
||||||
</p>
|
</p>
|
||||||
</article>
|
</article>
|
||||||
<article class="">
|
<article class="">
|
||||||
@@ -39,9 +39,9 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
?
|
?
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div>
|
<div>
|
||||||
{% compare stats.compare.avg_load_time stats.avg_load_time "DOWN" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
{% compare stats.compare.avg_load_time stats.avg_load_time "DOWN" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
||||||
</div>
|
</div>
|
||||||
</p>
|
</p>
|
||||||
</article>
|
</article>
|
||||||
<article class="">
|
<article class="">
|
||||||
@@ -52,9 +52,9 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
?
|
?
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div>
|
<div>
|
||||||
{% compare stats.compare.bounce_rate_pct stats.bounce_rate_pct "DOWN" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
{% compare stats.compare.bounce_rate_pct stats.bounce_rate_pct "DOWN" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
||||||
</div>
|
</div>
|
||||||
</p>
|
</p>
|
||||||
</article>
|
</article>
|
||||||
<article class="">
|
<article class="">
|
||||||
@@ -65,9 +65,9 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
?
|
?
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div>
|
<div>
|
||||||
{% compare stats.compare.avg_session_duration stats.avg_session_duration "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
{% compare stats.compare.avg_session_duration stats.avg_session_duration "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
||||||
</div>
|
</div>
|
||||||
</p>
|
</p>
|
||||||
</article>
|
</article>
|
||||||
<article class="">
|
<article class="">
|
||||||
@@ -78,9 +78,9 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
?
|
?
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div>
|
<div>
|
||||||
{% compare stats.compare.avg_hits_per_session stats.avg_hits_per_session "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
{% compare stats.compare.avg_hits_per_session stats.avg_hits_per_session "UP" classes=classes good_classes=good_classes bad_classes=bad_classes neutral_classes=neutral_classes %}
|
||||||
</div>
|
</div>
|
||||||
</p>
|
</p>
|
||||||
</article>
|
</article>
|
||||||
{% endwith %}
|
{% endwith %}
|
||||||
@@ -166,7 +166,7 @@
|
|||||||
<tbody>
|
<tbody>
|
||||||
{% for os in stats.operating_systems %}
|
{% for os in stats.operating_systems %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{os.os|default:"Unknown"}}</td>
|
<td class="flex items-center">{{os.os|iconify}}<span>{{os.os|default:"Unknown"}}</span></td>
|
||||||
<td class="rf">{{os.count|intcomma}}</td>
|
<td class="rf">{{os.count|intcomma}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% empty %}
|
{% empty %}
|
||||||
@@ -188,7 +188,8 @@
|
|||||||
<tbody>
|
<tbody>
|
||||||
{% for browser in stats.browsers %}
|
{% for browser in stats.browsers %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{browser.browser|default:"Unknown"}}</td>
|
<td class="flex items-center">
|
||||||
|
{{browser.browser|iconify}}<span>{{browser.browser|default:"Unknown"}}</span></td>
|
||||||
<td class="rf">{{browser.count|intcomma}}</td>
|
<td class="rf">{{browser.count|intcomma}}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% empty %}
|
{% empty %}
|
||||||
@@ -225,7 +226,8 @@
|
|||||||
<div class="card ~neutral !low limited-height py-2">
|
<div class="card ~neutral !low limited-height py-2">
|
||||||
{% include 'dashboard/includes/session_list.html' %}
|
{% include 'dashboard/includes/session_list.html' %}
|
||||||
<hr class="sep h-8">
|
<hr class="sep h-8">
|
||||||
<a href="{% url 'dashboard:service_session_list' service.uuid %}" class="button ~neutral w-auto mb-2">View more sessions
|
<a href="{% url 'dashboard:service_session_list' service.uuid %}" class="button ~neutral w-auto mb-2">View more
|
||||||
|
sessions
|
||||||
→</a>
|
→</a>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@@ -15,7 +15,7 @@
|
|||||||
<div class="card ~neutral !high font-mono text-sm">
|
<div class="card ~neutral !high font-mono text-sm">
|
||||||
{% filter force_escape %}<noscript><img
|
{% filter force_escape %}<noscript><img
|
||||||
src="{{script_protocol}}{{request.site.domain}}{% url 'ingress:endpoint_pixel' object.uuid %}"></noscript>
|
src="{{script_protocol}}{{request.site.domain}}{% url 'ingress:endpoint_pixel' object.uuid %}"></noscript>
|
||||||
<script src="{{script_protocol}}{{request.site.domain}}{% url 'ingress:endpoint_script' object.uuid %}"></script>
|
<script defer src="{{script_protocol}}{{request.site.domain}}{% url 'ingress:endpoint_script' object.uuid %}"></script>
|
||||||
{% endfilter %}
|
{% endfilter %}
|
||||||
</div>
|
</div>
|
||||||
<hr class="sep h-4">
|
<hr class="sep h-4">
|
||||||
@@ -36,4 +36,4 @@
|
|||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|||||||
@@ -8,6 +8,7 @@
|
|||||||
<div class="md:flex justify-between items-center" id="heading">
|
<div class="md:flex justify-between items-center" id="heading">
|
||||||
<a class="flex items-center mb-4 md:mb-0" href="{% url 'dashboard:service' object.uuid %}">
|
<a class="flex items-center mb-4 md:mb-0" href="{% url 'dashboard:service' object.uuid %}">
|
||||||
<h3 class="heading leading-none mr-4">
|
<h3 class="heading leading-none mr-4">
|
||||||
|
{{object.link|iconify}}
|
||||||
{{object.name}}
|
{{object.name}}
|
||||||
</h3>
|
</h3>
|
||||||
<div class='text-3xl'>
|
<div class='text-3xl'>
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
from datetime import timedelta
|
||||||
from urllib.parse import urlparse
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
import flag
|
import flag
|
||||||
@@ -43,7 +44,12 @@ def country_name(isocode):
|
|||||||
|
|
||||||
@register.simple_tag
|
@register.simple_tag
|
||||||
def relative_stat_tone(
|
def relative_stat_tone(
|
||||||
start, end, good="UP", good_classes=None, bad_classes=None, neutral_classes=None,
|
start,
|
||||||
|
end,
|
||||||
|
good="UP",
|
||||||
|
good_classes=None,
|
||||||
|
bad_classes=None,
|
||||||
|
neutral_classes=None,
|
||||||
):
|
):
|
||||||
good_classes = good_classes or "~positive"
|
good_classes = good_classes or "~positive"
|
||||||
bad_classes = bad_classes or "~critical"
|
bad_classes = bad_classes or "~critical"
|
||||||
@@ -73,7 +79,7 @@ def percent_change_display(start, end):
|
|||||||
elif start == 0:
|
elif start == 0:
|
||||||
pct_change = "0%"
|
pct_change = "0%"
|
||||||
else:
|
else:
|
||||||
change = int(round(100 * abs(end - start) / start))
|
change = int(round(100 * abs(end - start) / max(start, 1)))
|
||||||
if change > 999:
|
if change > 999:
|
||||||
return "> 999%"
|
return "> 999%"
|
||||||
else:
|
else:
|
||||||
@@ -84,8 +90,7 @@ def percent_change_display(start, end):
|
|||||||
|
|
||||||
@register.inclusion_tag("dashboard/includes/sidebar_footer.html")
|
@register.inclusion_tag("dashboard/includes/sidebar_footer.html")
|
||||||
def sidebar_footer():
|
def sidebar_footer():
|
||||||
return {"version": settings.VERSION if settings.SHOW_SHYNET_VERSION
|
return {"version": settings.VERSION if settings.SHOW_SHYNET_VERSION else ""}
|
||||||
else ""}
|
|
||||||
|
|
||||||
|
|
||||||
@register.inclusion_tag("dashboard/includes/stat_comparison.html")
|
@register.inclusion_tag("dashboard/includes/stat_comparison.html")
|
||||||
@@ -98,6 +103,12 @@ def compare(
|
|||||||
bad_classes=None,
|
bad_classes=None,
|
||||||
neutral_classes=None,
|
neutral_classes=None,
|
||||||
):
|
):
|
||||||
|
if isinstance(start, timedelta):
|
||||||
|
start = start.seconds
|
||||||
|
|
||||||
|
if isinstance(end, timedelta):
|
||||||
|
end = end.seconds
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"start": start,
|
"start": start,
|
||||||
"end": end,
|
"end": end,
|
||||||
@@ -116,12 +127,60 @@ def startswith(text, starts):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
@register.filter
|
||||||
|
def iconify(text):
|
||||||
|
if not settings.SHOW_THIRD_PARTY_ICONS:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
text = text.lower()
|
||||||
|
icons = {
|
||||||
|
"chrome": "chrome.com",
|
||||||
|
"safari": "www.apple.com",
|
||||||
|
"windows": "windows.com",
|
||||||
|
"edge": "microsoft.com",
|
||||||
|
"firefox": "firefox.com",
|
||||||
|
"opera": "opera.com",
|
||||||
|
"unknown": "example.com",
|
||||||
|
"linux": "kernel.org",
|
||||||
|
"ios": "www.apple.com",
|
||||||
|
"mac": "www.apple.com",
|
||||||
|
"macos": "www.apple.com",
|
||||||
|
"mac os x": "www.apple.com",
|
||||||
|
"android": "android.com",
|
||||||
|
"chrome os": "chrome.com",
|
||||||
|
"ubuntu": "ubuntu.com",
|
||||||
|
"fedora": "getfedora.org",
|
||||||
|
"mobile safari": "www.apple.com",
|
||||||
|
"chrome mobile ios": "chrome.com",
|
||||||
|
"chrome mobile": "chrome.com",
|
||||||
|
"samsung internet": "samsung.com",
|
||||||
|
"google": "google.com",
|
||||||
|
"chrome mobile webview": "chrome.com",
|
||||||
|
"firefox mobile": "firefox.com",
|
||||||
|
"edge mobile": "microsoft.com",
|
||||||
|
"chromium": "chromium.org",
|
||||||
|
}
|
||||||
|
|
||||||
|
domain = None
|
||||||
|
if text.startswith("http"):
|
||||||
|
domain = urlparse(text).netloc
|
||||||
|
elif text in icons:
|
||||||
|
domain = icons[text]
|
||||||
|
else:
|
||||||
|
# This fallback works better than you'd think!
|
||||||
|
domain = text + ".com"
|
||||||
|
|
||||||
|
return SafeString(
|
||||||
|
f'<span class="icon mr-1"><img src="https://icons.duckduckgo.com/ip3/{domain}.ico"></span>'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@register.filter
|
@register.filter
|
||||||
def urldisplay(url):
|
def urldisplay(url):
|
||||||
if url.startswith("http"):
|
if url.startswith("http"):
|
||||||
display_url = url.replace("http://", "").replace("https://", "")
|
display_url = url.replace("http://", "").replace("https://", "")
|
||||||
return SafeString(
|
return SafeString(
|
||||||
f"<a href='{url}' title='{url}' rel='nofollow'>{escape(display_url if len(display_url) < 40 else display_url[:40] + '...')}</a>"
|
f"<a href='{url}' title='{url}' rel='nofollow' class='flex items-center'>{iconify(url)} {escape(display_url if len(display_url) < 40 else display_url[:40] + '...')}</a>"
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return url
|
return url
|
||||||
|
|||||||
@@ -22,16 +22,24 @@ from .forms import ServiceForm
|
|||||||
from .mixins import DateRangeMixin
|
from .mixins import DateRangeMixin
|
||||||
|
|
||||||
|
|
||||||
class DashboardView(LoginRequiredMixin, DateRangeMixin, TemplateView):
|
class DashboardView(LoginRequiredMixin, DateRangeMixin, ListView):
|
||||||
|
model = Service
|
||||||
template_name = "dashboard/pages/dashboard.html"
|
template_name = "dashboard/pages/dashboard.html"
|
||||||
|
paginate_by = 5
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
return Service.objects.filter(
|
||||||
|
Q(owner=self.request.user) | Q(collaborators__in=[self.request.user])
|
||||||
|
).distinct()
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
data = super().get_context_data(**kwargs)
|
data = super().get_context_data(**kwargs)
|
||||||
data["services"] = Service.objects.filter(
|
|
||||||
Q(owner=self.request.user) | Q(collaborators__in=[self.request.user])
|
for service in data["object_list"]:
|
||||||
).distinct()
|
service.stats = service.get_core_stats(
|
||||||
for service in data["services"]:
|
self.get_start_date(), self.get_end_date()
|
||||||
service.stats = service.get_core_stats(data["start_date"], data["end_date"])
|
)
|
||||||
|
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ import urllib.parse as urlparse
|
|||||||
from django.contrib.messages import constants as messages
|
from django.contrib.messages import constants as messages
|
||||||
|
|
||||||
# Increment on new releases
|
# Increment on new releases
|
||||||
VERSION = "v0.6.8"
|
VERSION = "v0.8.1"
|
||||||
|
|
||||||
# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
|
# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
|
||||||
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
||||||
@@ -58,6 +58,7 @@ INSTALLED_APPS = [
|
|||||||
"allauth",
|
"allauth",
|
||||||
"allauth.account",
|
"allauth.account",
|
||||||
"allauth.socialaccount",
|
"allauth.socialaccount",
|
||||||
|
"debug_toolbar",
|
||||||
]
|
]
|
||||||
|
|
||||||
MIDDLEWARE = [
|
MIDDLEWARE = [
|
||||||
@@ -70,6 +71,7 @@ MIDDLEWARE = [
|
|||||||
"django.contrib.sites.middleware.CurrentSiteMiddleware",
|
"django.contrib.sites.middleware.CurrentSiteMiddleware",
|
||||||
"django.contrib.messages.middleware.MessageMiddleware",
|
"django.contrib.messages.middleware.MessageMiddleware",
|
||||||
"django.middleware.clickjacking.XFrameOptionsMiddleware",
|
"django.middleware.clickjacking.XFrameOptionsMiddleware",
|
||||||
|
"debug_toolbar.middleware.DebugToolbarMiddleware",
|
||||||
]
|
]
|
||||||
|
|
||||||
ROOT_URLCONF = "shynet.urls"
|
ROOT_URLCONF = "shynet.urls"
|
||||||
@@ -145,9 +147,15 @@ AUTH_PASSWORD_VALIDATORS = [
|
|||||||
{
|
{
|
||||||
"NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
|
"NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
|
||||||
},
|
},
|
||||||
{"NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",},
|
{
|
||||||
{"NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",},
|
"NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",
|
||||||
{"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",},
|
},
|
||||||
|
{
|
||||||
|
"NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",
|
||||||
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
# Logging
|
# Logging
|
||||||
@@ -247,6 +255,10 @@ LOGIN_REDIRECT_URL = "/"
|
|||||||
|
|
||||||
SITE_ID = 1
|
SITE_ID = 1
|
||||||
|
|
||||||
|
INTERNAL_IPS = [
|
||||||
|
"127.0.0.1",
|
||||||
|
]
|
||||||
|
|
||||||
# Celery
|
# Celery
|
||||||
|
|
||||||
CELERY_TASK_ALWAYS_EAGER = os.getenv("CELERY_TASK_ALWAYS_EAGER", "True") == "True"
|
CELERY_TASK_ALWAYS_EAGER = os.getenv("CELERY_TASK_ALWAYS_EAGER", "True") == "True"
|
||||||
@@ -318,3 +330,12 @@ SESSION_MEMORY_TIMEOUT = int(os.getenv("SESSION_MEMORY_TIMEOUT", "1800"))
|
|||||||
|
|
||||||
# Should the Shynet version information be displayed?
|
# Should the Shynet version information be displayed?
|
||||||
SHOW_SHYNET_VERSION = os.getenv("SHOW_SHYNET_VERSION", "True") == "True"
|
SHOW_SHYNET_VERSION = os.getenv("SHOW_SHYNET_VERSION", "True") == "True"
|
||||||
|
|
||||||
|
# Should Shynet show third-party icons in the dashboard?
|
||||||
|
SHOW_THIRD_PARTY_ICONS = os.getenv("SHOW_THIRD_PARTY_ICONS", "True") == "True"
|
||||||
|
|
||||||
|
# Should Shynet never collect any IP?
|
||||||
|
BLOCK_ALL_IPS = os.getenv("BLOCK_ALL_IPS", "False") == "True"
|
||||||
|
|
||||||
|
# Include date and service ID in salt?
|
||||||
|
AGGRESSIVE_HASH_SALTING = os.getenv("AGGRESSIVE_HASH_SALTING", "False") == "True"
|
||||||
|
|||||||
@@ -15,8 +15,10 @@ Including another URLconf
|
|||||||
"""
|
"""
|
||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
from django.urls import include, path
|
from django.urls import include, path
|
||||||
|
import debug_toolbar
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
|
path("__debug__/", include(debug_toolbar.urls)),
|
||||||
path("admin/", admin.site.urls),
|
path("admin/", admin.site.urls),
|
||||||
path("accounts/", include("allauth.urls")),
|
path("accounts/", include("allauth.urls")),
|
||||||
path("ingress/", include(("analytics.ingress_urls", "ingress")), name="ingress"),
|
path("ingress/", include(("analytics.ingress_urls", "ingress")), name="ingress"),
|
||||||
|
|||||||
Reference in New Issue
Block a user