mirror of
https://github.com/simon987/sist2.git
synced 2025-12-12 15:08:53 +00:00
Compare commits
385 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 58741058cf | |||
| 0a7e59b646 | |||
| 43a566fe2f | |||
| b2631a86c8 | |||
| d0a1deca30 | |||
| b03ce90a05 | |||
| a5eacb4950 | |||
| 0887046b41 | |||
| 17fda1e540 | |||
| 34b363bfd8 | |||
| c9aa4bed72 | |||
| 7267d4bd2c | |||
| 43470e9ce6 | |||
| 0331d46fff | |||
| bbf1aca936 | |||
| 27560a82bb | |||
| f16ead1902 | |||
| e2e07e80c7 | |||
| 9499c6b189 | |||
| c5cd00b76c | |||
| ec5f07cab8 | |||
| f098f7916a | |||
| 85d67a9393 | |||
| c5ac89813f | |||
| ec5642a3df | |||
| c1de74e7eb | |||
| f31f138f2e | |||
| 6a48b219e6 | |||
| 97e56b664c | |||
| 0acdaab31a | |||
| d0290b886b | |||
| 4eea376869 | |||
| f4e1d90a6b | |||
| 391d8ed9d9 | |||
| 89fd68be8e | |||
| 9c0f3e0e31 | |||
| 648e4817c4 | |||
| bb23651087 | |||
| 78f3c897e2 | |||
| a2209e91ca | |||
| ba31531d3a | |||
| d5a47b8dab | |||
| d8c0b80524 | |||
| 142a4869e6 | |||
| ddb7f8d5d7 | |||
| dfb8c67490 | |||
| 3da2c8cae3 | |||
| 2f0e999b06 | |||
| bf28dc8993 | |||
| c6fee7f6e2 | |||
| 201c2a1a47 | |||
| 7c46ad632a | |||
| 5b8c13fd13 | |||
| efa4a06e56 | |||
| 81670ee107 | |||
| f9dac80905 | |||
| f8d9b718c0 | |||
| 6f5fdc2935 | |||
| a01f6dff1f | |||
| 22dd58e140 | |||
| f3e07fb7f7 | |||
| 7990e5cd2e | |||
| e3ca660983 | |||
| b87fb25458 | |||
| c7a77869ad | |||
| 523c123e2e | |||
| fc7f30d670 | |||
| 152fe11669 | |||
| 33f97f6bfb | |||
| 71f9dfcfe0 | |||
| 5f657d61b3 | |||
| 908def1016 | |||
| db3d312835 | |||
| 32c9cb28a3 | |||
| f839127129 | |||
| 8111a6c143 | |||
| 707a570828 | |||
|
|
5073b00225 | ||
|
|
4923d1b51f | ||
|
|
097e332015 | ||
|
|
d4babe216b | ||
|
|
44511a2202 | ||
| 50771bd1dc | |||
| bc884e137c | |||
| ce1e241dea | |||
| 5fe9c9efa3 | |||
| 75e4e93ddd | |||
| 013c54daa0 | |||
| 54308ef5e2 | |||
| 638c2a5c1a | |||
| 9587caddd9 | |||
| f5bbe0dc97 | |||
| f87eac1f90 | |||
| ddafbab6a6 | |||
| b91d574756 | |||
| 576140e542 | |||
| 050c1283a3 | |||
| c6e1ba03bc | |||
| 10e32f707f | |||
| 86e83bafaf | |||
| 51a40c8819 | |||
|
|
36281a5108 | ||
|
|
76a0bda48b | ||
| 0cf29a660c | |||
| 6cd0741848 | |||
| bc120f349d | |||
| 8cac8c98d7 | |||
| 30921ac52e | |||
| 95bbe39afc | |||
| 72ce217f9c | |||
| 641a8ec90c | |||
| 7a505c2287 | |||
| 12f162d760 | |||
| 4b4ab12fac | |||
| ae283f77ad | |||
| d3bd53a5ea | |||
| f7887f24d1 | |||
| 5c8de19188 | |||
| d861d278a4 | |||
| b6ddeee0e0 | |||
| 0cd2523b05 | |||
| 5e798f9367 | |||
| 5da6c1488b | |||
| 9568e25f84 | |||
| 6a8027789a | |||
| b1d16d8abf | |||
| b2a157e24d | |||
| 9aead9389a | |||
| a32c68cba8 | |||
| d116cf9d91 | |||
|
|
a020a8b32c | ||
| 5d5d9c3092 | |||
| 3379d5ce71 | |||
| a0ff4a1f01 | |||
| 4589f3bde7 | |||
| 1c898640cf | |||
| a0739d5177 | |||
| 8f9d29dbc6 | |||
| 3ff4b70223 | |||
| 02ad035b09 | |||
| c11feb213d | |||
| 72902947cd | |||
| a18bb81222 | |||
| 1520288f19 | |||
| e507de194b | |||
| 0e517d5e2b | |||
| 8223ef3860 | |||
| 995a196690 | |||
| 465d017e18 | |||
| ca994d3914 | |||
| db2285973f | |||
| 61de9e9f14 | |||
| 3015ef0ff4 | |||
| b55d432841 | |||
| ed90a140ce | |||
| 052df82373 | |||
| 5676136777 | |||
| c061613302 | |||
| d0325fd9b9 | |||
| e05a6f3863 | |||
| f1690a9cca | |||
| 100a264413 | |||
| 29390bb454 | |||
| 4d43036ded | |||
| 0b5cdbd130 | |||
| 53d7695f66 | |||
| 8d53456404 | |||
| cbc08a7cc9 | |||
| e629b4d7d3 | |||
| 22f7073b39 | |||
| 1781a74960 | |||
| db96c95ac7 | |||
| 7b9fa4cc0a | |||
| 5cc1fa86a9 | |||
| 649689ce30 | |||
| c8536f65a8 | |||
| 75b5e249c1 | |||
|
|
f49e03ac79 | ||
| a6d2afc8dc | |||
| 8f8f66ba05 | |||
| 1d9fcf7105 | |||
| 8127745f2b | |||
| 230988d6d1 | |||
| 13f4dbed2d | |||
| ed15e89f45 | |||
| c636d3d921 | |||
| 7e92d4b7d1 | |||
| 8ffe780ab2 | |||
| d3c8928fe8 | |||
| d9f628fca4 | |||
| 68289268c1 | |||
| 649c50c465 | |||
| 7b49a0dc49 | |||
| eb559b53aa | |||
| 6d01f9c0df | |||
| e724fec668 | |||
| fe5e93b300 | |||
| ecad85fd7d | |||
| 74cc898259 | |||
| dc2e4443c4 | |||
| 1a64431b52 | |||
|
|
9bad515e06 | ||
| 648559cedb | |||
| 3e6cd9cd5c | |||
| f249992798 | |||
|
|
e9645ecdaa | ||
| 046edea0e2 | |||
| a011b7e97b | |||
| 8c1c1697e0 | |||
| 018b49fa4c | |||
| 27b4e6403e | |||
| 13fdbd9e69 | |||
| 5e7fdaf8dd | |||
| 19d5c8ac9f | |||
| 99497049a8 | |||
|
|
1a3181d78b | ||
| 449aa77c8f | |||
| 3058c55510 | |||
| dedf9287b2 | |||
| ab199b0c0c | |||
| c4fbae123e | |||
| dd2397ef5c | |||
| ee0f71f4d3 | |||
| 0bbb96b149 | |||
| 78f6e16701 | |||
| 4625bca9a9 | |||
| f2ae653886 | |||
| 5686bc864d | |||
| cf513b4ad8 | |||
| 013423424e | |||
| 16514fd6b0 | |||
| 27509f97e1 | |||
| 4c540eae1c | |||
| d2b53ff6fc | |||
| 0ef4292abf | |||
| e6fde38c24 | |||
| 5fa343d40f | |||
| 7ee1374802 | |||
| bd9e56829c | |||
| 718169345e | |||
| 5a6aa763ca | |||
| 695d9abd83 | |||
| e436af7b2a | |||
| 4501a7810f | |||
|
|
e36761fa6a | ||
| fe53b79d56 | |||
| 09615bbed6 | |||
| a2be9b955c | |||
| 9298bd2d9d | |||
| 317034ba21 | |||
| 0505303503 | |||
| 6e5772f13b | |||
| ccccdb3b78 | |||
| 12d17acf4f | |||
| 48b56cdb7b | |||
| 048f707f80 | |||
| 98e0a5fd64 | |||
| 740a49a09f | |||
| 81be662574 | |||
| 02fa3f02f5 | |||
| cfdd7bdd87 | |||
| 7ceb645926 | |||
| 7d0091f647 | |||
| b3cd630399 | |||
| 5f7a1acfe3 | |||
| 513a21cca2 | |||
| 04dbfb23ab | |||
| 1abddabeec | |||
| 9ace5774af | |||
| eab6101cf7 | |||
| d7cbd5d2b6 | |||
| 641edf2715 | |||
| 7efb4957bf | |||
| 9ae77fdedb | |||
| 98c40901ed | |||
| 363375d5da | |||
| 149de95d88 | |||
| e5bb4856d2 | |||
| d78994d427 | |||
| f2d68d54df | |||
| e03625838b | |||
| 86840b46f4 | |||
| e57f9916eb | |||
| 565ba6ee76 | |||
| d83fc2c373 | |||
| d4da28249e | |||
| 483a454c8d | |||
| 018ac86640 | |||
| 398f1aead4 | |||
| d19a75926b | |||
| 1ac8b40e3d | |||
| a8505cb8c1 | |||
| ae8652d86e | |||
| 849beb09d8 | |||
| e1aaaee617 | |||
| c02b940945 | |||
| 2934ddb07f | |||
| 7f6f3c02fa | |||
| 7f98d5a682 | |||
| 7eb9c5d7d5 | |||
| 184439aa38 | |||
| 1ce8b298a1 | |||
| 75f99025d9 | |||
| ebe852bd5a | |||
| 402b103c49 | |||
| e9b6e1cdc2 | |||
| ed1ce8ab5e | |||
| d1fa4febc4 | |||
| 048c55df7b | |||
| f77bc6a025 | |||
| efdde2734e | |||
| 66658fa8f7 | |||
| df41c251e4 | |||
| 3282ab56ba | |||
| 8300838d30 | |||
| c9870a6d3d | |||
| a143cc4fcf | |||
| 9ef1f3781d | |||
| bbee8aa721 | |||
| d22f83c797 | |||
| 50615486a4 | |||
| ca79e4f797 | |||
| 6a9fd08a80 | |||
| cab890dc9b | |||
| b3c4faf2df | |||
| 353937171a | |||
| c80002bea4 | |||
| 56adee9d81 | |||
| d6493d6d5f | |||
| 0967e9676d | |||
| 487e998ea0 | |||
| 919f45c79c | |||
| d42129cfcb | |||
| 754983e34a | |||
| 7c8a3e2f9d | |||
| 3bb24b4453 | |||
| 9a56b959d3 | |||
| 5e3a2dbcc2 | |||
| 573f94f24e | |||
| f5db78a69f | |||
| 5a2820d339 | |||
| b7f13f425c | |||
| d1a2f9b1d5 | |||
| 71f17986db | |||
| acdd2fb3c1 | |||
| 0cda6c00e1 | |||
| 14d0e5a1e1 | |||
| 0d06d39281 | |||
| 80708ca636 | |||
|
|
43b7b40dc4 | ||
| d051f541e2 | |||
| 0eefbac7b4 | |||
| 663f8e21c1 | |||
| 80fbcb2a01 | |||
| 8451109ecd | |||
| d6fe61cfdc | |||
| 254094130f | |||
| eaaa75c04c | |||
| bb87f4270f | |||
| be23201210 | |||
| 9778acda77 | |||
| 8d187926d9 | |||
| 88c37e3523 | |||
| d816dae8b3 | |||
| 4346c3e063 | |||
| 1a1032a8a7 | |||
| 4ab2ba1a02 | |||
| d089601dc5 | |||
| 11df6cc88f | |||
| 373ac01e4e | |||
| 893ff145c5 | |||
| 6111ded77f | |||
| 34cc26b2fd | |||
| 204034d859 | |||
| 16ccc6c0d3 | |||
| 94c617fdc3 | |||
| ebfd7e03ce | |||
| 6931d320a2 | |||
| fc22e52eae | |||
| ba81748a74 | |||
| e72fa1587b | |||
| ea4fb7fa0d | |||
| b0a868bb73 | |||
| d761a3b595 | |||
| 2d7a8a2fdc |
30
.dockerignore
Normal file
30
.dockerignore
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
.idea/
|
||||||
|
*.sist2
|
||||||
|
docs/
|
||||||
|
test_i/
|
||||||
|
test_i_inc/
|
||||||
|
Testing/
|
||||||
|
.drone.yml
|
||||||
|
**/cmake_install.cmake
|
||||||
|
**/CMakeCache.txt
|
||||||
|
**/CMakeFiles/
|
||||||
|
LICENSE
|
||||||
|
Makefile
|
||||||
|
**/*.md
|
||||||
|
**/*.cbp
|
||||||
|
VERSION
|
||||||
|
**/node_modules/
|
||||||
|
.git/
|
||||||
|
sist2-*-linux-debug
|
||||||
|
sist2-*-linux
|
||||||
|
sist2_debug
|
||||||
|
sist2
|
||||||
|
**/libscan-test-files
|
||||||
|
**/scan_ub_test
|
||||||
|
**/scan_a_test
|
||||||
|
**/scan_test
|
||||||
|
**/ext_ffmpeg
|
||||||
|
**/ext_libmobi
|
||||||
|
**/ext_libwpd
|
||||||
|
**/core
|
||||||
|
*.a
|
||||||
88
.drone.yml
Normal file
88
.drone.yml
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
kind: pipeline
|
||||||
|
type: docker
|
||||||
|
name: amd64
|
||||||
|
|
||||||
|
platform:
|
||||||
|
os: linux
|
||||||
|
arch: amd64
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: build
|
||||||
|
image: simon987/sist2-build
|
||||||
|
commands:
|
||||||
|
- ./ci/build.sh
|
||||||
|
- name: docker
|
||||||
|
image: plugins/docker
|
||||||
|
settings:
|
||||||
|
username:
|
||||||
|
from_secret: DOCKER_USER
|
||||||
|
password:
|
||||||
|
from_secret: DOCKER_PASSWORD
|
||||||
|
repo: simon987/sist2
|
||||||
|
context: ./
|
||||||
|
dockerfile: ./Dockerfile
|
||||||
|
auto_tag: true
|
||||||
|
auto_tag_suffix: x64-linux
|
||||||
|
when:
|
||||||
|
event:
|
||||||
|
- tag
|
||||||
|
- name: scp files
|
||||||
|
image: appleboy/drone-scp
|
||||||
|
settings:
|
||||||
|
host:
|
||||||
|
from_secret: SSH_HOST
|
||||||
|
port:
|
||||||
|
from_secret: SSH_PORT
|
||||||
|
user:
|
||||||
|
from_secret: SSH_USER
|
||||||
|
key:
|
||||||
|
from_secret: SSH_KEY
|
||||||
|
target: /files/sist2/${DRONE_REPO_OWNER}_${DRONE_REPO_NAME}/${DRONE_BRANCH}_${DRONE_BUILD_NUMBER}_${DRONE_COMMIT}/
|
||||||
|
source:
|
||||||
|
- ./VERSION
|
||||||
|
- ./sist2-x64-linux
|
||||||
|
- ./sist2-x64-linux-debug
|
||||||
|
|
||||||
|
---
|
||||||
|
kind: pipeline
|
||||||
|
type: docker
|
||||||
|
name: arm64
|
||||||
|
|
||||||
|
platform:
|
||||||
|
arch: arm64
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: build
|
||||||
|
image: simon987/sist2-build-arm64
|
||||||
|
commands:
|
||||||
|
- ./ci/build_arm64.sh
|
||||||
|
- name: scp files
|
||||||
|
image: appleboy/drone-scp
|
||||||
|
settings:
|
||||||
|
host:
|
||||||
|
from_secret: SSH_HOST
|
||||||
|
port:
|
||||||
|
from_secret: SSH_PORT
|
||||||
|
user:
|
||||||
|
from_secret: SSH_USER
|
||||||
|
key:
|
||||||
|
from_secret: SSH_KEY
|
||||||
|
target: /files/sist2/${DRONE_REPO_OWNER}_${DRONE_REPO_NAME}/arm_${DRONE_BRANCH}_${DRONE_BUILD_NUMBER}_${DRONE_COMMIT}/
|
||||||
|
source:
|
||||||
|
- ./sist2-arm64-linux
|
||||||
|
- ./sist2-arm64-linux-debug
|
||||||
|
- name: docker
|
||||||
|
image: plugins/docker
|
||||||
|
settings:
|
||||||
|
username:
|
||||||
|
from_secret: DOCKER_USER
|
||||||
|
password:
|
||||||
|
from_secret: DOCKER_PASSWORD
|
||||||
|
repo: simon987/sist2
|
||||||
|
context: ./
|
||||||
|
dockerfile: ./Dockerfile.arm64
|
||||||
|
auto_tag: true
|
||||||
|
auto_tag_suffix: arm64-linux
|
||||||
|
when:
|
||||||
|
event:
|
||||||
|
- tag
|
||||||
4
.gitattributes
vendored
4
.gitattributes
vendored
@@ -1,3 +1,3 @@
|
|||||||
CMakeModules/* linguist-vendored
|
CMakeModules/* linguist-vendored
|
||||||
web/js/*.min.js linguist-vendored
|
**/*_generated.c linguist-vendored
|
||||||
web/css/*.min.css linguist-vendored
|
**/*_generated.h linguist-vendored
|
||||||
|
|||||||
40
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
40
.github/ISSUE_TEMPLATE/bug_report.md
vendored
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
---
|
||||||
|
name: "🐞 Bug Report"
|
||||||
|
about: Submit a bug report
|
||||||
|
title: ''
|
||||||
|
labels: bug
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Device Information (please complete the following information):**
|
||||||
|
- OS: `[e.g., Ubuntu 20.04, WSL2]`
|
||||||
|
- Deployment: `[Linux, Linux ARM64 or Docker]`
|
||||||
|
- Browser *(if relevant)*: `[e.g., chrome, safari]`
|
||||||
|
- SIST2 Version: `[e.g., v2.9.0]`
|
||||||
|
- Elasticsearch Version *(if relevant)* : ``
|
||||||
|
|
||||||
|
**Command with arguments**
|
||||||
|
<!-- `ex: "scan ~/Documents -o ./i2 --threads 3 -q 1.0` -->
|
||||||
|
|
||||||
|
**Describe the bug**
|
||||||
|
<!-- A clear and concise description of what the bug is. -->
|
||||||
|
|
||||||
|
**Steps To Reproduce**
|
||||||
|
Please be specific!
|
||||||
|
1. Go to '...'
|
||||||
|
2. Click on '....'
|
||||||
|
3. etc.
|
||||||
|
|
||||||
|
**Expected behavior**
|
||||||
|
<!-- A clear and concise description of what you expected to happen. -->
|
||||||
|
|
||||||
|
**Actual Behavior**
|
||||||
|
<!-- A clear and concise description of what actually happens. -->
|
||||||
|
|
||||||
|
**Screenshots**
|
||||||
|
<!-- If applicable, add screenshots to help explain your problem. -->
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
<!-- Add any other context about the problem here. If applicable, please include why you think the bug is occurring and/or troubleshooting you have already performed. -->
|
||||||
|
<!-- If the issue is related to the `scan` module, please attach the files necessary to reproduce the error or email them to me[at]simon987.net. -->
|
||||||
5
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
5
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
blank_issues_enabled: false
|
||||||
|
contact_links:
|
||||||
|
- name: SIST2 Documentation
|
||||||
|
url: https://github.com/simon987/sist2/blob/master/docs/USAGE.md
|
||||||
|
about: Check out the SIST2 documentation for answers to common questions
|
||||||
18
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
18
.github/ISSUE_TEMPLATE/feature_request.md
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
name: "🚀 Feature Request"
|
||||||
|
about: Suggest an idea for SIST2
|
||||||
|
title: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
**Which SIST2 component is your Feature Request related to?**
|
||||||
|
<!-- e.g., Scan, Index, or Web? -->
|
||||||
|
|
||||||
|
**Is your feature request related to a problem? Please describe.**
|
||||||
|
<!-- A clear and concise description of what the problem is. e.g., "I'm always frustrated when [...]" -->
|
||||||
|
|
||||||
|
**What would you like to see happen?**
|
||||||
|
<!-- A clear and concise description of what you want to happen. -->
|
||||||
|
|
||||||
|
**Additional context**
|
||||||
|
<!-- Add any other context or screenshots about the feature request here. -->
|
||||||
18
.github/ISSUE_TEMPLATE/issue-template.md
vendored
Normal file
18
.github/ISSUE_TEMPLATE/issue-template.md
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
name: Issue template
|
||||||
|
about: General
|
||||||
|
title: ''
|
||||||
|
labels: ''
|
||||||
|
assignees: ''
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
sist2 version:
|
||||||
|
|
||||||
|
Platform (Linux or Docker, x86-64 or arm64):
|
||||||
|
|
||||||
|
Elasticsearch version:
|
||||||
|
|
||||||
|
Command with arguments: `ex: "scan ~/Documents -o ./i2 --threads 3 -q 1.0`
|
||||||
|
|
||||||
|
If the issue is related to the `scan` module, please attach the files necessary to reproduce the error or email them to me[at]simon987.net.
|
||||||
11
.gitignore
vendored
11
.gitignore
vendored
@@ -1,6 +1,5 @@
|
|||||||
.idea
|
.idea
|
||||||
thumbs
|
thumbs
|
||||||
test
|
|
||||||
*.cbp
|
*.cbp
|
||||||
CMakeCache.txt
|
CMakeCache.txt
|
||||||
CMakeFiles
|
CMakeFiles
|
||||||
@@ -10,8 +9,18 @@ Makefile
|
|||||||
*.out
|
*.out
|
||||||
LOG
|
LOG
|
||||||
sist2*
|
sist2*
|
||||||
|
!sist2-vue/
|
||||||
index.sist2/
|
index.sist2/
|
||||||
bundle*.css
|
bundle*.css
|
||||||
bundle.js
|
bundle.js
|
||||||
*.a
|
*.a
|
||||||
vgcore.*
|
vgcore.*
|
||||||
|
build/
|
||||||
|
third-party/
|
||||||
|
*.idx/
|
||||||
|
VERSION
|
||||||
|
git_hash.h
|
||||||
|
Testing/
|
||||||
|
test_i
|
||||||
|
test_i_inc
|
||||||
|
node_modules/
|
||||||
31
.gitmodules
vendored
31
.gitmodules
vendored
@@ -1,27 +1,6 @@
|
|||||||
[submodule "argparse"]
|
[submodule "third-party/libscan"]
|
||||||
path = argparse
|
path = third-party/libscan
|
||||||
|
url = https://github.com/simon987/libscan
|
||||||
|
[submodule "third-party/argparse"]
|
||||||
|
path = third-party/argparse
|
||||||
url = https://github.com/cofyc/argparse
|
url = https://github.com/cofyc/argparse
|
||||||
[submodule "cJSON"]
|
|
||||||
path = cJSON
|
|
||||||
url = https://github.com/DaveGamble/cJSON
|
|
||||||
[submodule "lib/mupdf"]
|
|
||||||
path = lib/mupdf
|
|
||||||
url = git://git.ghostscript.com/mupdf.git
|
|
||||||
[submodule "lib/onion"]
|
|
||||||
path = lib/onion
|
|
||||||
url = https://github.com/davidmoreno/onion
|
|
||||||
[submodule "lib/ffmpeg"]
|
|
||||||
path = lib/ffmpeg
|
|
||||||
url = https://git.ffmpeg.org/ffmpeg.git
|
|
||||||
[submodule "lmdb"]
|
|
||||||
path = lmdb
|
|
||||||
url = https://github.com/LMDB/lmdb
|
|
||||||
[submodule "utf8.h"]
|
|
||||||
path = utf8.h
|
|
||||||
url = https://github.com/sheredom/utf8.h
|
|
||||||
[submodule "lib/openjpeg"]
|
|
||||||
path = lib/openjpeg
|
|
||||||
url = https://github.com/uclouvain/openjpeg
|
|
||||||
[submodule "lib/harfbuzz"]
|
|
||||||
path = lib/harfbuzz
|
|
||||||
url = https://github.com/harfbuzz/harfbuzz
|
|
||||||
|
|||||||
301
CMakeLists.txt
301
CMakeLists.txt
@@ -1,242 +1,135 @@
|
|||||||
cmake_minimum_required(VERSION 3.7)
|
cmake_minimum_required(VERSION 3.7)
|
||||||
set(CMAKE_C_STANDARD 11)
|
set(CMAKE_C_STANDARD 11)
|
||||||
|
|
||||||
option(WITH_SIST2 "Build main executable" ON)
|
|
||||||
option(WITH_SIST2_SCAN "Build scan executable" ON)
|
|
||||||
|
|
||||||
project(sist2 C)
|
project(sist2 C)
|
||||||
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_LIST_DIR}/CMakeModules")
|
|
||||||
|
|
||||||
if (WITH_SIST2)
|
option(SIST_DEBUG "Build a debug executable" on)
|
||||||
add_executable(
|
option(SIST_FAKE_STORE "Disable IO operations of LMDB stores for debugging purposes" 0)
|
||||||
sist2
|
|
||||||
src/main.c
|
|
||||||
src/sist.h
|
|
||||||
src/io/walk.h src/io/walk.c
|
|
||||||
src/parsing/media.h src/parsing/media.c
|
|
||||||
src/parsing/pdf.h src/parsing/pdf.c
|
|
||||||
src/io/store.h src/io/store.c
|
|
||||||
src/tpool.h src/tpool.c
|
|
||||||
src/parsing/parse.h src/parsing/parse.c
|
|
||||||
src/io/serialize.h src/io/serialize.c
|
|
||||||
src/parsing/mime.h src/parsing/mime.c src/parsing/mime_generated.c
|
|
||||||
src/parsing/text.h src/parsing/text.c
|
|
||||||
src/index/web.c src/index/web.h
|
|
||||||
src/web/serve.c src/web/serve.h
|
|
||||||
src/index/elastic.c src/index/elastic.h
|
|
||||||
src/util.c src/util.h
|
|
||||||
src/ctx.h src/types.h src/parsing/font.c src/parsing/font.h
|
|
||||||
|
|
||||||
# argparse
|
add_compile_definitions(
|
||||||
argparse/argparse.h argparse/argparse.c
|
"SIST_PLATFORM=${SIST_PLATFORM}"
|
||||||
|
)
|
||||||
|
|
||||||
# cJSON
|
if (SIST_DEBUG)
|
||||||
cJSON/cJSON.h cJSON/cJSON.c
|
add_compile_definitions(
|
||||||
|
"SIST_DEBUG=${SIST_DEBUG}"
|
||||||
# LMDB
|
|
||||||
lmdb/libraries/liblmdb/lmdb.h lmdb/libraries/liblmdb/mdb.c
|
|
||||||
lmdb/libraries/liblmdb/midl.h lmdb/libraries/liblmdb/midl.c
|
|
||||||
src/cli.c src/cli.h
|
|
||||||
|
|
||||||
# utf8.h
|
|
||||||
utf8.h/utf8.h
|
|
||||||
)
|
)
|
||||||
endif ()
|
endif()
|
||||||
|
|
||||||
if (WITH_SIST2_SCAN)
|
add_subdirectory(third-party/libscan)
|
||||||
add_executable(
|
set(ARGPARSE_SHARED off)
|
||||||
sist2_scan
|
add_subdirectory(third-party/argparse)
|
||||||
src/main.c
|
|
||||||
src/sist.h
|
|
||||||
src/io/walk.h src/io/walk.c
|
|
||||||
src/parsing/media.h src/parsing/media.c
|
|
||||||
src/parsing/pdf.h src/parsing/pdf.c
|
|
||||||
src/io/store.h src/io/store.c
|
|
||||||
src/tpool.h src/tpool.c
|
|
||||||
src/parsing/parse.h src/parsing/parse.c
|
|
||||||
src/io/serialize.h src/io/serialize.c
|
|
||||||
src/parsing/mime.h src/parsing/mime.c src/parsing/mime_generated.c
|
|
||||||
src/parsing/text.h src/parsing/text.c
|
|
||||||
src/util.c src/util.h
|
|
||||||
src/ctx.h src/types.h src/parsing/font.c src/parsing/font.h
|
|
||||||
|
|
||||||
# argparse
|
add_executable(sist2
|
||||||
argparse/argparse.h argparse/argparse.c
|
|
||||||
|
|
||||||
# cJSON
|
# argparse
|
||||||
cJSON/cJSON.h cJSON/cJSON.c
|
third-party/argparse/argparse.h third-party/argparse/argparse.c
|
||||||
|
|
||||||
# LMDB
|
src/main.c
|
||||||
lmdb/libraries/liblmdb/lmdb.h lmdb/libraries/liblmdb/mdb.c
|
src/sist.h
|
||||||
lmdb/libraries/liblmdb/midl.h lmdb/libraries/liblmdb/midl.c
|
src/io/walk.h src/io/walk.c
|
||||||
src/cli.c src/cli.h
|
src/io/store.h src/io/store.c
|
||||||
|
src/tpool.h src/tpool.c
|
||||||
|
src/parsing/parse.h src/parsing/parse.c
|
||||||
|
src/io/serialize.h src/io/serialize.c
|
||||||
|
src/parsing/mime.h src/parsing/mime.c src/parsing/mime_generated.c
|
||||||
|
src/index/web.c src/index/web.h
|
||||||
|
src/web/serve.c src/web/serve.h
|
||||||
|
src/index/elastic.c src/index/elastic.h
|
||||||
|
src/util.c src/util.h
|
||||||
|
src/ctx.h src/types.h
|
||||||
|
src/log.c src/log.h
|
||||||
|
src/cli.c src/cli.h
|
||||||
|
src/stats.c src/stats.h src/ctx.c
|
||||||
|
src/parsing/sidecar.c src/parsing/sidecar.h)
|
||||||
|
|
||||||
# utf8.h
|
target_link_directories(sist2 PRIVATE BEFORE ${_VCPKG_INSTALLED_DIR}/${VCPKG_TARGET_TRIPLET}/lib/)
|
||||||
utf8.h/utf8.h
|
set(CMAKE_FIND_LIBRARY_SUFFIXES .a .lib)
|
||||||
)
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
find_package(PkgConfig REQUIRED)
|
find_package(PkgConfig REQUIRED)
|
||||||
set(ENV{PKG_CONFIG_PATH} "$ENV{PKG_CONFIG_PATH}:/usr/local/lib/pkgconfig/")
|
|
||||||
|
|
||||||
find_package(LibMagic REQUIRED)
|
pkg_search_module(GLIB REQUIRED glib-2.0)
|
||||||
find_package(FFmpeg REQUIRED)
|
|
||||||
find_package(OpenSSL REQUIRED)
|
|
||||||
find_package(Freetype REQUIRED)
|
|
||||||
|
|
||||||
pkg_check_modules(GLIB REQUIRED glib-2.0)
|
find_package(lmdb CONFIG REQUIRED)
|
||||||
pkg_check_modules(GOBJECT REQUIRED gobject-2.0)
|
find_package(cJSON CONFIG REQUIRED)
|
||||||
pkg_check_modules(UUID REQUIRED uuid)
|
find_package(unofficial-mongoose CONFIG REQUIRED)
|
||||||
|
find_package(CURL CONFIG REQUIRED)
|
||||||
add_definitions(${LIBMAGIC_CFLAGS_OTHER})
|
|
||||||
add_definitions(${UUID_CFLAGS_OTHER})
|
|
||||||
add_definitions(${GLIB_CFLAGS_OTHER})
|
|
||||||
add_definitions(${GOBJECT_CFLAGS_OTHER})
|
|
||||||
add_definitions(${FREETYPE_CFLAGS_OTHER})
|
|
||||||
|
|
||||||
list(REMOVE_ITEM GLIB_LIBRARIES pcre)
|
|
||||||
list(REMOVE_ITEM GOBJECT_LIBRARIES pcre)
|
|
||||||
list(REMOVE_ITEM UUID_LIBRARIES pcre)
|
|
||||||
|
|
||||||
if (WITH_SIST2)
|
|
||||||
target_include_directories(
|
|
||||||
sist2 PUBLIC
|
|
||||||
${LIBMAGIC_INCLUDE_DIRS}
|
|
||||||
${GOBJECT_INCLUDE_DIRS}
|
|
||||||
${OPENSSL_INCLUDE_DIR}
|
|
||||||
${FFMPEG_INCLUDE_DIRS}
|
|
||||||
${GLIB_INCLUDE_DIRS}
|
|
||||||
${FREETYPE_INCLUDE_DIRS}
|
|
||||||
${UUID_INCLUDE_DIRS}
|
|
||||||
${PROJECT_SOURCE_DIR}/
|
|
||||||
${PROJECT_SOURCE_DIR}/lmdb/libraries/liblmdb/
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/onion/src/
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/mupdf/include/
|
|
||||||
)
|
|
||||||
target_link_directories(
|
|
||||||
sist2 PUBLIC
|
|
||||||
${UUID_LIBRARY_DIRS}
|
|
||||||
${FFMPEG_LIBRARY_DIRS}
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
target_compile_options(sist2
|
target_include_directories(
|
||||||
PRIVATE
|
sist2 PUBLIC
|
||||||
-Ofast
|
${CMAKE_SOURCE_DIR}/third-party/onion/src/
|
||||||
# -march=native
|
${CMAKE_SOURCE_DIR}/third-party/utf8.h/
|
||||||
-fno-stack-protector
|
${CMAKE_SOURCE_DIR}/third-party/libscan/
|
||||||
-fomit-frame-pointer
|
${CMAKE_SOURCE_DIR}/
|
||||||
)
|
${GLIB_INCLUDE_DIRS}
|
||||||
|
)
|
||||||
|
|
||||||
TARGET_LINK_LIBRARIES(
|
target_compile_options(
|
||||||
|
sist2
|
||||||
|
PRIVATE
|
||||||
|
-fPIC
|
||||||
|
)
|
||||||
|
|
||||||
|
if (SIST_DEBUG)
|
||||||
|
target_compile_options(
|
||||||
sist2
|
sist2
|
||||||
|
PRIVATE
|
||||||
${GLIB_LIBRARIES}
|
-g
|
||||||
${GOBJECT_LIBRARIES}
|
-fstack-protector
|
||||||
${UUID_LIBRARIES}
|
-fno-omit-frame-pointer
|
||||||
|
-fsanitize=address
|
||||||
# ffmpeg
|
-fno-inline
|
||||||
${PROJECT_SOURCE_DIR}/lib/libavcodec.a
|
# -O2
|
||||||
${PROJECT_SOURCE_DIR}/lib/libavformat.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libavutil.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libswscale.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libswresample.a
|
|
||||||
# ${FFMPEG_LIBRARIES}
|
|
||||||
# swscale
|
|
||||||
|
|
||||||
# mupdf
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libmupdf.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libmupdf-third.a
|
|
||||||
|
|
||||||
# onion
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libonion_static.a
|
|
||||||
|
|
||||||
pthread
|
|
||||||
curl
|
|
||||||
m
|
|
||||||
bz2
|
|
||||||
magic
|
|
||||||
harfbuzz
|
|
||||||
openjp2
|
|
||||||
freetype
|
|
||||||
)
|
)
|
||||||
|
target_link_options(
|
||||||
endif ()
|
sist2
|
||||||
|
PRIVATE
|
||||||
if (WITH_SIST2_SCAN)
|
-fsanitize=address
|
||||||
|
|
||||||
set_target_properties(
|
|
||||||
sist2_scan
|
|
||||||
PROPERTIES COMPILE_DEFINITIONS SIST_SCAN_ONLY
|
|
||||||
)
|
)
|
||||||
set_target_properties(
|
set_target_properties(
|
||||||
sist2_scan
|
sist2
|
||||||
PROPERTIES
|
PROPERTIES
|
||||||
COMPILE_DEFINITIONS SIST_SCAN_ONLY
|
OUTPUT_NAME sist2_debug
|
||||||
LINK_FLAGS -static
|
|
||||||
)
|
)
|
||||||
target_include_directories(
|
else ()
|
||||||
sist2_scan PUBLIC
|
target_compile_options(
|
||||||
${LIBMAGIC_INCLUDE_DIRS}
|
sist2
|
||||||
${GOBJECT_INCLUDE_DIRS}
|
|
||||||
${OPENSSL_INCLUDE_DIR}
|
|
||||||
${FFMPEG_INCLUDE_DIRS}
|
|
||||||
${GLIB_INCLUDE_DIRS}
|
|
||||||
${UUID_INCLUDE_DIRS}
|
|
||||||
${FREETYPE_INCLUDE_DIRS}
|
|
||||||
${PROJECT_SOURCE_DIR}/
|
|
||||||
${PROJECT_SOURCE_DIR}/lmdb/libraries/liblmdb/
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/onion/src/
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/mupdf/include/
|
|
||||||
)
|
|
||||||
target_link_directories(
|
|
||||||
sist2_scan PUBLIC
|
|
||||||
${UUID_LIBRARY_DIRS}
|
|
||||||
${FFMPEG_LIBRARY_DIRS}
|
|
||||||
)
|
|
||||||
target_compile_options(sist2_scan
|
|
||||||
PRIVATE
|
PRIVATE
|
||||||
-Ofast
|
-Ofast
|
||||||
# -march=native
|
|
||||||
-fno-stack-protector
|
-fno-stack-protector
|
||||||
-fomit-frame-pointer
|
-fomit-frame-pointer
|
||||||
)
|
|
||||||
TARGET_LINK_LIBRARIES(
|
|
||||||
sist2_scan
|
|
||||||
|
|
||||||
${GLIB_LIBRARIES}
|
|
||||||
${GOBJECT_LIBRARIES}
|
|
||||||
${UUID_LIBRARIES}
|
|
||||||
|
|
||||||
# ffmpeg
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libavcodec.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libavformat.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libavutil.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libswscale.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libswresample.a
|
|
||||||
|
|
||||||
# mupdf
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libmupdf.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libmupdf-third.a
|
|
||||||
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libbz2.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libmagic.a
|
|
||||||
|
|
||||||
pthread
|
|
||||||
m
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libharfbuzz.a
|
|
||||||
${PROJECT_SOURCE_DIR}/lib/libopenjp2.a
|
|
||||||
freetype
|
|
||||||
)
|
)
|
||||||
endif ()
|
endif ()
|
||||||
|
|
||||||
|
add_dependencies(
|
||||||
|
sist2
|
||||||
|
scan
|
||||||
|
argparse
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(
|
||||||
|
sist2
|
||||||
|
|
||||||
|
z
|
||||||
|
lmdb
|
||||||
|
cjson
|
||||||
|
argparse
|
||||||
|
${GLIB_LDFLAGS}
|
||||||
|
unofficial::mongoose::mongoose
|
||||||
|
CURL::libcurl
|
||||||
|
|
||||||
|
pthread
|
||||||
|
magic
|
||||||
|
|
||||||
|
c
|
||||||
|
|
||||||
|
scan
|
||||||
|
)
|
||||||
|
|
||||||
add_custom_target(
|
add_custom_target(
|
||||||
before_sist2
|
before_sist2
|
||||||
COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/scripts/before_build.sh
|
COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/scripts/before_build.sh
|
||||||
)
|
)
|
||||||
|
|
||||||
IF (WITH_SIST2)
|
add_dependencies(sist2 before_sist2)
|
||||||
add_dependencies(sist2 before_sist2)
|
|
||||||
else ()
|
|
||||||
add_dependencies(sist2_scan before_sist2)
|
|
||||||
endif ()
|
|
||||||
|
|||||||
80
CMakeModules/FindFFmpeg.cmake
vendored
80
CMakeModules/FindFFmpeg.cmake
vendored
@@ -1,80 +0,0 @@
|
|||||||
# - Try to find ffmpeg libraries (libavcodec, libavformat and libavutil)
|
|
||||||
# Once done this will define
|
|
||||||
#
|
|
||||||
# FFMPEG_FOUND - system has ffmpeg or libav
|
|
||||||
# FFMPEG_INCLUDE_DIR - the ffmpeg include directory
|
|
||||||
# FFMPEG_LIBRARIES - Link these to use ffmpeg
|
|
||||||
# FFMPEG_LIBAVCODEC
|
|
||||||
# FFMPEG_LIBAVFORMAT
|
|
||||||
# FFMPEG_LIBAVUTIL
|
|
||||||
#
|
|
||||||
# Copyright (c) 2008 Andreas Schneider <mail@cynapses.org>
|
|
||||||
# Modified for other libraries by Lasse Kärkkäinen <tronic>
|
|
||||||
# Modified for Hedgewars by Stepik777
|
|
||||||
#
|
|
||||||
# Redistribution and use is allowed according to the terms of the New
|
|
||||||
# BSD license.
|
|
||||||
#
|
|
||||||
|
|
||||||
if (FFMPEG_LIBRARIES AND FFMPEG_INCLUDE_DIR)
|
|
||||||
# in cache already
|
|
||||||
set(FFMPEG_FOUND TRUE)
|
|
||||||
else (FFMPEG_LIBRARIES AND FFMPEG_INCLUDE_DIR)
|
|
||||||
# use pkg-config to get the directories and then use these values
|
|
||||||
# in the FIND_PATH() and FIND_LIBRARY() calls
|
|
||||||
find_package(PkgConfig)
|
|
||||||
if (PKG_CONFIG_FOUND)
|
|
||||||
pkg_check_modules(_FFMPEG_AVCODEC libavcodec)
|
|
||||||
pkg_check_modules(_FFMPEG_AVFORMAT libavformat)
|
|
||||||
pkg_check_modules(_FFMPEG_AVUTIL libavutil)
|
|
||||||
endif (PKG_CONFIG_FOUND)
|
|
||||||
|
|
||||||
find_path(FFMPEG_AVCODEC_INCLUDE_DIR
|
|
||||||
NAMES libavcodec/avcodec.h
|
|
||||||
PATHS ${_FFMPEG_AVCODEC_INCLUDE_DIRS} /usr/include /usr/local/include /opt/local/include /sw/include
|
|
||||||
PATH_SUFFIXES ffmpeg libav
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(FFMPEG_LIBAVCODEC
|
|
||||||
NAMES avcodec
|
|
||||||
PATHS ${_FFMPEG_AVCODEC_LIBRARY_DIRS} /usr/lib /usr/local/lib /opt/local/lib /sw/lib
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(FFMPEG_LIBAVFORMAT
|
|
||||||
NAMES avformat
|
|
||||||
PATHS ${_FFMPEG_AVFORMAT_LIBRARY_DIRS} /usr/lib /usr/local/lib /opt/local/lib /sw/lib
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(FFMPEG_LIBAVUTIL
|
|
||||||
NAMES avutil
|
|
||||||
PATHS ${_FFMPEG_AVUTIL_LIBRARY_DIRS} /usr/lib /usr/local/lib /opt/local/lib /sw/lib
|
|
||||||
)
|
|
||||||
|
|
||||||
if (FFMPEG_LIBAVCODEC AND FFMPEG_LIBAVFORMAT)
|
|
||||||
set(FFMPEG_FOUND TRUE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (FFMPEG_FOUND)
|
|
||||||
set(FFMPEG_INCLUDE_DIR ${FFMPEG_AVCODEC_INCLUDE_DIR})
|
|
||||||
|
|
||||||
set(FFMPEG_LIBRARIES
|
|
||||||
${FFMPEG_LIBAVCODEC}
|
|
||||||
${FFMPEG_LIBAVFORMAT}
|
|
||||||
${FFMPEG_LIBAVUTIL}
|
|
||||||
)
|
|
||||||
|
|
||||||
endif (FFMPEG_FOUND)
|
|
||||||
|
|
||||||
if (FFMPEG_FOUND)
|
|
||||||
if (NOT FFMPEG_FIND_QUIETLY)
|
|
||||||
message(STATUS "Found FFMPEG or Libav: ${FFMPEG_LIBRARIES}, ${FFMPEG_INCLUDE_DIR}")
|
|
||||||
endif (NOT FFMPEG_FIND_QUIETLY)
|
|
||||||
else (FFMPEG_FOUND)
|
|
||||||
if (FFMPEG_FIND_REQUIRED)
|
|
||||||
message(FATAL_ERROR "Could not find libavcodec or libavformat or libavutil")
|
|
||||||
endif (FFMPEG_FIND_REQUIRED)
|
|
||||||
endif (FFMPEG_FOUND)
|
|
||||||
|
|
||||||
endif (FFMPEG_LIBRARIES AND FFMPEG_INCLUDE_DIR)
|
|
||||||
|
|
||||||
|
|
||||||
100
CMakeModules/FindLibMagic.cmake
vendored
100
CMakeModules/FindLibMagic.cmake
vendored
@@ -1,100 +0,0 @@
|
|||||||
|
|
||||||
#-------------------------------------------------------------------------------
|
|
||||||
# Copyright (c) 2013-2013, Lars Baehren <lbaehren@gmail.com>
|
|
||||||
# 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.
|
|
||||||
#
|
|
||||||
# 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.
|
|
||||||
#-------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
# - Check for the presence of LIBMAGIC
|
|
||||||
#
|
|
||||||
# The following variables are set when LIBMAGIC is found:
|
|
||||||
# LIBMAGIC_FOUND = Set to true, if all components of LIBMAGIC have been
|
|
||||||
# found.
|
|
||||||
# LIBMAGIC_INCLUDES = Include path for the header files of LIBMAGIC
|
|
||||||
# LIBMAGIC_LIBRARIES = Link these to use LIBMAGIC
|
|
||||||
# LIBMAGIC_LFLAGS = Linker flags (optional)
|
|
||||||
|
|
||||||
if (NOT LIBMAGIC_FOUND)
|
|
||||||
|
|
||||||
if (NOT LIBMAGIC_ROOT_DIR)
|
|
||||||
set (LIBMAGIC_ROOT_DIR ${CMAKE_INSTALL_PREFIX})
|
|
||||||
endif (NOT LIBMAGIC_ROOT_DIR)
|
|
||||||
|
|
||||||
##____________________________________________________________________________
|
|
||||||
## Check for the header files
|
|
||||||
|
|
||||||
find_path (LIBMAGIC_FILE_H
|
|
||||||
NAMES file/file.h
|
|
||||||
HINTS ${LIBMAGIC_ROOT_DIR} ${CMAKE_INSTALL_PREFIX}
|
|
||||||
PATH_SUFFIXES include
|
|
||||||
)
|
|
||||||
if (LIBMAGIC_FILE_H)
|
|
||||||
list (APPEND LIBMAGIC_INCLUDES ${LIBMAGIC_FILE_H})
|
|
||||||
endif (LIBMAGIC_FILE_H)
|
|
||||||
|
|
||||||
find_path (LIBMAGIC_MAGIC_H
|
|
||||||
NAMES magic.h
|
|
||||||
HINTS ${LIBMAGIC_ROOT_DIR} ${CMAKE_INSTALL_PREFIX}
|
|
||||||
PATH_SUFFIXES include include/linux
|
|
||||||
)
|
|
||||||
if (LIBMAGIC_MAGIC_H)
|
|
||||||
list (APPEND LIBMAGIC_INCLUDES ${LIBMAGIC_MAGIC_H})
|
|
||||||
endif (LIBMAGIC_MAGIC_H)
|
|
||||||
|
|
||||||
list (REMOVE_DUPLICATES LIBMAGIC_INCLUDES)
|
|
||||||
|
|
||||||
##____________________________________________________________________________
|
|
||||||
## Check for the library
|
|
||||||
|
|
||||||
find_library (LIBMAGIC_LIBRARIES magic
|
|
||||||
HINTS ${LIBMAGIC_ROOT_DIR} ${CMAKE_INSTALL_PREFIX}
|
|
||||||
PATH_SUFFIXES lib
|
|
||||||
)
|
|
||||||
|
|
||||||
##____________________________________________________________________________
|
|
||||||
## Actions taken when all components have been found
|
|
||||||
|
|
||||||
#find_package_handle_standard_args (LIBMAGIC DEFAULT_MSG LIBMAGIC_LIBRARIES LIBMAGIC_INCLUDES)
|
|
||||||
|
|
||||||
if (LIBMAGIC_FOUND)
|
|
||||||
if (NOT LIBMAGIC_FIND_QUIETLY)
|
|
||||||
message (STATUS "Found components for LIBMAGIC")
|
|
||||||
message (STATUS "LIBMAGIC_ROOT_DIR = ${LIBMAGIC_ROOT_DIR}")
|
|
||||||
message (STATUS "LIBMAGIC_INCLUDES = ${LIBMAGIC_INCLUDES}")
|
|
||||||
message (STATUS "LIBMAGIC_LIBRARIES = ${LIBMAGIC_LIBRARIES}")
|
|
||||||
endif (NOT LIBMAGIC_FIND_QUIETLY)
|
|
||||||
else (LIBMAGIC_FOUND)
|
|
||||||
if (LIBMAGIC_FIND_REQUIRED)
|
|
||||||
message (FATAL_ERROR "Could not find LIBMAGIC!")
|
|
||||||
endif (LIBMAGIC_FIND_REQUIRED)
|
|
||||||
endif (LIBMAGIC_FOUND)
|
|
||||||
|
|
||||||
##____________________________________________________________________________
|
|
||||||
## Mark advanced variables
|
|
||||||
|
|
||||||
mark_as_advanced (
|
|
||||||
LIBMAGIC_ROOT_DIR
|
|
||||||
LIBMAGIC_INCLUDES
|
|
||||||
LIBMAGIC_LIBRARIES
|
|
||||||
)
|
|
||||||
|
|
||||||
endif (NOT LIBMAGIC_FOUND)
|
|
||||||
478
CMakeModules/FindOpenSSL.cmake
vendored
478
CMakeModules/FindOpenSSL.cmake
vendored
@@ -1,478 +0,0 @@
|
|||||||
# Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
||||||
# file Copyright.txt or https://cmake.org/licensing for details.
|
|
||||||
|
|
||||||
macro(_OpenSSL_test_and_find_dependencies ssl_library crypto_library)
|
|
||||||
if((CMAKE_SYSTEM_NAME STREQUAL "Linux") AND
|
|
||||||
(("${ssl_library}" MATCHES "\\${CMAKE_STATIC_LIBRARY_SUFFIX}$") OR
|
|
||||||
("${crypto_library}" MATCHES "\\${CMAKE_STATIC_LIBRARY_SUFFIX}$")))
|
|
||||||
set(_OpenSSL_has_dependencies TRUE)
|
|
||||||
find_package(Threads)
|
|
||||||
else()
|
|
||||||
set(_OpenSSL_has_dependencies FALSE)
|
|
||||||
endif()
|
|
||||||
endmacro()
|
|
||||||
|
|
||||||
function(_OpenSSL_add_dependencies libraries_var library)
|
|
||||||
if(CMAKE_THREAD_LIBS_INIT)
|
|
||||||
list(APPEND ${libraries_var} ${CMAKE_THREAD_LIBS_INIT})
|
|
||||||
endif()
|
|
||||||
list(APPEND ${libraries_var} ${CMAKE_DL_LIBS})
|
|
||||||
set(${libraries_var} ${${libraries_var}} PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
function(_OpenSSL_target_add_dependencies target)
|
|
||||||
if(_OpenSSL_has_dependencies)
|
|
||||||
set_property( TARGET ${target} APPEND PROPERTY INTERFACE_LINK_LIBRARIES Threads::Threads )
|
|
||||||
set_property( TARGET ${target} APPEND PROPERTY INTERFACE_LINK_LIBRARIES ${CMAKE_DL_LIBS} )
|
|
||||||
endif()
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
if (UNIX)
|
|
||||||
find_package(PkgConfig QUIET)
|
|
||||||
pkg_check_modules(_OPENSSL QUIET openssl)
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
# Support preference of static libs by adjusting CMAKE_FIND_LIBRARY_SUFFIXES
|
|
||||||
if(OPENSSL_USE_STATIC_LIBS)
|
|
||||||
set(_openssl_ORIG_CMAKE_FIND_LIBRARY_SUFFIXES ${CMAKE_FIND_LIBRARY_SUFFIXES})
|
|
||||||
if(WIN32)
|
|
||||||
set(CMAKE_FIND_LIBRARY_SUFFIXES .lib .a ${CMAKE_FIND_LIBRARY_SUFFIXES})
|
|
||||||
else()
|
|
||||||
set(CMAKE_FIND_LIBRARY_SUFFIXES .a )
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (WIN32)
|
|
||||||
# http://www.slproweb.com/products/Win32OpenSSL.html
|
|
||||||
set(_OPENSSL_ROOT_HINTS
|
|
||||||
${OPENSSL_ROOT_DIR}
|
|
||||||
"[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\OpenSSL (32-bit)_is1;Inno Setup: App Path]"
|
|
||||||
"[HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\OpenSSL (64-bit)_is1;Inno Setup: App Path]"
|
|
||||||
ENV OPENSSL_ROOT_DIR
|
|
||||||
)
|
|
||||||
file(TO_CMAKE_PATH "$ENV{PROGRAMFILES}" _programfiles)
|
|
||||||
set(_OPENSSL_ROOT_PATHS
|
|
||||||
"${_programfiles}/OpenSSL"
|
|
||||||
"${_programfiles}/OpenSSL-Win32"
|
|
||||||
"${_programfiles}/OpenSSL-Win64"
|
|
||||||
"C:/OpenSSL/"
|
|
||||||
"C:/OpenSSL-Win32/"
|
|
||||||
"C:/OpenSSL-Win64/"
|
|
||||||
)
|
|
||||||
unset(_programfiles)
|
|
||||||
else ()
|
|
||||||
set(_OPENSSL_ROOT_HINTS
|
|
||||||
${OPENSSL_ROOT_DIR}
|
|
||||||
ENV OPENSSL_ROOT_DIR
|
|
||||||
)
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
set(_OPENSSL_ROOT_HINTS_AND_PATHS
|
|
||||||
HINTS ${_OPENSSL_ROOT_HINTS}
|
|
||||||
PATHS ${_OPENSSL_ROOT_PATHS}
|
|
||||||
)
|
|
||||||
|
|
||||||
find_path(OPENSSL_INCLUDE_DIR
|
|
||||||
NAMES
|
|
||||||
openssl/ssl.h
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
HINTS
|
|
||||||
${_OPENSSL_INCLUDEDIR}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
include
|
|
||||||
)
|
|
||||||
|
|
||||||
if(WIN32 AND NOT CYGWIN)
|
|
||||||
if(MSVC)
|
|
||||||
# /MD and /MDd are the standard values - if someone wants to use
|
|
||||||
# others, the libnames have to change here too
|
|
||||||
# use also ssl and ssleay32 in debug as fallback for openssl < 0.9.8b
|
|
||||||
# enable OPENSSL_MSVC_STATIC_RT to get the libs build /MT (Multithreaded no-DLL)
|
|
||||||
# In Visual C++ naming convention each of these four kinds of Windows libraries has it's standard suffix:
|
|
||||||
# * MD for dynamic-release
|
|
||||||
# * MDd for dynamic-debug
|
|
||||||
# * MT for static-release
|
|
||||||
# * MTd for static-debug
|
|
||||||
|
|
||||||
# Implementation details:
|
|
||||||
# We are using the libraries located in the VC subdir instead of the parent directory even though :
|
|
||||||
# libeay32MD.lib is identical to ../libeay32.lib, and
|
|
||||||
# ssleay32MD.lib is identical to ../ssleay32.lib
|
|
||||||
# enable OPENSSL_USE_STATIC_LIBS to use the static libs located in lib/VC/static
|
|
||||||
|
|
||||||
if (OPENSSL_MSVC_STATIC_RT)
|
|
||||||
set(_OPENSSL_MSVC_RT_MODE "MT")
|
|
||||||
else ()
|
|
||||||
set(_OPENSSL_MSVC_RT_MODE "MD")
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
# Since OpenSSL 1.1, lib names are like libcrypto32MTd.lib and libssl32MTd.lib
|
|
||||||
if( "${CMAKE_SIZEOF_VOID_P}" STREQUAL "8" )
|
|
||||||
set(_OPENSSL_MSVC_ARCH_SUFFIX "64")
|
|
||||||
else()
|
|
||||||
set(_OPENSSL_MSVC_ARCH_SUFFIX "32")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(OPENSSL_USE_STATIC_LIBS)
|
|
||||||
set(_OPENSSL_PATH_SUFFIXES
|
|
||||||
"lib/VC/static"
|
|
||||||
"VC/static"
|
|
||||||
"lib"
|
|
||||||
)
|
|
||||||
else()
|
|
||||||
set(_OPENSSL_PATH_SUFFIXES
|
|
||||||
"lib/VC"
|
|
||||||
"VC"
|
|
||||||
"lib"
|
|
||||||
)
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
find_library(LIB_EAY_DEBUG
|
|
||||||
NAMES
|
|
||||||
libcrypto${_OPENSSL_MSVC_ARCH_SUFFIX}${_OPENSSL_MSVC_RT_MODE}d
|
|
||||||
libcrypto${_OPENSSL_MSVC_RT_MODE}d
|
|
||||||
libcryptod
|
|
||||||
libeay32${_OPENSSL_MSVC_RT_MODE}d
|
|
||||||
libeay32d
|
|
||||||
cryptod
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
${_OPENSSL_PATH_SUFFIXES}
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(LIB_EAY_RELEASE
|
|
||||||
NAMES
|
|
||||||
libcrypto${_OPENSSL_MSVC_ARCH_SUFFIX}${_OPENSSL_MSVC_RT_MODE}
|
|
||||||
libcrypto${_OPENSSL_MSVC_RT_MODE}
|
|
||||||
libcrypto
|
|
||||||
libeay32${_OPENSSL_MSVC_RT_MODE}
|
|
||||||
libeay32
|
|
||||||
crypto
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
${_OPENSSL_PATH_SUFFIXES}
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(SSL_EAY_DEBUG
|
|
||||||
NAMES
|
|
||||||
libssl${_OPENSSL_MSVC_ARCH_SUFFIX}${_OPENSSL_MSVC_RT_MODE}d
|
|
||||||
libssl${_OPENSSL_MSVC_RT_MODE}d
|
|
||||||
libssld
|
|
||||||
ssleay32${_OPENSSL_MSVC_RT_MODE}d
|
|
||||||
ssleay32d
|
|
||||||
ssld
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
${_OPENSSL_PATH_SUFFIXES}
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(SSL_EAY_RELEASE
|
|
||||||
NAMES
|
|
||||||
libssl${_OPENSSL_MSVC_ARCH_SUFFIX}${_OPENSSL_MSVC_RT_MODE}
|
|
||||||
libssl${_OPENSSL_MSVC_RT_MODE}
|
|
||||||
libssl
|
|
||||||
ssleay32${_OPENSSL_MSVC_RT_MODE}
|
|
||||||
ssleay32
|
|
||||||
ssl
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
${_OPENSSL_PATH_SUFFIXES}
|
|
||||||
)
|
|
||||||
|
|
||||||
set(LIB_EAY_LIBRARY_DEBUG "${LIB_EAY_DEBUG}")
|
|
||||||
set(LIB_EAY_LIBRARY_RELEASE "${LIB_EAY_RELEASE}")
|
|
||||||
set(SSL_EAY_LIBRARY_DEBUG "${SSL_EAY_DEBUG}")
|
|
||||||
set(SSL_EAY_LIBRARY_RELEASE "${SSL_EAY_RELEASE}")
|
|
||||||
|
|
||||||
include(${CMAKE_CURRENT_LIST_DIR}/SelectLibraryConfigurations.cmake)
|
|
||||||
select_library_configurations(LIB_EAY)
|
|
||||||
select_library_configurations(SSL_EAY)
|
|
||||||
|
|
||||||
mark_as_advanced(LIB_EAY_LIBRARY_DEBUG LIB_EAY_LIBRARY_RELEASE
|
|
||||||
SSL_EAY_LIBRARY_DEBUG SSL_EAY_LIBRARY_RELEASE)
|
|
||||||
set(OPENSSL_SSL_LIBRARY ${SSL_EAY_LIBRARY} )
|
|
||||||
set(OPENSSL_CRYPTO_LIBRARY ${LIB_EAY_LIBRARY} )
|
|
||||||
elseif(MINGW)
|
|
||||||
# same player, for MinGW
|
|
||||||
set(LIB_EAY_NAMES crypto libeay32)
|
|
||||||
set(SSL_EAY_NAMES ssl ssleay32)
|
|
||||||
find_library(LIB_EAY
|
|
||||||
NAMES
|
|
||||||
${LIB_EAY_NAMES}
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
"lib/MinGW"
|
|
||||||
"lib"
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(SSL_EAY
|
|
||||||
NAMES
|
|
||||||
${SSL_EAY_NAMES}
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
"lib/MinGW"
|
|
||||||
"lib"
|
|
||||||
)
|
|
||||||
|
|
||||||
mark_as_advanced(SSL_EAY LIB_EAY)
|
|
||||||
set(OPENSSL_SSL_LIBRARY ${SSL_EAY} )
|
|
||||||
set(OPENSSL_CRYPTO_LIBRARY ${LIB_EAY} )
|
|
||||||
unset(LIB_EAY_NAMES)
|
|
||||||
unset(SSL_EAY_NAMES)
|
|
||||||
else()
|
|
||||||
# Not sure what to pick for -say- intel, let's use the toplevel ones and hope someone report issues:
|
|
||||||
find_library(LIB_EAY
|
|
||||||
NAMES
|
|
||||||
libcrypto
|
|
||||||
libeay32
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
HINTS
|
|
||||||
${_OPENSSL_LIBDIR}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
lib
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(SSL_EAY
|
|
||||||
NAMES
|
|
||||||
libssl
|
|
||||||
ssleay32
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
HINTS
|
|
||||||
${_OPENSSL_LIBDIR}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
lib
|
|
||||||
)
|
|
||||||
|
|
||||||
mark_as_advanced(SSL_EAY LIB_EAY)
|
|
||||||
set(OPENSSL_SSL_LIBRARY ${SSL_EAY} )
|
|
||||||
set(OPENSSL_CRYPTO_LIBRARY ${LIB_EAY} )
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
|
|
||||||
find_library(OPENSSL_SSL_LIBRARY
|
|
||||||
NAMES
|
|
||||||
ssl
|
|
||||||
ssleay32
|
|
||||||
ssleay32MD
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
HINTS
|
|
||||||
${_OPENSSL_LIBDIR}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
lib
|
|
||||||
)
|
|
||||||
|
|
||||||
find_library(OPENSSL_CRYPTO_LIBRARY
|
|
||||||
NAMES
|
|
||||||
crypto
|
|
||||||
NAMES_PER_DIR
|
|
||||||
${_OPENSSL_ROOT_HINTS_AND_PATHS}
|
|
||||||
HINTS
|
|
||||||
${_OPENSSL_LIBDIR}
|
|
||||||
PATH_SUFFIXES
|
|
||||||
lib
|
|
||||||
)
|
|
||||||
|
|
||||||
mark_as_advanced(OPENSSL_CRYPTO_LIBRARY OPENSSL_SSL_LIBRARY)
|
|
||||||
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# compat defines
|
|
||||||
set(OPENSSL_SSL_LIBRARIES ${OPENSSL_SSL_LIBRARY})
|
|
||||||
set(OPENSSL_CRYPTO_LIBRARIES ${OPENSSL_CRYPTO_LIBRARY})
|
|
||||||
_OpenSSL_test_and_find_dependencies("${OPENSSL_SSL_LIBRARY}" "${OPENSSL_CRYPTO_LIBRARY}")
|
|
||||||
if(_OpenSSL_has_dependencies)
|
|
||||||
_OpenSSL_add_dependencies( OPENSSL_SSL_LIBRARIES "${OPENSSL_SSL_LIBRARY}" )
|
|
||||||
_OpenSSL_add_dependencies( OPENSSL_CRYPTO_LIBRARIES "${OPENSSL_CRYPTO_LIBRARY}" )
|
|
||||||
endif()
|
|
||||||
|
|
||||||
function(from_hex HEX DEC)
|
|
||||||
string(TOUPPER "${HEX}" HEX)
|
|
||||||
set(_res 0)
|
|
||||||
string(LENGTH "${HEX}" _strlen)
|
|
||||||
|
|
||||||
while (_strlen GREATER 0)
|
|
||||||
math(EXPR _res "${_res} * 16")
|
|
||||||
string(SUBSTRING "${HEX}" 0 1 NIBBLE)
|
|
||||||
string(SUBSTRING "${HEX}" 1 -1 HEX)
|
|
||||||
if (NIBBLE STREQUAL "A")
|
|
||||||
math(EXPR _res "${_res} + 10")
|
|
||||||
elseif (NIBBLE STREQUAL "B")
|
|
||||||
math(EXPR _res "${_res} + 11")
|
|
||||||
elseif (NIBBLE STREQUAL "C")
|
|
||||||
math(EXPR _res "${_res} + 12")
|
|
||||||
elseif (NIBBLE STREQUAL "D")
|
|
||||||
math(EXPR _res "${_res} + 13")
|
|
||||||
elseif (NIBBLE STREQUAL "E")
|
|
||||||
math(EXPR _res "${_res} + 14")
|
|
||||||
elseif (NIBBLE STREQUAL "F")
|
|
||||||
math(EXPR _res "${_res} + 15")
|
|
||||||
else()
|
|
||||||
math(EXPR _res "${_res} + ${NIBBLE}")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
string(LENGTH "${HEX}" _strlen)
|
|
||||||
endwhile()
|
|
||||||
|
|
||||||
set(${DEC} ${_res} PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
if(OPENSSL_INCLUDE_DIR AND EXISTS "${OPENSSL_INCLUDE_DIR}/openssl/opensslv.h")
|
|
||||||
file(STRINGS "${OPENSSL_INCLUDE_DIR}/openssl/opensslv.h" openssl_version_str
|
|
||||||
REGEX "^#[\t ]*define[\t ]+OPENSSL_VERSION_NUMBER[\t ]+0x([0-9a-fA-F])+.*")
|
|
||||||
|
|
||||||
if(openssl_version_str)
|
|
||||||
# The version number is encoded as 0xMNNFFPPS: major minor fix patch status
|
|
||||||
# The status gives if this is a developer or prerelease and is ignored here.
|
|
||||||
# Major, minor, and fix directly translate into the version numbers shown in
|
|
||||||
# the string. The patch field translates to the single character suffix that
|
|
||||||
# indicates the bug fix state, which 00 -> nothing, 01 -> a, 02 -> b and so
|
|
||||||
# on.
|
|
||||||
|
|
||||||
string(REGEX REPLACE "^.*OPENSSL_VERSION_NUMBER[\t ]+0x([0-9a-fA-F])([0-9a-fA-F][0-9a-fA-F])([0-9a-fA-F][0-9a-fA-F])([0-9a-fA-F][0-9a-fA-F])([0-9a-fA-F]).*$"
|
|
||||||
"\\1;\\2;\\3;\\4;\\5" OPENSSL_VERSION_LIST "${openssl_version_str}")
|
|
||||||
list(GET OPENSSL_VERSION_LIST 0 OPENSSL_VERSION_MAJOR)
|
|
||||||
list(GET OPENSSL_VERSION_LIST 1 OPENSSL_VERSION_MINOR)
|
|
||||||
from_hex("${OPENSSL_VERSION_MINOR}" OPENSSL_VERSION_MINOR)
|
|
||||||
list(GET OPENSSL_VERSION_LIST 2 OPENSSL_VERSION_FIX)
|
|
||||||
from_hex("${OPENSSL_VERSION_FIX}" OPENSSL_VERSION_FIX)
|
|
||||||
list(GET OPENSSL_VERSION_LIST 3 OPENSSL_VERSION_PATCH)
|
|
||||||
|
|
||||||
if (NOT OPENSSL_VERSION_PATCH STREQUAL "00")
|
|
||||||
from_hex("${OPENSSL_VERSION_PATCH}" _tmp)
|
|
||||||
# 96 is the ASCII code of 'a' minus 1
|
|
||||||
math(EXPR OPENSSL_VERSION_PATCH_ASCII "${_tmp} + 96")
|
|
||||||
unset(_tmp)
|
|
||||||
# Once anyone knows how OpenSSL would call the patch versions beyond 'z'
|
|
||||||
# this should be updated to handle that, too. This has not happened yet
|
|
||||||
# so it is simply ignored here for now.
|
|
||||||
string(ASCII "${OPENSSL_VERSION_PATCH_ASCII}" OPENSSL_VERSION_PATCH_STRING)
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
set(OPENSSL_VERSION "${OPENSSL_VERSION_MAJOR}.${OPENSSL_VERSION_MINOR}.${OPENSSL_VERSION_FIX}${OPENSSL_VERSION_PATCH_STRING}")
|
|
||||||
endif ()
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
set(OPENSSL_LIBRARIES ${OPENSSL_SSL_LIBRARIES} ${OPENSSL_CRYPTO_LIBRARIES} )
|
|
||||||
list(REMOVE_DUPLICATES OPENSSL_LIBRARIES)
|
|
||||||
|
|
||||||
foreach(_comp IN LISTS OpenSSL_FIND_COMPONENTS)
|
|
||||||
if(_comp STREQUAL "Crypto")
|
|
||||||
if(EXISTS "${OPENSSL_INCLUDE_DIR}" AND
|
|
||||||
(EXISTS "${OPENSSL_CRYPTO_LIBRARY}" OR
|
|
||||||
EXISTS "${LIB_EAY_LIBRARY_DEBUG}" OR
|
|
||||||
EXISTS "${LIB_EAY_LIBRARY_RELEASE}")
|
|
||||||
)
|
|
||||||
set(OpenSSL_${_comp}_FOUND TRUE)
|
|
||||||
else()
|
|
||||||
set(OpenSSL_${_comp}_FOUND FALSE)
|
|
||||||
endif()
|
|
||||||
elseif(_comp STREQUAL "SSL")
|
|
||||||
if(EXISTS "${OPENSSL_INCLUDE_DIR}" AND
|
|
||||||
(EXISTS "${OPENSSL_SSL_LIBRARY}" OR
|
|
||||||
EXISTS "${SSL_EAY_LIBRARY_DEBUG}" OR
|
|
||||||
EXISTS "${SSL_EAY_LIBRARY_RELEASE}")
|
|
||||||
)
|
|
||||||
set(OpenSSL_${_comp}_FOUND TRUE)
|
|
||||||
else()
|
|
||||||
set(OpenSSL_${_comp}_FOUND FALSE)
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
message(WARNING "${_comp} is not a valid OpenSSL component")
|
|
||||||
set(OpenSSL_${_comp}_FOUND FALSE)
|
|
||||||
endif()
|
|
||||||
endforeach()
|
|
||||||
unset(_comp)
|
|
||||||
|
|
||||||
include(${CMAKE_CURRENT_LIST_DIR}/FindPackageHandleStandardArgs.cmake)
|
|
||||||
find_package_handle_standard_args(OpenSSL
|
|
||||||
REQUIRED_VARS
|
|
||||||
OPENSSL_CRYPTO_LIBRARY
|
|
||||||
OPENSSL_INCLUDE_DIR
|
|
||||||
VERSION_VAR
|
|
||||||
OPENSSL_VERSION
|
|
||||||
HANDLE_COMPONENTS
|
|
||||||
FAIL_MESSAGE
|
|
||||||
"Could NOT find OpenSSL, try to set the path to OpenSSL root folder in the system variable OPENSSL_ROOT_DIR"
|
|
||||||
)
|
|
||||||
|
|
||||||
mark_as_advanced(OPENSSL_INCLUDE_DIR OPENSSL_LIBRARIES)
|
|
||||||
|
|
||||||
if(OPENSSL_FOUND)
|
|
||||||
if(NOT TARGET OpenSSL::Crypto AND
|
|
||||||
(EXISTS "${OPENSSL_CRYPTO_LIBRARY}" OR
|
|
||||||
EXISTS "${LIB_EAY_LIBRARY_DEBUG}" OR
|
|
||||||
EXISTS "${LIB_EAY_LIBRARY_RELEASE}")
|
|
||||||
)
|
|
||||||
add_library(OpenSSL::Crypto UNKNOWN IMPORTED)
|
|
||||||
set_target_properties(OpenSSL::Crypto PROPERTIES
|
|
||||||
INTERFACE_INCLUDE_DIRECTORIES "${OPENSSL_INCLUDE_DIR}")
|
|
||||||
if(EXISTS "${OPENSSL_CRYPTO_LIBRARY}")
|
|
||||||
set_target_properties(OpenSSL::Crypto PROPERTIES
|
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES "C"
|
|
||||||
IMPORTED_LOCATION "${OPENSSL_CRYPTO_LIBRARY}")
|
|
||||||
endif()
|
|
||||||
if(EXISTS "${LIB_EAY_LIBRARY_RELEASE}")
|
|
||||||
set_property(TARGET OpenSSL::Crypto APPEND PROPERTY
|
|
||||||
IMPORTED_CONFIGURATIONS RELEASE)
|
|
||||||
set_target_properties(OpenSSL::Crypto PROPERTIES
|
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES_RELEASE "C"
|
|
||||||
IMPORTED_LOCATION_RELEASE "${LIB_EAY_LIBRARY_RELEASE}")
|
|
||||||
endif()
|
|
||||||
if(EXISTS "${LIB_EAY_LIBRARY_DEBUG}")
|
|
||||||
set_property(TARGET OpenSSL::Crypto APPEND PROPERTY
|
|
||||||
IMPORTED_CONFIGURATIONS DEBUG)
|
|
||||||
set_target_properties(OpenSSL::Crypto PROPERTIES
|
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES_DEBUG "C"
|
|
||||||
IMPORTED_LOCATION_DEBUG "${LIB_EAY_LIBRARY_DEBUG}")
|
|
||||||
endif()
|
|
||||||
_OpenSSL_target_add_dependencies(OpenSSL::Crypto)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT TARGET OpenSSL::SSL AND
|
|
||||||
(EXISTS "${OPENSSL_SSL_LIBRARY}" OR
|
|
||||||
EXISTS "${SSL_EAY_LIBRARY_DEBUG}" OR
|
|
||||||
EXISTS "${SSL_EAY_LIBRARY_RELEASE}")
|
|
||||||
)
|
|
||||||
add_library(OpenSSL::SSL UNKNOWN IMPORTED)
|
|
||||||
set_target_properties(OpenSSL::SSL PROPERTIES
|
|
||||||
INTERFACE_INCLUDE_DIRECTORIES "${OPENSSL_INCLUDE_DIR}")
|
|
||||||
if(EXISTS "${OPENSSL_SSL_LIBRARY}")
|
|
||||||
set_target_properties(OpenSSL::SSL PROPERTIES
|
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES "C"
|
|
||||||
IMPORTED_LOCATION "${OPENSSL_SSL_LIBRARY}")
|
|
||||||
endif()
|
|
||||||
if(EXISTS "${SSL_EAY_LIBRARY_RELEASE}")
|
|
||||||
set_property(TARGET OpenSSL::SSL APPEND PROPERTY
|
|
||||||
IMPORTED_CONFIGURATIONS RELEASE)
|
|
||||||
set_target_properties(OpenSSL::SSL PROPERTIES
|
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES_RELEASE "C"
|
|
||||||
IMPORTED_LOCATION_RELEASE "${SSL_EAY_LIBRARY_RELEASE}")
|
|
||||||
endif()
|
|
||||||
if(EXISTS "${SSL_EAY_LIBRARY_DEBUG}")
|
|
||||||
set_property(TARGET OpenSSL::SSL APPEND PROPERTY
|
|
||||||
IMPORTED_CONFIGURATIONS DEBUG)
|
|
||||||
set_target_properties(OpenSSL::SSL PROPERTIES
|
|
||||||
IMPORTED_LINK_INTERFACE_LANGUAGES_DEBUG "C"
|
|
||||||
IMPORTED_LOCATION_DEBUG "${SSL_EAY_LIBRARY_DEBUG}")
|
|
||||||
endif()
|
|
||||||
if(TARGET OpenSSL::Crypto)
|
|
||||||
set_target_properties(OpenSSL::SSL PROPERTIES
|
|
||||||
INTERFACE_LINK_LIBRARIES OpenSSL::Crypto)
|
|
||||||
endif()
|
|
||||||
_OpenSSL_target_add_dependencies(OpenSSL::SSL)
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Restore the original find library ordering
|
|
||||||
if(OPENSSL_USE_STATIC_LIBS)
|
|
||||||
set(CMAKE_FIND_LIBRARY_SUFFIXES ${_openssl_ORIG_CMAKE_FIND_LIBRARY_SUFFIXES})
|
|
||||||
endif()
|
|
||||||
268
CMakeModules/FindPackageHandleStandardArgs.cmake
vendored
268
CMakeModules/FindPackageHandleStandardArgs.cmake
vendored
@@ -1,268 +0,0 @@
|
|||||||
# Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
||||||
# file Copyright.txt or https://cmake.org/licensing for details.
|
|
||||||
|
|
||||||
include(${CMAKE_CURRENT_LIST_DIR}/FindPackageMessage.cmake)
|
|
||||||
|
|
||||||
# internal helper macro
|
|
||||||
macro(_FPHSA_FAILURE_MESSAGE _msg)
|
|
||||||
set (__msg "${_msg}")
|
|
||||||
if (FPHSA_REASON_FAILURE_MESSAGE)
|
|
||||||
string(APPEND __msg "\n Reason given by package: ${FPHSA_REASON_FAILURE_MESSAGE}\n")
|
|
||||||
endif()
|
|
||||||
if (${_NAME}_FIND_REQUIRED)
|
|
||||||
message(FATAL_ERROR "${__msg}")
|
|
||||||
else ()
|
|
||||||
if (NOT ${_NAME}_FIND_QUIETLY)
|
|
||||||
message(STATUS "${__msg}")
|
|
||||||
endif ()
|
|
||||||
endif ()
|
|
||||||
endmacro()
|
|
||||||
|
|
||||||
|
|
||||||
# internal helper macro to generate the failure message when used in CONFIG_MODE:
|
|
||||||
macro(_FPHSA_HANDLE_FAILURE_CONFIG_MODE)
|
|
||||||
# <PackageName>_CONFIG is set, but FOUND is false, this means that some other of the REQUIRED_VARS was not found:
|
|
||||||
if(${_NAME}_CONFIG)
|
|
||||||
_FPHSA_FAILURE_MESSAGE("${FPHSA_FAIL_MESSAGE}: missing:${MISSING_VARS} (found ${${_NAME}_CONFIG} ${VERSION_MSG})")
|
|
||||||
else()
|
|
||||||
# If _CONSIDERED_CONFIGS is set, the config-file has been found, but no suitable version.
|
|
||||||
# List them all in the error message:
|
|
||||||
if(${_NAME}_CONSIDERED_CONFIGS)
|
|
||||||
set(configsText "")
|
|
||||||
list(LENGTH ${_NAME}_CONSIDERED_CONFIGS configsCount)
|
|
||||||
math(EXPR configsCount "${configsCount} - 1")
|
|
||||||
foreach(currentConfigIndex RANGE ${configsCount})
|
|
||||||
list(GET ${_NAME}_CONSIDERED_CONFIGS ${currentConfigIndex} filename)
|
|
||||||
list(GET ${_NAME}_CONSIDERED_VERSIONS ${currentConfigIndex} version)
|
|
||||||
string(APPEND configsText "\n ${filename} (version ${version})")
|
|
||||||
endforeach()
|
|
||||||
if (${_NAME}_NOT_FOUND_MESSAGE)
|
|
||||||
if (FPHSA_REASON_FAILURE_MESSAGE)
|
|
||||||
string(PREPEND FPHSA_REASON_FAILURE_MESSAGE "${${_NAME}_NOT_FOUND_MESSAGE}\n ")
|
|
||||||
else()
|
|
||||||
set(FPHSA_REASON_FAILURE_MESSAGE "${${_NAME}_NOT_FOUND_MESSAGE}")
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
string(APPEND configsText "\n")
|
|
||||||
endif()
|
|
||||||
_FPHSA_FAILURE_MESSAGE("${FPHSA_FAIL_MESSAGE} ${VERSION_MSG}, checked the following files:${configsText}")
|
|
||||||
|
|
||||||
else()
|
|
||||||
# Simple case: No Config-file was found at all:
|
|
||||||
_FPHSA_FAILURE_MESSAGE("${FPHSA_FAIL_MESSAGE}: found neither ${_NAME}Config.cmake nor ${_NAME_LOWER}-config.cmake ${VERSION_MSG}")
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
endmacro()
|
|
||||||
|
|
||||||
|
|
||||||
function(FIND_PACKAGE_HANDLE_STANDARD_ARGS _NAME _FIRST_ARG)
|
|
||||||
|
|
||||||
# Set up the arguments for `cmake_parse_arguments`.
|
|
||||||
set(options CONFIG_MODE HANDLE_COMPONENTS)
|
|
||||||
set(oneValueArgs FAIL_MESSAGE REASON_FAILURE_MESSAGE VERSION_VAR FOUND_VAR)
|
|
||||||
set(multiValueArgs REQUIRED_VARS)
|
|
||||||
|
|
||||||
# Check whether we are in 'simple' or 'extended' mode:
|
|
||||||
set(_KEYWORDS_FOR_EXTENDED_MODE ${options} ${oneValueArgs} ${multiValueArgs} )
|
|
||||||
list(FIND _KEYWORDS_FOR_EXTENDED_MODE "${_FIRST_ARG}" INDEX)
|
|
||||||
|
|
||||||
if(${INDEX} EQUAL -1)
|
|
||||||
set(FPHSA_FAIL_MESSAGE ${_FIRST_ARG})
|
|
||||||
set(FPHSA_REQUIRED_VARS ${ARGN})
|
|
||||||
set(FPHSA_VERSION_VAR)
|
|
||||||
else()
|
|
||||||
cmake_parse_arguments(FPHSA "${options}" "${oneValueArgs}" "${multiValueArgs}" ${_FIRST_ARG} ${ARGN})
|
|
||||||
|
|
||||||
if(FPHSA_UNPARSED_ARGUMENTS)
|
|
||||||
message(FATAL_ERROR "Unknown keywords given to FIND_PACKAGE_HANDLE_STANDARD_ARGS(): \"${FPHSA_UNPARSED_ARGUMENTS}\"")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT FPHSA_FAIL_MESSAGE)
|
|
||||||
set(FPHSA_FAIL_MESSAGE "DEFAULT_MSG")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# In config-mode, we rely on the variable <PackageName>_CONFIG, which is set by find_package()
|
|
||||||
# when it successfully found the config-file, including version checking:
|
|
||||||
if(FPHSA_CONFIG_MODE)
|
|
||||||
list(INSERT FPHSA_REQUIRED_VARS 0 ${_NAME}_CONFIG)
|
|
||||||
list(REMOVE_DUPLICATES FPHSA_REQUIRED_VARS)
|
|
||||||
set(FPHSA_VERSION_VAR ${_NAME}_VERSION)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT FPHSA_REQUIRED_VARS)
|
|
||||||
message(FATAL_ERROR "No REQUIRED_VARS specified for FIND_PACKAGE_HANDLE_STANDARD_ARGS()")
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# now that we collected all arguments, process them
|
|
||||||
|
|
||||||
if("x${FPHSA_FAIL_MESSAGE}" STREQUAL "xDEFAULT_MSG")
|
|
||||||
set(FPHSA_FAIL_MESSAGE "Could NOT find ${_NAME}")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
list(GET FPHSA_REQUIRED_VARS 0 _FIRST_REQUIRED_VAR)
|
|
||||||
|
|
||||||
string(TOUPPER ${_NAME} _NAME_UPPER)
|
|
||||||
string(TOLOWER ${_NAME} _NAME_LOWER)
|
|
||||||
|
|
||||||
if(FPHSA_FOUND_VAR)
|
|
||||||
if(FPHSA_FOUND_VAR MATCHES "^${_NAME}_FOUND$" OR FPHSA_FOUND_VAR MATCHES "^${_NAME_UPPER}_FOUND$")
|
|
||||||
set(_FOUND_VAR ${FPHSA_FOUND_VAR})
|
|
||||||
else()
|
|
||||||
message(FATAL_ERROR "The argument for FOUND_VAR is \"${FPHSA_FOUND_VAR}\", but only \"${_NAME}_FOUND\" and \"${_NAME_UPPER}_FOUND\" are valid names.")
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
set(_FOUND_VAR ${_NAME_UPPER}_FOUND)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# collect all variables which were not found, so they can be printed, so the
|
|
||||||
# user knows better what went wrong (#6375)
|
|
||||||
set(MISSING_VARS "")
|
|
||||||
set(DETAILS "")
|
|
||||||
# check if all passed variables are valid
|
|
||||||
set(FPHSA_FOUND_${_NAME} TRUE)
|
|
||||||
foreach(_CURRENT_VAR ${FPHSA_REQUIRED_VARS})
|
|
||||||
if(NOT ${_CURRENT_VAR})
|
|
||||||
set(FPHSA_FOUND_${_NAME} FALSE)
|
|
||||||
string(APPEND MISSING_VARS " ${_CURRENT_VAR}")
|
|
||||||
else()
|
|
||||||
string(APPEND DETAILS "[${${_CURRENT_VAR}}]")
|
|
||||||
endif()
|
|
||||||
endforeach()
|
|
||||||
if(FPHSA_FOUND_${_NAME})
|
|
||||||
set(${_NAME}_FOUND TRUE)
|
|
||||||
set(${_NAME_UPPER}_FOUND TRUE)
|
|
||||||
else()
|
|
||||||
set(${_NAME}_FOUND FALSE)
|
|
||||||
set(${_NAME_UPPER}_FOUND FALSE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# component handling
|
|
||||||
unset(FOUND_COMPONENTS_MSG)
|
|
||||||
unset(MISSING_COMPONENTS_MSG)
|
|
||||||
|
|
||||||
if(FPHSA_HANDLE_COMPONENTS)
|
|
||||||
foreach(comp ${${_NAME}_FIND_COMPONENTS})
|
|
||||||
if(${_NAME}_${comp}_FOUND)
|
|
||||||
|
|
||||||
if(NOT DEFINED FOUND_COMPONENTS_MSG)
|
|
||||||
set(FOUND_COMPONENTS_MSG "found components:")
|
|
||||||
endif()
|
|
||||||
string(APPEND FOUND_COMPONENTS_MSG " ${comp}")
|
|
||||||
|
|
||||||
else()
|
|
||||||
|
|
||||||
if(NOT DEFINED MISSING_COMPONENTS_MSG)
|
|
||||||
set(MISSING_COMPONENTS_MSG "missing components:")
|
|
||||||
endif()
|
|
||||||
string(APPEND MISSING_COMPONENTS_MSG " ${comp}")
|
|
||||||
|
|
||||||
if(${_NAME}_FIND_REQUIRED_${comp})
|
|
||||||
set(${_NAME}_FOUND FALSE)
|
|
||||||
string(APPEND MISSING_VARS " ${comp}")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
endif()
|
|
||||||
endforeach()
|
|
||||||
set(COMPONENT_MSG "${FOUND_COMPONENTS_MSG} ${MISSING_COMPONENTS_MSG}")
|
|
||||||
string(APPEND DETAILS "[c${COMPONENT_MSG}]")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# version handling:
|
|
||||||
set(VERSION_MSG "")
|
|
||||||
set(VERSION_OK TRUE)
|
|
||||||
|
|
||||||
# check with DEFINED here as the requested or found version may be "0"
|
|
||||||
if (DEFINED ${_NAME}_FIND_VERSION)
|
|
||||||
if(DEFINED ${FPHSA_VERSION_VAR})
|
|
||||||
set(_FOUND_VERSION ${${FPHSA_VERSION_VAR}})
|
|
||||||
|
|
||||||
if(${_NAME}_FIND_VERSION_EXACT) # exact version required
|
|
||||||
# count the dots in the version string
|
|
||||||
string(REGEX REPLACE "[^.]" "" _VERSION_DOTS "${_FOUND_VERSION}")
|
|
||||||
# add one dot because there is one dot more than there are components
|
|
||||||
string(LENGTH "${_VERSION_DOTS}." _VERSION_DOTS)
|
|
||||||
if (_VERSION_DOTS GREATER ${_NAME}_FIND_VERSION_COUNT)
|
|
||||||
# Because of the C++ implementation of find_package() ${_NAME}_FIND_VERSION_COUNT
|
|
||||||
# is at most 4 here. Therefore a simple lookup table is used.
|
|
||||||
if (${_NAME}_FIND_VERSION_COUNT EQUAL 1)
|
|
||||||
set(_VERSION_REGEX "[^.]*")
|
|
||||||
elseif (${_NAME}_FIND_VERSION_COUNT EQUAL 2)
|
|
||||||
set(_VERSION_REGEX "[^.]*\\.[^.]*")
|
|
||||||
elseif (${_NAME}_FIND_VERSION_COUNT EQUAL 3)
|
|
||||||
set(_VERSION_REGEX "[^.]*\\.[^.]*\\.[^.]*")
|
|
||||||
else ()
|
|
||||||
set(_VERSION_REGEX "[^.]*\\.[^.]*\\.[^.]*\\.[^.]*")
|
|
||||||
endif ()
|
|
||||||
string(REGEX REPLACE "^(${_VERSION_REGEX})\\..*" "\\1" _VERSION_HEAD "${_FOUND_VERSION}")
|
|
||||||
unset(_VERSION_REGEX)
|
|
||||||
if (NOT ${_NAME}_FIND_VERSION VERSION_EQUAL _VERSION_HEAD)
|
|
||||||
set(VERSION_MSG "Found unsuitable version \"${_FOUND_VERSION}\", but required is exact version \"${${_NAME}_FIND_VERSION}\"")
|
|
||||||
set(VERSION_OK FALSE)
|
|
||||||
else ()
|
|
||||||
set(VERSION_MSG "(found suitable exact version \"${_FOUND_VERSION}\")")
|
|
||||||
endif ()
|
|
||||||
unset(_VERSION_HEAD)
|
|
||||||
else ()
|
|
||||||
if (NOT ${_NAME}_FIND_VERSION VERSION_EQUAL _FOUND_VERSION)
|
|
||||||
set(VERSION_MSG "Found unsuitable version \"${_FOUND_VERSION}\", but required is exact version \"${${_NAME}_FIND_VERSION}\"")
|
|
||||||
set(VERSION_OK FALSE)
|
|
||||||
else ()
|
|
||||||
set(VERSION_MSG "(found suitable exact version \"${_FOUND_VERSION}\")")
|
|
||||||
endif ()
|
|
||||||
endif ()
|
|
||||||
unset(_VERSION_DOTS)
|
|
||||||
|
|
||||||
else() # minimum version specified:
|
|
||||||
if (${_NAME}_FIND_VERSION VERSION_GREATER _FOUND_VERSION)
|
|
||||||
set(VERSION_MSG "Found unsuitable version \"${_FOUND_VERSION}\", but required is at least \"${${_NAME}_FIND_VERSION}\"")
|
|
||||||
set(VERSION_OK FALSE)
|
|
||||||
else ()
|
|
||||||
set(VERSION_MSG "(found suitable version \"${_FOUND_VERSION}\", minimum required is \"${${_NAME}_FIND_VERSION}\")")
|
|
||||||
endif ()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
else()
|
|
||||||
|
|
||||||
# if the package was not found, but a version was given, add that to the output:
|
|
||||||
if(${_NAME}_FIND_VERSION_EXACT)
|
|
||||||
set(VERSION_MSG "(Required is exact version \"${${_NAME}_FIND_VERSION}\")")
|
|
||||||
else()
|
|
||||||
set(VERSION_MSG "(Required is at least version \"${${_NAME}_FIND_VERSION}\")")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
endif()
|
|
||||||
else ()
|
|
||||||
# Check with DEFINED as the found version may be 0.
|
|
||||||
if(DEFINED ${FPHSA_VERSION_VAR})
|
|
||||||
set(VERSION_MSG "(found version \"${${FPHSA_VERSION_VAR}}\")")
|
|
||||||
endif()
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
if(VERSION_OK)
|
|
||||||
string(APPEND DETAILS "[v${${FPHSA_VERSION_VAR}}(${${_NAME}_FIND_VERSION})]")
|
|
||||||
else()
|
|
||||||
set(${_NAME}_FOUND FALSE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
|
|
||||||
# print the result:
|
|
||||||
if (${_NAME}_FOUND)
|
|
||||||
FIND_PACKAGE_MESSAGE(${_NAME} "Found ${_NAME}: ${${_FIRST_REQUIRED_VAR}} ${VERSION_MSG} ${COMPONENT_MSG}" "${DETAILS}")
|
|
||||||
else ()
|
|
||||||
|
|
||||||
if(FPHSA_CONFIG_MODE)
|
|
||||||
_FPHSA_HANDLE_FAILURE_CONFIG_MODE()
|
|
||||||
else()
|
|
||||||
if(NOT VERSION_OK)
|
|
||||||
_FPHSA_FAILURE_MESSAGE("${FPHSA_FAIL_MESSAGE}: ${VERSION_MSG} (found ${${_FIRST_REQUIRED_VAR}})")
|
|
||||||
else()
|
|
||||||
_FPHSA_FAILURE_MESSAGE("${FPHSA_FAIL_MESSAGE} (missing:${MISSING_VARS}) ${VERSION_MSG}")
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
endif ()
|
|
||||||
|
|
||||||
set(${_NAME}_FOUND ${${_NAME}_FOUND} PARENT_SCOPE)
|
|
||||||
set(${_NAME_UPPER}_FOUND ${${_NAME}_FOUND} PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
48
CMakeModules/FindPackageMessage.cmake
vendored
48
CMakeModules/FindPackageMessage.cmake
vendored
@@ -1,48 +0,0 @@
|
|||||||
# Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
||||||
# file Copyright.txt or https://cmake.org/licensing for details.
|
|
||||||
|
|
||||||
#[=======================================================================[.rst:
|
|
||||||
FindPackageMessage
|
|
||||||
------------------
|
|
||||||
|
|
||||||
.. code-block:: cmake
|
|
||||||
|
|
||||||
find_package_message(<name> "message for user" "find result details")
|
|
||||||
|
|
||||||
This function is intended to be used in FindXXX.cmake modules files.
|
|
||||||
It will print a message once for each unique find result. This is
|
|
||||||
useful for telling the user where a package was found. The first
|
|
||||||
argument specifies the name (XXX) of the package. The second argument
|
|
||||||
specifies the message to display. The third argument lists details
|
|
||||||
about the find result so that if they change the message will be
|
|
||||||
displayed again. The macro also obeys the QUIET argument to the
|
|
||||||
find_package command.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
.. code-block:: cmake
|
|
||||||
|
|
||||||
if(X11_FOUND)
|
|
||||||
find_package_message(X11 "Found X11: ${X11_X11_LIB}"
|
|
||||||
"[${X11_X11_LIB}][${X11_INCLUDE_DIR}]")
|
|
||||||
else()
|
|
||||||
...
|
|
||||||
endif()
|
|
||||||
#]=======================================================================]
|
|
||||||
|
|
||||||
function(find_package_message pkg msg details)
|
|
||||||
# Avoid printing a message repeatedly for the same find result.
|
|
||||||
if(NOT ${pkg}_FIND_QUIETLY)
|
|
||||||
string(REPLACE "\n" "" details "${details}")
|
|
||||||
set(DETAILS_VAR FIND_PACKAGE_MESSAGE_DETAILS_${pkg})
|
|
||||||
if(NOT "${details}" STREQUAL "${${DETAILS_VAR}}")
|
|
||||||
# The message has not yet been printed.
|
|
||||||
message(STATUS "${msg}")
|
|
||||||
|
|
||||||
# Save the find details in the cache to avoid printing the same
|
|
||||||
# message again.
|
|
||||||
set("${DETAILS_VAR}" "${details}"
|
|
||||||
CACHE INTERNAL "Details about finding ${pkg}")
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
endfunction()
|
|
||||||
30
Dockerfile
Normal file
30
Dockerfile
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
FROM simon987/sist2-build as build
|
||||||
|
MAINTAINER simon987 <me@simon987.net>
|
||||||
|
|
||||||
|
WORKDIR /build/
|
||||||
|
COPY . .
|
||||||
|
RUN cmake -DSIST_PLATFORM=x64_linux -DSIST_DEBUG=off -DBUILD_TESTS=off -DCMAKE_TOOLCHAIN_FILE=/vcpkg/scripts/buildsystems/vcpkg.cmake .
|
||||||
|
RUN make -j$(nproc)
|
||||||
|
RUN strip sist2
|
||||||
|
RUN ls -lh
|
||||||
|
RUN ls -lh sist2-vue/dist/
|
||||||
|
|
||||||
|
FROM ubuntu:20.10
|
||||||
|
|
||||||
|
RUN apt update && apt install -y curl libasan5
|
||||||
|
|
||||||
|
RUN mkdir -p /usr/share/tessdata && \
|
||||||
|
cd /usr/share/tessdata/ && \
|
||||||
|
curl -o /usr/share/tessdata/hin.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/hin.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/jpn.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/jpn.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/eng.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/eng.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/fra.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/fra.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/rus.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/rus.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/spa.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/spa.traineddata
|
||||||
|
|
||||||
|
COPY --from=build /build/sist2 /root/sist2
|
||||||
|
|
||||||
|
ENV LANG C.UTF-8
|
||||||
|
ENV LC_ALL C.UTF-8
|
||||||
|
|
||||||
|
ENTRYPOINT ["/root/sist2"]
|
||||||
28
Dockerfile.arm64
Normal file
28
Dockerfile.arm64
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
FROM simon987/sist2-build-arm64 as build
|
||||||
|
MAINTAINER simon987 <me@simon987.net>
|
||||||
|
|
||||||
|
WORKDIR /build/
|
||||||
|
ADD . /build/
|
||||||
|
RUN cmake -DSIST_PLATFORM=arm64_linux -DSIST_DEBUG=off -DBUILD_TESTS=off -DCMAKE_TOOLCHAIN_FILE=/vcpkg/scripts/buildsystems/vcpkg.cmake .
|
||||||
|
RUN make -j$(nproc)
|
||||||
|
RUN strip sist2
|
||||||
|
|
||||||
|
FROM ubuntu:20.10
|
||||||
|
|
||||||
|
RUN apt update && apt install -y curl libasan5
|
||||||
|
|
||||||
|
RUN mkdir -p /usr/share/tessdata && \
|
||||||
|
cd /usr/share/tessdata/ && \
|
||||||
|
curl -o /usr/share/tessdata/hin.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/hin.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/jpn.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/jpn.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/eng.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/eng.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/fra.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/fra.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/rus.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/rus.traineddata &&\
|
||||||
|
curl -o /usr/share/tessdata/spa.traineddata https://raw.githubusercontent.com/tesseract-ocr/tessdata/master/spa.traineddata
|
||||||
|
|
||||||
|
COPY --from=build /build/sist2 /root/sist2
|
||||||
|
|
||||||
|
ENV LANG C.UTF-8
|
||||||
|
ENV LC_ALL C.UTF-8
|
||||||
|
|
||||||
|
ENTRYPOINT ["/root/sist2"]
|
||||||
165
README.md
165
README.md
@@ -1,5 +1,8 @@
|
|||||||

|

|
||||||
[](https://www.codefactor.io/repository/github/simon987/sist2)
|
[](https://www.codefactor.io/repository/github/simon987/sist2)
|
||||||
|
[](https://files.simon987.net/.gate/sist2/simon987_sist2/)
|
||||||
|
|
||||||
|
**Demo**: [sist2.simon987.net](https://sist2.simon987.net/?i=Demo%20files)
|
||||||
|
|
||||||
# sist2
|
# sist2
|
||||||
|
|
||||||
@@ -7,90 +10,146 @@ sist2 (Simple incremental search tool)
|
|||||||
|
|
||||||
*Warning: sist2 is in early development*
|
*Warning: sist2 is in early development*
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
* Fast, low memory usage, multi-threaded
|
* Fast, low memory usage, multi-threaded
|
||||||
|
* Mobile-friendly Web interface
|
||||||
* Portable (all its features are packaged in a single executable)
|
* Portable (all its features are packaged in a single executable)
|
||||||
* Extracts text from common file types\*
|
* Extracts text and metadata from common file types \*
|
||||||
* Generates thumbnails\*
|
* Generates thumbnails \*
|
||||||
* Incremental scanning
|
* Incremental scanning
|
||||||
|
* Manual tagging from the UI and automatic tagging based on file attributes via [user scripts](docs/scripting.md)
|
||||||
|
* Recursive scan inside archive files \*\*
|
||||||
|
* OCR support with tesseract \*\*\*
|
||||||
|
* Stats page & disk utilisation visualization
|
||||||
|
|
||||||
|
\* See [format support](#format-support)
|
||||||
|
\*\* See [Archive files](#archive-files)
|
||||||
|
\*\*\* See [OCR](#ocr)
|
||||||
|
|
||||||
\* See [format support](#format-support)
|

|
||||||
|
|
||||||
## Getting Started
|
## Getting Started
|
||||||
|
|
||||||
1. Have an [Elasticsearch](https://www.elastic.co/downloads/elasticsearch) instance running
|
1. Have an Elasticsearch (>= 6.X.X) instance running
|
||||||
1. Download the [latest sist2 release](https://github.com/simon987/sist2/releases)
|
1. Download [from official website](https://www.elastic.co/downloads/elasticsearch)
|
||||||
|
1. *(or)* Run using docker:
|
||||||
|
```bash
|
||||||
|
docker run -d --name es1 --net sist2_net -p 9200:9200 \
|
||||||
|
-e "discovery.type=single-node" elasticsearch:7.14.0
|
||||||
|
```
|
||||||
|
1. *(or)* Run using docker-compose:
|
||||||
|
```yaml
|
||||||
|
elasticsearch:
|
||||||
|
image: docker.elastic.co/elasticsearch/elasticsearch:7.14.0
|
||||||
|
environment:
|
||||||
|
- discovery.type=single-node
|
||||||
|
- "ES_JAVA_OPTS=-Xms1G -Xmx2G"
|
||||||
|
```
|
||||||
|
1. Download sist2 executable
|
||||||
|
1. Download the [latest sist2 release](https://github.com/simon987/sist2/releases) *
|
||||||
|
1. *(or)* Download a [development snapshot](https://files.simon987.net/.gate/sist2/simon987_sist2/) *(Not
|
||||||
|
recommended!)*
|
||||||
|
1. *(or)* `docker pull simon987/sist2:2.11.2-x64-linux`
|
||||||
|
|
||||||
*Windows users*: `sist2` runs under [WSL](https://en.wikipedia.org/wiki/Windows_Subsystem_for_Linux)
|
1. See [Usage guide](docs/USAGE.md)
|
||||||
|
|
||||||
*Mac users*: See [#1](https://github.com/simon987/sist2/issues/1)
|
|
||||||
|
|
||||||
|
\* *Windows users*: **sist2** runs under [WSL](https://en.wikipedia.org/wiki/Windows_Subsystem_for_Linux)
|
||||||
|
|
||||||
## Example usage
|
## Example usage
|
||||||
|
|
||||||

|
See [Usage guide](docs/USAGE.md) for more details
|
||||||
|
|
||||||
See help page `sist2 --help` for more details.
|
1. Scan a directory: `sist2 scan ~/Documents -o ./docs_idx`
|
||||||
|
1. Push index to Elasticsearch: `sist2 index ./docs_idx`
|
||||||
**Scan a directory**
|
1. Start web interface: `sist2 web ./docs_idx`
|
||||||
```bash
|
|
||||||
sist2 scan ~/Documents -o ./orig_idx/
|
|
||||||
sist2 scan --threads 4 --content-size 16384 /mnt/Pictures
|
|
||||||
sist2 scan --incremental ./orig_idx/ -o ./updated_idx/ ~/Documents
|
|
||||||
```
|
|
||||||
|
|
||||||
**Push index to Elasticsearch or file**
|
|
||||||
```bash
|
|
||||||
sist2 index --force-reset ./my_idx
|
|
||||||
sist2 index --print ./my_idx > raw_documents.ndjson
|
|
||||||
```
|
|
||||||
|
|
||||||
**Start web interface**
|
|
||||||
```bash
|
|
||||||
sist2 web --bind 0.0.0.0 --port 4321 ./my_idx1 ./my_idx2 ./my_idx3
|
|
||||||
```
|
|
||||||
|
|
||||||
## Format support
|
## Format support
|
||||||
|
|
||||||
File type | Library | Content | Thumbnail | Metadata
|
File type | Library | Content | Thumbnail | Metadata
|
||||||
:---|:---|:---|:---|:---
|
:---|:---|:---|:---|:---
|
||||||
pdf,xps,cbz,cbr,fb2,epub | MuPDF | yes | yes, `png` | title |
|
pdf,xps,fb2,epub | MuPDF | text+ocr | yes | author, title |
|
||||||
`audio/*` | ffmpeg | - | yes, `jpeg` | ID3 tags |
|
cbz,cbr | [libscan](https://github.com/simon987/libscan) | - | yes | - |
|
||||||
`video/*` | ffmpeg | - | yes, `jpeg` | title, comment |
|
`audio/*` | ffmpeg | - | yes | ID3 tags |
|
||||||
`image/*` | ffmpeg | - | yes, `jpeg` | *planned* |
|
`video/*` | ffmpeg | - | yes | title, comment, artist |
|
||||||
|
`image/*` | ffmpeg | - | yes | [Common EXIF tags](https://github.com/simon987/sist2/blob/efdde2734eca9b14a54f84568863b7ffd59bdba3/src/parsing/media.c#L190), GPS tags |
|
||||||
|
raw, rw2, dng, cr2, crw, dcr, k25, kdc, mrw, pef, xf3, arw, sr2, srf, erf | LibRaw | - | yes | Common EXIF tags, GPS tags |
|
||||||
ttf,ttc,cff,woff,fnt,otf | Freetype2 | - | yes, `bmp` | Name & style |
|
ttf,ttc,cff,woff,fnt,otf | Freetype2 | - | yes, `bmp` | Name & style |
|
||||||
`text/plain` | *(none)* | yes | no | - |
|
`text/plain` | [libscan](https://github.com/simon987/libscan) | yes | no | - |
|
||||||
docx, xlsx, pptx | | *planned* | no | *planned* |
|
html, xml | [libscan](https://github.com/simon987/libscan) | yes | no | - |
|
||||||
|
tar, zip, rar, 7z, ar ... | Libarchive | yes\* | - | no |
|
||||||
|
docx, xlsx, pptx | [libscan](https://github.com/simon987/libscan) | yes | if embedded | creator, modified_by, title |
|
||||||
|
doc (MS Word 97-2003) | antiword | yes | yes | author, title |
|
||||||
|
mobi, azw, azw3 | libmobi | yes | no | author, title |
|
||||||
|
wpd (WordPerfect) | libwpd | yes | no | *planned* |
|
||||||
|
json, jsonl, ndjson | [libscan](https://github.com/simon987/libscan) | yes | - | - |
|
||||||
|
|
||||||
|
\* *See [Archive files](#archive-files)*
|
||||||
|
|
||||||
|
### Archive files
|
||||||
|
|
||||||
|
**sist2** will scan files stored into archive files (zip, tar, 7z...) as if they were directly in the file system.
|
||||||
|
Recursive (archives inside archives)
|
||||||
|
scan is also supported.
|
||||||
|
|
||||||
|
**Limitations**:
|
||||||
|
|
||||||
|
* Support for parsing media files with formats that require *seek* (e.g. `.gif`, `.mp4` w/ fragmented metadata etc.)
|
||||||
|
is limitted (see `--mem-buffer` option)
|
||||||
|
* Archive files are scanned sequentially, by a single thread. On systems where
|
||||||
|
**sist2** is not I/O bound, scans might be faster when larger archives are split into smaller parts.
|
||||||
|
|
||||||
|
### OCR
|
||||||
|
|
||||||
|
You can enable OCR support for pdf,xps,fb2,epub file types with the
|
||||||
|
`--ocr <lang>` option. Download the language data files with your package manager (`apt install tesseract-ocr-eng`) or
|
||||||
|
directly [from Github](https://github.com/tesseract-ocr/tesseract/wiki/Data-Files).
|
||||||
|
|
||||||
|
The `simon987/sist2` image comes with common languages
|
||||||
|
(hin, jpn, eng, fra, rus, spa) pre-installed.
|
||||||
|
|
||||||
|
Examples
|
||||||
|
|
||||||
|
```bash
|
||||||
|
sist2 scan --ocr jpn ~/Books/Manga/
|
||||||
|
sist2 scan --ocr eng ~/Books/Textbooks/
|
||||||
|
```
|
||||||
|
|
||||||
## Build from source
|
## Build from source
|
||||||
|
|
||||||
You can compile **sist2** by yourself if you don't want to use the pre-compiled
|
You can compile **sist2** by yourself if you don't want to use the pre-compiled binaries
|
||||||
binaries.
|
|
||||||
|
### With docker (recommended)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone --recursive https://github.com/simon987/sist2/
|
||||||
|
cd sist2
|
||||||
|
docker build . -f ./Dockerfile -t my-sist2-image
|
||||||
|
docker run --rm my-sist2-image cat /root/sist2 > sist2-x64-linux
|
||||||
|
```
|
||||||
|
|
||||||
|
### On a linux computer
|
||||||
|
|
||||||
1. Install compile-time dependencies
|
1. Install compile-time dependencies
|
||||||
|
|
||||||
*(Debian)*
|
```bash
|
||||||
```bash
|
apt install gcc g++ python3 yasm ragel automake autotools-dev wget libtool libssl-dev curl zip unzip tar xorg-dev libglu1-mesa-dev libxcursor-dev libxml2-dev libxinerama-dev gettext nasm git
|
||||||
apt install git cmake pkg-config libglib2.0-dev\
|
|
||||||
libssl-dev uuid-dev libavformat-dev libswscale-dev \
|
|
||||||
python3 libmagic-dev libfreetype6-dev libcurl-dev \
|
|
||||||
libbz2-dev yasm libharfbuzz-dev
|
|
||||||
```
|
|
||||||
*(FreeBSD)*
|
|
||||||
```bash
|
|
||||||
pkg install cmake gcc yasm gmake bash ffmpeg e2fsprogs-uuid\
|
|
||||||
autotools ragel
|
|
||||||
```
|
```
|
||||||
|
|
||||||
2. Build
|
1. Apply vcpkg patches, as per [sist2-build](https://github.com/simon987/sist2-build) Dockerfile
|
||||||
|
|
||||||
|
1. Install vcpkg dependencies
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone --recurse-submodules https://github.com/simon987/sist2
|
vcpkg install curl[core,openssl]
|
||||||
./scripts/get_static_libs.sh
|
vcpkg install lmdb cjson glib brotli libarchive[core,bzip2,libxml2,lz4,lzma,lzo] pthread tesseract libxml2 libmupdf gtest mongoose libuuid libmagic libraw jasper lcms gumbo
|
||||||
cmake .
|
```
|
||||||
|
|
||||||
|
1. Build
|
||||||
|
```bash
|
||||||
|
git clone --recursive https://github.com/simon987/sist2/
|
||||||
|
cmake -DSIST_DEBUG=off -DCMAKE_TOOLCHAIN_FILE=<VCPKG_ROOT>/scripts/buildsystems/vcpkg.cmake .
|
||||||
make
|
make
|
||||||
```
|
```
|
||||||
|
|||||||
1
argparse
1
argparse
Submodule argparse deleted from fafc503d23
1
cJSON
1
cJSON
Submodule cJSON deleted from 533ff8a783
17
ci/build.sh
Executable file
17
ci/build.sh
Executable file
@@ -0,0 +1,17 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
VCPKG_ROOT="/vcpkg"
|
||||||
|
|
||||||
|
git submodule update --init --recursive
|
||||||
|
|
||||||
|
rm -rf CMakeFiles CMakeCache.txt
|
||||||
|
cmake -DSIST_PLATFORM=x64_linux -DSIST_DEBUG=off -DBUILD_TESTS=off -DCMAKE_TOOLCHAIN_FILE="${VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake" .
|
||||||
|
make -j $(nproc)
|
||||||
|
strip sist2
|
||||||
|
./sist2 -v > VERSION
|
||||||
|
mv sist2 sist2-x64-linux
|
||||||
|
|
||||||
|
rm -rf CMakeFiles CMakeCache.txt
|
||||||
|
cmake -DSIST_PLATFORM=x64_linux -DSIST_DEBUG=on -DBUILD_TESTS=off -DCMAKE_TOOLCHAIN_FILE="${VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake" .
|
||||||
|
make -j $(nproc)
|
||||||
|
mv sist2_debug sist2-x64-linux-debug
|
||||||
17
ci/build_arm64.sh
Executable file
17
ci/build_arm64.sh
Executable file
@@ -0,0 +1,17 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
VCPKG_ROOT="/vcpkg"
|
||||||
|
|
||||||
|
git submodule update --init --recursive
|
||||||
|
|
||||||
|
rm -rf CMakeFiles CMakeCache.txt
|
||||||
|
cmake -DSIST_PLATFORM=arm64_linux -DSIST_DEBUG=off -DBUILD_TESTS=off -DCMAKE_TOOLCHAIN_FILE="${VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake" .
|
||||||
|
make -j $(nproc)
|
||||||
|
strip sist2
|
||||||
|
mv sist2 sist2-arm64-linux
|
||||||
|
|
||||||
|
rm -rf CMakeFiles CMakeCache.txt
|
||||||
|
cmake -DSIST_PLATFORM=arm64_linux -DSIST_DEBUG=on -DBUILD_TESTS=off -DCMAKE_TOOLCHAIN_FILE="${VCPKG_ROOT}/scripts/buildsystems/vcpkg.cmake" .
|
||||||
|
make -j $(nproc)
|
||||||
|
strip sist2
|
||||||
|
mv sist2_debug sist2-arm64-linux-debug
|
||||||
340
docs/USAGE.md
Normal file
340
docs/USAGE.md
Normal file
@@ -0,0 +1,340 @@
|
|||||||
|
# Usage
|
||||||
|
|
||||||
|
*More examples (specifically with docker/compose) are in progress*
|
||||||
|
|
||||||
|
* [scan](#scan)
|
||||||
|
* [options](#scan-options)
|
||||||
|
* [examples](#scan-examples)
|
||||||
|
* [index format](#index-format)
|
||||||
|
* [index](#index)
|
||||||
|
* [options](#index-options)
|
||||||
|
* [examples](#index-examples)
|
||||||
|
* [web](#web)
|
||||||
|
* [options](#web-options)
|
||||||
|
* [examples](#web-examples)
|
||||||
|
* [rewrite_url](#rewrite_url)
|
||||||
|
* [link to specific indices](#link-to-specific-indices)
|
||||||
|
* [exec-script](#exec-script)
|
||||||
|
* [tagging](#tagging)
|
||||||
|
* [sidecar files](#sidecar-files)
|
||||||
|
|
||||||
|
```
|
||||||
|
Usage: sist2 scan [OPTION]... PATH
|
||||||
|
or: sist2 index [OPTION]... INDEX
|
||||||
|
or: sist2 web [OPTION]... INDEX...
|
||||||
|
or: sist2 exec-script [OPTION]... INDEX
|
||||||
|
Lightning-fast file system indexer and search tool.
|
||||||
|
|
||||||
|
-h, --help show this help message and exit
|
||||||
|
-v, --version Show version and exit
|
||||||
|
--verbose Turn on logging
|
||||||
|
--very-verbose Turn on debug messages
|
||||||
|
|
||||||
|
Scan options
|
||||||
|
-t, --threads=<int> Number of threads. DEFAULT=1
|
||||||
|
-q, --quality=<flt> Thumbnail quality, on a scale of 1.0 to 31.0, 1.0 being the best. DEFAULT=3
|
||||||
|
--size=<int> Thumbnail size, in pixels. Use negative value to disable. DEFAULT=500
|
||||||
|
--content-size=<int> Number of bytes to be extracted from text documents. Use negative value to disable. DEFAULT=32768
|
||||||
|
--incremental=<str> Reuse an existing index and only scan modified files.
|
||||||
|
-o, --output=<str> Output directory. DEFAULT=index.sist2/
|
||||||
|
--rewrite-url=<str> Serve files from this url instead of from disk.
|
||||||
|
--name=<str> Index display name. DEFAULT: (name of the directory)
|
||||||
|
--depth=<int> Scan up to DEPTH subdirectories deep. Use 0 to only scan files in PATH. DEFAULT: -1
|
||||||
|
--archive=<str> Archive file mode (skip|list|shallow|recurse). skip: Don't parse, list: only get file names as text, shallow: Don't parse archives inside archives. DEFAULT: recurse
|
||||||
|
--archive-passphrase=<str> Passphrase for encrypted archive files
|
||||||
|
--ocr=<str> Tesseract language (use tesseract --list-langs to see which are installed on your machine)
|
||||||
|
-e, --exclude=<str> Files that match this regex will not be scanned
|
||||||
|
--fast Only index file names & mime type
|
||||||
|
--treemap-threshold=<str> Relative size threshold for treemap (see USAGE.md). DEFAULT: 0.0005
|
||||||
|
--mem-buffer=<int> Maximum memory buffer size per thread in MB for files inside archives (see USAGE.md). DEFAULT: 2000
|
||||||
|
--read-subtitles Read subtitles from media files.
|
||||||
|
--fast-epub Faster but less accurate EPUB parsing (no thumbnails, metadata)
|
||||||
|
--checksums Calculate file checksums when scanning.
|
||||||
|
|
||||||
|
Index options
|
||||||
|
-t, --threads=<int> Number of threads. DEFAULT=1
|
||||||
|
--es-url=<str> Elasticsearch url with port. DEFAULT=http://localhost:9200
|
||||||
|
--es-index=<str> Elasticsearch index name. DEFAULT=sist2
|
||||||
|
-p, --print Just print JSON documents to stdout.
|
||||||
|
--script-file=<str> Path to user script.
|
||||||
|
--mappings-file=<str> Path to Elasticsearch mappings.
|
||||||
|
--settings-file=<str> Path to Elasticsearch settings.
|
||||||
|
--async-script Execute user script asynchronously.
|
||||||
|
--batch-size=<int> Index batch size. DEFAULT: 100
|
||||||
|
-f, --force-reset Reset Elasticsearch mappings and settings. (You must use this option the first time you use the index command)
|
||||||
|
|
||||||
|
Web options
|
||||||
|
--es-url=<str> Elasticsearch url. DEFAULT=http://localhost:9200
|
||||||
|
--es-index=<str> Elasticsearch index name. DEFAULT=sist2
|
||||||
|
--bind=<str> Listen on this address. DEFAULT=localhost:4090
|
||||||
|
--auth=<str> Basic auth in user:password format
|
||||||
|
--tag-auth=<str> Basic auth in user:password format for tagging
|
||||||
|
--tagline=<str> Tagline in navbar
|
||||||
|
--dev Serve html & js files from disk (for development)
|
||||||
|
|
||||||
|
Exec-script options
|
||||||
|
--es-url=<str> Elasticsearch url. DEFAULT=http://localhost:9200
|
||||||
|
--es-index=<str> Elasticsearch index name. DEFAULT=sist2
|
||||||
|
--script-file=<str> Path to user script.
|
||||||
|
--async-script Execute user script asynchronously.
|
||||||
|
```
|
||||||
|
|
||||||
|
## Scan
|
||||||
|
|
||||||
|
### Scan options
|
||||||
|
|
||||||
|
* `-t, --threads`
|
||||||
|
Number of threads for file parsing. **Do not set a number higher than `$(nproc)` or `$(Get-WmiObject Win32_ComputerSystem).NumberOfLogicalProcessors` in Windows!**
|
||||||
|
* `-q, --quality`
|
||||||
|
Thumbnail quality, on a scale of 1.0 to 31.0, 1.0 being the best.
|
||||||
|
* `--size`
|
||||||
|
Thumbnail size in pixels.
|
||||||
|
* `--content-size`
|
||||||
|
Number of bytes of text to be extracted from the content of files (plain text and PDFs).
|
||||||
|
Repeated whitespace and special characters do not count toward this limit.
|
||||||
|
* `--incremental`
|
||||||
|
Specify an existing index. Information about files in this index that were not modified (based on *mtime* attribute)
|
||||||
|
will be copied to the new index and will not be parsed again.
|
||||||
|
* `-o, --output` Output directory.
|
||||||
|
* `--rewrite-url` Set the `rewrite_url` option for the web module (See [rewrite_url](#rewrite_url))
|
||||||
|
* `--name` Set the `name` option for the web module
|
||||||
|
* `--depth` Maximum scan dept. Set to 0 only scan files directly in the root directory, set to -1 for infinite depth
|
||||||
|
* `--archive` Archive file mode.
|
||||||
|
* skip: Don't parse
|
||||||
|
* list: Only get file names as text
|
||||||
|
* shallow: Don't parse archives inside archives.
|
||||||
|
* recurse: Scan archives recursively (default)
|
||||||
|
* `--ocr` See [OCR](../README.md#OCR)
|
||||||
|
* `-e, --exclude` Regex pattern to exclude files. A file is excluded if the pattern matches any
|
||||||
|
part of the full absolute path.
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
* `-e ".*\.ttf"`: Ignore ttf files
|
||||||
|
* `-e ".*\.(ttf|rar)"`: Ignore ttf and rar files
|
||||||
|
* `-e "^/mnt/backups/"`: Ignore all files in the `/mnt/backups/` directory
|
||||||
|
* `-e "^/mnt/Data[12]/"`: Ignore all files in the `/mnt/Data1/` and `/mnt/Data2/` directory
|
||||||
|
* `-e "(^/usr/)|(^/var/)|(^/media/DRIVE-A/tmp/)|(^/media/DRIVE-B/Trash/)"` Exclude the
|
||||||
|
`/usr`, `/var`, `/media/DRIVE-A/tmp`, `/media/DRIVE-B/Trash` directories
|
||||||
|
* `--fast` Only index file names and mime type
|
||||||
|
* `--treemap-threshold` Directories smaller than (`treemap-threshold` * `<total size of the index>`)
|
||||||
|
will not be considered for the disk utilisation visualization; their size will be added to
|
||||||
|
the parent directory. If the parent directory is still smaller than the threshold, it will also be "merged upwards"
|
||||||
|
and so on.
|
||||||
|
|
||||||
|
In effect, smaller `treemap-threshold` values will yield a more detailed
|
||||||
|
(but also a more cluttered and harder to read) visualization.
|
||||||
|
|
||||||
|
* `--mem-buffer` Maximum memory buffer size in MB (per thread) for files inside archives. Media files
|
||||||
|
larger than this number will be read sequentially and no *seek* operations will be supported.
|
||||||
|
|
||||||
|
To check if a media file can be parsed without *seek*, execute `cat file.mp4 | ffprobe -`
|
||||||
|
* `--read-subtitles` When enabled, will attempt to read the subtitles stream from media files.
|
||||||
|
* `--fast-epub` Much faster but less accurate EPUB parsing. When enabled, sist2 will use a simple HTML parser to read epub files instead of the MuPDF library. No thumbnails are generated and author/title metadata are not parsed.
|
||||||
|
* `--checksums` Calculate file checksums (sha1) when scanning files. This option does not cause any additional read
|
||||||
|
operations. Checksums are not calculated for all file types, unless the file is inside an archive. When enabled, duplicate
|
||||||
|
files are hidden in the web UI (this behaviour can be toggled in the Configuration page).
|
||||||
|
|
||||||
|
### Scan examples
|
||||||
|
|
||||||
|
Simple scan
|
||||||
|
```bash
|
||||||
|
sist2 scan ~/Documents
|
||||||
|
|
||||||
|
sist2 scan \
|
||||||
|
--threads 4 --content-size 16000000 --quality 1.0 --archive shallow \
|
||||||
|
--name "My Documents" --rewrite-url "http://nas.domain.local/My Documents/" \
|
||||||
|
~/Documents -o ./documents.idx/
|
||||||
|
```
|
||||||
|
|
||||||
|
Incremental scan
|
||||||
|
```
|
||||||
|
sist2 scan --incremental ./orig_idx/ -o ./updated_idx/ ~/Documents
|
||||||
|
```
|
||||||
|
|
||||||
|
### Index format
|
||||||
|
|
||||||
|
A typical `ndjson` type index structure looks like this:
|
||||||
|
```
|
||||||
|
documents.idx/
|
||||||
|
├── descriptor.json
|
||||||
|
├── _index_main.ndjson.zst
|
||||||
|
├── treemap.csv
|
||||||
|
├── agg_mime.csv
|
||||||
|
├── agg_date.csv
|
||||||
|
├── add_size.csv
|
||||||
|
├── thumbs/
|
||||||
|
| ├── data.mdb
|
||||||
|
| └── lock.mdb
|
||||||
|
├── tags/
|
||||||
|
| ├── data.mdb
|
||||||
|
| └── lock.mdb
|
||||||
|
└── meta/
|
||||||
|
├── data.mdb
|
||||||
|
└── lock.mdb
|
||||||
|
```
|
||||||
|
|
||||||
|
The `_index_*.ndjson.zst` files contain the document data in JSON format, in a compressed newline-delemited file.
|
||||||
|
|
||||||
|
The `thumbs/` folder is a [LMDB](https://en.wikipedia.org/wiki/Lightning_Memory-Mapped_Database)
|
||||||
|
database containing the thumbnails.
|
||||||
|
|
||||||
|
The `descriptor.json` file contains general information about the index. The
|
||||||
|
following fields are safe to modify manually: `root`, `name`, [rewrite_url](#rewrite_url) and `timestamp`.
|
||||||
|
|
||||||
|
The `.csv` are pre-computed aggregations necessary for the stats page.
|
||||||
|
|
||||||
|
*thumbs/*:
|
||||||
|
|
||||||
|
LMDB key-value store. Keys are **binary** 16-byte md5 hash* (`_id` field)
|
||||||
|
and values are raw image bytes.
|
||||||
|
|
||||||
|
*\* Hash is calculated from the full path of the file, including the extension, relative to the index root*
|
||||||
|
|
||||||
|
|
||||||
|
## Index
|
||||||
|
### Index options
|
||||||
|
* `--es-url`
|
||||||
|
Elasticsearch url and port. If you are using docker, make sure that both containers are on the
|
||||||
|
same network.
|
||||||
|
* `--es-index`
|
||||||
|
Elasticsearch index name. DEFAULT=sist2
|
||||||
|
* `-p, --print`
|
||||||
|
Print index in JSON format to stdout.
|
||||||
|
* `--script-file`
|
||||||
|
Path to user script. See [Scripting](scripting.md).
|
||||||
|
* `--mappings-file`
|
||||||
|
Path to custom Elasticsearch mappings. If none is specified, [the bundled mappings](https://github.com/simon987/sist2/tree/master/schema) will be used.
|
||||||
|
* `--settings-file`
|
||||||
|
Path to custom Elasticsearch settings. *(See above)*
|
||||||
|
* `--async-script`
|
||||||
|
Use `wait_for_completion=false` elasticsearch option while executing user script.
|
||||||
|
(See [Elasticsearch documentation](https://www.elastic.co/guide/en/elasticsearch/reference/current/tasks.html))
|
||||||
|
* `--batch-size=<int>`
|
||||||
|
Index batch size. Indexing is generally faster with larger batches, but payloads that
|
||||||
|
are too large will fail and additional overhead for retrying with smaller sizes may slow
|
||||||
|
down the process.
|
||||||
|
* `-f, --force-reset`
|
||||||
|
Reset Elasticsearch mappings and settings.
|
||||||
|
* `-t, --threads` Number of threads to use. Ideally, choose a number equal to the number of logical cores of the machine hosting Elasticsearch.
|
||||||
|
|
||||||
|
### Index examples
|
||||||
|
|
||||||
|
**Push to elasticsearch**
|
||||||
|
```bash
|
||||||
|
sist2 index --force-reset --batch-size 1000 --es-url http://localhost:9200 ./my_index/
|
||||||
|
sist2 index ./my_index/
|
||||||
|
```
|
||||||
|
|
||||||
|
**Save index in JSON format**
|
||||||
|
```bash
|
||||||
|
sist2 index --print ./my_index/ > my_index.ndjson
|
||||||
|
```
|
||||||
|
|
||||||
|
**Inspect contents of an index**
|
||||||
|
```bash
|
||||||
|
sist2 index --print ./my_index/ | jq | less
|
||||||
|
```
|
||||||
|
|
||||||
|
## Web
|
||||||
|
|
||||||
|
### Web options
|
||||||
|
* `--es-url=<str>` Elasticsearch url.
|
||||||
|
* `--es-index`
|
||||||
|
Elasticsearch index name. DEFAULT=sist2
|
||||||
|
* `--bind=<str>` Listen on this address.
|
||||||
|
* `--auth=<str>` Basic auth in user:password format
|
||||||
|
* `--tag-auth=<str>` Basic auth in user:password format. Works the same way as the
|
||||||
|
`--auth` argument, but authentication is only applied the `/tag/` endpoint.
|
||||||
|
* `--tagline=<str>` When specified, will replace the default tagline in the navbar.
|
||||||
|
* `--dev` Serve html & js files from disk (for development, used to modify frontend files without having to recompile)
|
||||||
|
|
||||||
|
### Web examples
|
||||||
|
|
||||||
|
**Single index**
|
||||||
|
```bash
|
||||||
|
sist2 web --auth admin:hunter2 --bind 0.0.0.0:8888 my_index
|
||||||
|
```
|
||||||
|
|
||||||
|
**Multiple indices**
|
||||||
|
```bash
|
||||||
|
# Indices will be displayed in this order in the web interface
|
||||||
|
sist2 web index1 index2 index3 index4
|
||||||
|
```
|
||||||
|
|
||||||
|
### rewrite_url
|
||||||
|
|
||||||
|
When the `rewrite_url` field is not empty, the web module ignores the `root`
|
||||||
|
field and will return a HTTP redirect to `<rewrite_url><path>/<name><extension>`
|
||||||
|
instead of serving the file from disk.
|
||||||
|
Both the `root` and `rewrite_url` fields are safe to manually modify from the
|
||||||
|
`descriptor.json` file.
|
||||||
|
|
||||||
|
## exec-script
|
||||||
|
|
||||||
|
The `exec-script` command is used to execute a user script for an index that has already been imported to Elasticsearch with the `index` command. Note that the documents will not be reset to their default state before each execution as the `index` command does: if you make undesired changes to the documents by accident, you will need to run `index` again to revert to the original state.
|
||||||
|
|
||||||
|
|
||||||
|
# Tagging
|
||||||
|
|
||||||
|
### Manual tagging
|
||||||
|
|
||||||
|
You can modify tags of individual documents directly from the
|
||||||
|
`web` interface. Note that you can setup authentication for this feature
|
||||||
|
with the `--tag-auth` option (See [web options](#web-options))
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Tags that are manually added are saved both in the
|
||||||
|
index folder (in `/tags/`) and in Elasticsearch*. When re-`index`ing,
|
||||||
|
they are read from the index and automatically applied.
|
||||||
|
|
||||||
|
You can safely copy the `/tags/` database to another index.
|
||||||
|
|
||||||
|
See [Automatic tagging](#automatic-tagging) for information about tag
|
||||||
|
hierarchies and tag colors.
|
||||||
|
|
||||||
|
\* *It can take a few seconds to take effect in new search queries.*
|
||||||
|
|
||||||
|
|
||||||
|
### Automatic tagging
|
||||||
|
|
||||||
|
See [scripting](scripting.md) documentation.
|
||||||
|
|
||||||
|
# Sidecar files
|
||||||
|
|
||||||
|
When scanning, sist2 will read metadata from `.s2meta` JSON files and overwrite the
|
||||||
|
original document's metadata. Sidecar metadata files will also work inside archives.
|
||||||
|
Sidecar files themselves are not saved in the index.
|
||||||
|
|
||||||
|
This feature is useful to leverage third-party applications such as speech-to-text or
|
||||||
|
OCR to add additional metadata to a file.
|
||||||
|
|
||||||
|
**Example**
|
||||||
|
|
||||||
|
```
|
||||||
|
~/Documents/
|
||||||
|
├── Video.mp4
|
||||||
|
└── Video.mp4.s2meta
|
||||||
|
```
|
||||||
|
|
||||||
|
The sidecar file must have exactly the same file path and the `.s2meta` suffix.
|
||||||
|
|
||||||
|
`Video.mp4.s2meta`:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"content": "This sidecar file will overwrite some metadata fields of Video.mp4",
|
||||||
|
"author": "Some author",
|
||||||
|
"duration": 12345,
|
||||||
|
"bitrate": 67890,
|
||||||
|
"some_arbitrary_field": [1,2,3]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
sist2 scan ~/Documents -o ./docs.idx
|
||||||
|
sist2 index ./docs.idx
|
||||||
|
```
|
||||||
|
|
||||||
|
*NOTE*: It is technically possible to overwrite the `tag` value using sidecar files, however,
|
||||||
|
it is not currently possible to restore both manual tags and sidecar tags without user scripts
|
||||||
|
while reindexing.
|
||||||
BIN
docs/genre_example.png
Normal file
BIN
docs/genre_example.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 26 KiB |
BIN
docs/manual_tag.png
Normal file
BIN
docs/manual_tag.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.9 KiB |
142
docs/scripting.md
Normal file
142
docs/scripting.md
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
## User scripts
|
||||||
|
|
||||||
|
*This document is under construction, more in-depth guide coming soon*
|
||||||
|
|
||||||
|
During the `index` step, you can use the `--script-file <script>` option to
|
||||||
|
modify documents or add user tags. This option is mainly used to
|
||||||
|
implement automatic tagging based on file attributes.
|
||||||
|
|
||||||
|
The scripting language used
|
||||||
|
([Painless Scripting Language](https://www.elastic.co/guide/en/elasticsearch/painless/7.4/index.html))
|
||||||
|
is very similar to Java, but you should be able to create user scripts
|
||||||
|
without programming experience at all if you're somewhat familiar with
|
||||||
|
regex.
|
||||||
|
|
||||||
|
This is the base structure of the documents we're working with:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"_id": "e171405c-fdb5-4feb-bb32-82637bc32084",
|
||||||
|
"_index": "sist2",
|
||||||
|
"_type": "_doc",
|
||||||
|
"_source": {
|
||||||
|
"index": "206b3050-e821-421a-891d-12fcf6c2db0d",
|
||||||
|
"mime": "application/json",
|
||||||
|
"size": 1799,
|
||||||
|
"mtime": 1545443685,
|
||||||
|
"extension": "md",
|
||||||
|
"name": "README",
|
||||||
|
"path": "sist2/scripting",
|
||||||
|
"content": "..."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Example script**
|
||||||
|
|
||||||
|
This script checks if the `genre` attribute exists, if it does
|
||||||
|
it adds the `genre.<genre>` tag.
|
||||||
|
```Java
|
||||||
|
ArrayList tags = ctx._source.tag = new ArrayList();
|
||||||
|
|
||||||
|
if (ctx._source?.genre != null) {
|
||||||
|
tags.add("genre." + ctx._source.genre.toLowerCase());
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
You can use `.` to create a hierarchical tag tree:
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
To use regular expressions, you need to add this line in `/etc/elasticsearch/elasticsearch.yml`
|
||||||
|
```yaml
|
||||||
|
script.painless.regex.enabled: true
|
||||||
|
```
|
||||||
|
Or, if you're using docker add `-e "script.painless.regex.enabled=true"`
|
||||||
|
|
||||||
|
**Tag color**
|
||||||
|
|
||||||
|
You can specify the color for an individual tag by appending an
|
||||||
|
hexadecimal color code (`#RRGGBBAA`) to the tag name.
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
If `(20XX)` is in the file name, add the `year.<year>` tag:
|
||||||
|
```Java
|
||||||
|
ArrayList tags = ctx._source.tag = new ArrayList();
|
||||||
|
|
||||||
|
Matcher m = /[\(\.+](20[0-9]{2})[\)\.+]/.matcher(ctx._source.name);
|
||||||
|
if (m.find()) {
|
||||||
|
tags.add("year." + m.group(1));
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Use default *Calibre* folder structure to infer author.
|
||||||
|
```Java
|
||||||
|
ArrayList tags = ctx._source.tag = new ArrayList();
|
||||||
|
|
||||||
|
// We expect the book path to look like this:
|
||||||
|
// /path/to/Calibre Library/Author/Title/Title - Author.pdf
|
||||||
|
|
||||||
|
if (ctx._source.name.contains("-") && ctx._source.extension == "pdf") {
|
||||||
|
String[] names = ctx._source.name.splitOnToken('-');
|
||||||
|
tags.add("author." + names[1].strip());
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
If the file matches a specific pattern `AAAA-000 fName1 lName1, <fName2 lName2>...`, add the `actress.<actress>` and
|
||||||
|
`studio.<studio>` tag:
|
||||||
|
```Java
|
||||||
|
ArrayList tags = ctx._source.tag = new ArrayList();
|
||||||
|
|
||||||
|
Matcher m = /([A-Z]{4})-[0-9]{3} (.*)/.matcher(ctx._source.name);
|
||||||
|
if (m.find()) {
|
||||||
|
tags.add("studio." + m.group(1));
|
||||||
|
|
||||||
|
// Take the matched group (.*), and add a tag for
|
||||||
|
// each name, separated by comma
|
||||||
|
for (String name : m.group(2).splitOnToken(',')) {
|
||||||
|
tags.add("actress." + name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Set the name of the last folder (`/path/to/<studio>/file.mp4`) to `studio.<studio>` tag
|
||||||
|
```Java
|
||||||
|
ArrayList tags = ctx._source.tag = new ArrayList();
|
||||||
|
|
||||||
|
if (ctx._source.path != "") {
|
||||||
|
String[] names = ctx._source.path.splitOnToken('/');
|
||||||
|
tags.add("studio." + names[names.length-1]);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Parse `EXIF:F Number` tag
|
||||||
|
```Java
|
||||||
|
if (ctx._source?.exif_fnumber != null) {
|
||||||
|
String[] values = ctx._source.exif_fnumber.splitOnToken(' ');
|
||||||
|
String aperture = String.valueOf(Float.parseFloat(values[0]) / Float.parseFloat(values[1]));
|
||||||
|
if (aperture == "NaN") {
|
||||||
|
aperture = "0,0";
|
||||||
|
}
|
||||||
|
tags.add("Aperture.f/" + aperture.replace(".", ","));
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Display year and months from `EXIF:DateTime` tag
|
||||||
|
```Java
|
||||||
|
if (ctx._source?.exif_datetime != null) {
|
||||||
|
SimpleDateFormat parser = new SimpleDateFormat("yyyy:MM:dd HH:mm:ss");
|
||||||
|
Date date = parser.parse(ctx._source.exif_datetime);
|
||||||
|
|
||||||
|
SimpleDateFormat yp = new SimpleDateFormat("yyyy");
|
||||||
|
SimpleDateFormat mp = new SimpleDateFormat("MMMMMMMMM");
|
||||||
|
|
||||||
|
String year = yp.format(date);
|
||||||
|
String month = mp.format(date);
|
||||||
|
|
||||||
|
tags.add("Month." + month);
|
||||||
|
tags.add("Year." + year);
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
BIN
docs/sist2.png
Normal file
BIN
docs/sist2.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 889 KiB |
BIN
docs/stats.png
Normal file
BIN
docs/stats.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 167 KiB |
Submodule lib/ffmpeg deleted from 0481a1f6e5
Submodule lib/harfbuzz deleted from 7cde68f10c
Submodule lib/mupdf deleted from 91782a4348
Submodule lib/onion deleted from d8d4cc9290
Submodule lib/openjpeg deleted from 5875a6b446
1
lmdb
1
lmdb
Submodule lmdb deleted from 5c012bbe03
@@ -1,31 +1,70 @@
|
|||||||
{
|
{
|
||||||
"properties": {
|
"properties": {
|
||||||
|
"_tie": {
|
||||||
|
"type": "keyword",
|
||||||
|
"doc_values": true
|
||||||
|
},
|
||||||
|
"checksum": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"_depth": {
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
"path": {
|
"path": {
|
||||||
"type": "text",
|
"type": "text",
|
||||||
"analyzer": "path_analyzer",
|
"analyzer": "path_analyzer",
|
||||||
"copy_to": "suggest-path"
|
"copy_to": "suggest-path",
|
||||||
|
"fielddata": true,
|
||||||
|
"fields": {
|
||||||
|
"nGram": {
|
||||||
|
"type": "text",
|
||||||
|
"analyzer": "my_nGram"
|
||||||
|
},
|
||||||
|
"text": {
|
||||||
|
"type": "text",
|
||||||
|
"analyzer": "content_analyzer"
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"suggest-path": {
|
"suggest-path": {
|
||||||
"type": "completion",
|
"type": "completion",
|
||||||
"analyzer": "keyword"
|
"analyzer": "case_insensitive_kw_analyzer"
|
||||||
},
|
},
|
||||||
"mime": {
|
"mime": {
|
||||||
"type": "keyword"
|
"type": "keyword"
|
||||||
},
|
},
|
||||||
|
"parent": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"thumbnail": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
"videoc": {
|
"videoc": {
|
||||||
"type": "keyword"
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
},
|
},
|
||||||
"audioc": {
|
"audioc": {
|
||||||
"type": "keyword"
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
},
|
},
|
||||||
"duration": {
|
"duration": {
|
||||||
"type": "float"
|
"type": "integer",
|
||||||
|
"index": false
|
||||||
},
|
},
|
||||||
"width": {
|
"width": {
|
||||||
"type": "integer"
|
"type": "integer",
|
||||||
|
"index": false
|
||||||
},
|
},
|
||||||
"height": {
|
"height": {
|
||||||
"type": "integer"
|
"type": "integer",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"pages": {
|
||||||
|
"type": "integer",
|
||||||
|
"index": false
|
||||||
},
|
},
|
||||||
"mtime": {
|
"mtime": {
|
||||||
"type": "integer"
|
"type": "integer"
|
||||||
@@ -70,6 +109,23 @@
|
|||||||
"analyzer": "my_nGram",
|
"analyzer": "my_nGram",
|
||||||
"type": "text"
|
"type": "text"
|
||||||
},
|
},
|
||||||
|
"_keyword.*": {
|
||||||
|
"type": "keyword"
|
||||||
|
},
|
||||||
|
"_text.*": {
|
||||||
|
"analyzer": "content_analyzer",
|
||||||
|
"type": "text",
|
||||||
|
"fields": {
|
||||||
|
"nGram": {
|
||||||
|
"type": "text",
|
||||||
|
"analyzer": "my_nGram"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"_url": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
"content": {
|
"content": {
|
||||||
"analyzer": "content_analyzer",
|
"analyzer": "content_analyzer",
|
||||||
"type": "text",
|
"type": "text",
|
||||||
@@ -80,6 +136,70 @@
|
|||||||
"analyzer": "my_nGram"
|
"analyzer": "my_nGram"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"tag": {
|
||||||
|
"type": "text",
|
||||||
|
"fielddata": true,
|
||||||
|
"analyzer": "tag_analyzer",
|
||||||
|
"copy_to": "suggest-tag"
|
||||||
|
},
|
||||||
|
"suggest-tag": {
|
||||||
|
"type": "completion",
|
||||||
|
"analyzer": "case_insensitive_kw_analyzer"
|
||||||
|
},
|
||||||
|
"exif_make": {
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
"exif_model": {
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
"exif:software": {
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
"exif_exposure_time": {
|
||||||
|
"type": "keyword"
|
||||||
|
},
|
||||||
|
"exif_fnumber": {
|
||||||
|
"type": "keyword"
|
||||||
|
},
|
||||||
|
"exif_iso_speed_ratings": {
|
||||||
|
"type": "keyword"
|
||||||
|
},
|
||||||
|
"exif_focal_length": {
|
||||||
|
"type": "keyword"
|
||||||
|
},
|
||||||
|
"exif_user_comment": {
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
"exif_gps_longitude_ref": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"exif_gps_longitude_dms": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"exif_gps_longitude_dec": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"exif_gps_latitude_ref": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"exif_gps_latitude_dms": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"exif_gps_latitude_dec": {
|
||||||
|
"type": "keyword",
|
||||||
|
"index": false
|
||||||
|
},
|
||||||
|
"author": {
|
||||||
|
"type": "text"
|
||||||
|
},
|
||||||
|
"modified_by": {
|
||||||
|
"type": "text"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
10
schema/pipeline.json
Normal file
10
schema/pipeline.json
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
"description": "Copy _id to _tie, save path depth",
|
||||||
|
"processors": [
|
||||||
|
{
|
||||||
|
"script": {
|
||||||
|
"source": "ctx._tie = ctx._id; ctx._depth = ctx.path.length() == 0 ? 0 : 1 + ctx.path.length() - ctx.path.replace(\"/\", \"\").length();"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,12 +1,19 @@
|
|||||||
{
|
{
|
||||||
"index": {
|
"index": {
|
||||||
"refresh_interval": "30s",
|
"refresh_interval": "30s",
|
||||||
"codec": "best_compression"
|
"codec": "best_compression",
|
||||||
|
"number_of_replicas": 0,
|
||||||
|
"highlight.max_analyzed_offset": 10000000
|
||||||
},
|
},
|
||||||
"analysis": {
|
"analysis": {
|
||||||
"tokenizer": {
|
"tokenizer": {
|
||||||
"path_tokenizer": {
|
"path_tokenizer": {
|
||||||
"type": "path_hierarchy"
|
"type": "path_hierarchy",
|
||||||
|
"delimiter": "/"
|
||||||
|
},
|
||||||
|
"tag_tokenizer": {
|
||||||
|
"type": "path_hierarchy",
|
||||||
|
"delimiter": "."
|
||||||
},
|
},
|
||||||
"my_nGram_tokenizer": {
|
"my_nGram_tokenizer": {
|
||||||
"type": "nGram",
|
"type": "nGram",
|
||||||
@@ -21,6 +28,18 @@
|
|||||||
"lowercase"
|
"lowercase"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"tag_analyzer": {
|
||||||
|
"tokenizer": "tag_tokenizer",
|
||||||
|
"filter": [
|
||||||
|
"lowercase"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"case_insensitive_kw_analyzer": {
|
||||||
|
"tokenizer": "keyword",
|
||||||
|
"filter": [
|
||||||
|
"lowercase"
|
||||||
|
]
|
||||||
|
},
|
||||||
"my_nGram": {
|
"my_nGram": {
|
||||||
"tokenizer": "my_nGram_tokenizer",
|
"tokenizer": "my_nGram_tokenizer",
|
||||||
"filter": [
|
"filter": [
|
||||||
|
|||||||
@@ -2,16 +2,9 @@
|
|||||||
|
|
||||||
rm -rf index.sist2/
|
rm -rf index.sist2/
|
||||||
|
|
||||||
rm web/js/bundle.js 2> /dev/null
|
|
||||||
cat `ls web/js/*.min.js` > web/js/bundle.js
|
|
||||||
cat web/js/{util,dom,search}.js >> web/js/bundle.js
|
|
||||||
|
|
||||||
rm web/css/bundle*.css 2> /dev/null
|
|
||||||
cat web/css/*.min.css > web/css/bundle.css
|
|
||||||
cat web/css/light.css >> web/css/bundle.css
|
|
||||||
cat web/css/*.min.css > web/css/bundle_dark.css
|
|
||||||
cat web/css/dark.css >> web/css/bundle_dark.css
|
|
||||||
|
|
||||||
python3 scripts/mime.py > src/parsing/mime_generated.c
|
python3 scripts/mime.py > src/parsing/mime_generated.c
|
||||||
python3 scripts/serve_static.py > src/web/static_generated.c
|
python3 scripts/serve_static.py > src/web/static_generated.c
|
||||||
python3 scripts/index_static.py > src/index/static_generated.c
|
python3 scripts/index_static.py > src/index/static_generated.c
|
||||||
|
|
||||||
|
printf "static const char *const Sist2CommitHash = \"%s\";\n" $(git rev-parse HEAD) > src/git_hash.h
|
||||||
|
printf "static const char *const LibScanCommitHash = \"%s\";\n" $(cd third-party/libscan/ && git rev-parse HEAD) >> src/git_hash.h
|
||||||
|
|||||||
@@ -1,73 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
cd lib
|
|
||||||
|
|
||||||
cd mupdf
|
|
||||||
USE_SYSTEM_HARFBUZZ=yes USE_SYSTEM_OPENJPEG=yes HAVE_X11=no HAVE_GLUT=no make -j 4
|
|
||||||
cd ..
|
|
||||||
|
|
||||||
mv mupdf/build/release/libmupdf.a .
|
|
||||||
mv mupdf/build/release/libmupdf-third.a .
|
|
||||||
|
|
||||||
# openjp2
|
|
||||||
cd openjpeg
|
|
||||||
#cmake . -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_FLAGS="-O3 -march=native -DNDEBUG"
|
|
||||||
cmake . -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_FLAGS="-O3"
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
mv openjpeg/bin/libopenjp2.a .
|
|
||||||
|
|
||||||
# harfbuzz
|
|
||||||
cd harfbuzz
|
|
||||||
./autogen.sh
|
|
||||||
./configure --disable-shared --enable-static
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
mv harfbuzz/src/.libs/libharfbuzz.a .
|
|
||||||
|
|
||||||
# ffmpeg
|
|
||||||
cd ffmpeg
|
|
||||||
./configure --disable-shared --enable-static --disable-ffmpeg --disable-ffplay \
|
|
||||||
--disable-ffprobe --disable-doc\
|
|
||||||
--disable-manpages --disable-postproc --disable-avfilter \
|
|
||||||
--disable-alsa --disable-lzma --disable-xlib --disable-debug\
|
|
||||||
--disable-vdpau --disable-vaapi --disable-sdl2 --disable-network
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
|
|
||||||
mv ffmpeg/libavcodec/libavcodec.a .
|
|
||||||
mv ffmpeg/libavformat/libavformat.a .
|
|
||||||
mv ffmpeg/libavutil/libavutil.a .
|
|
||||||
mv ffmpeg/libswresample/libswresample.a .
|
|
||||||
mv ffmpeg/libswscale/libswscale.a .
|
|
||||||
|
|
||||||
# onion
|
|
||||||
cd onion
|
|
||||||
mkdir build 2> /dev/null
|
|
||||||
cd build
|
|
||||||
cmake -DONION_USE_SSL=false -DONION_USE_PAM=false -DONION_USE_PNG=false -DONION_USE_JPEG=false \
|
|
||||||
-DONION_USE_JPEG=false -DONION_USE_XML2=false -DONION_USE_SYSTEMD=false -DONION_USE_SQLITE3=false \
|
|
||||||
-DONION_USE_REDIS=false -DONION_USE_GC=false -DONION_USE_TESTS=false -DONION_EXAMPLES=false \
|
|
||||||
-DONION_USE_BINDINGS_CPP=false ..
|
|
||||||
make -j 4
|
|
||||||
cd ../..
|
|
||||||
|
|
||||||
mv onion/build/src/onion/libonion_static.a .
|
|
||||||
|
|
||||||
#bzip2
|
|
||||||
git clone https://github.com/enthought/bzip2-1.0.6
|
|
||||||
cd bzip2-1.0.6
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
mv bzip2-1.0.6/libbz2.a .
|
|
||||||
|
|
||||||
# magic
|
|
||||||
git clone https://github.com/threatstack/libmagic
|
|
||||||
cd libmagic
|
|
||||||
./autogen.sh
|
|
||||||
./configure --enable-static --disable-shared
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
mv libmagic/src/.libs/libmagic.a .
|
|
||||||
|
|
||||||
|
|
||||||
cd ..
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
cd lib
|
|
||||||
|
|
||||||
# mupdf
|
|
||||||
cd mupdf
|
|
||||||
HAVE_X11=no HAVE_GLUT=no gmake -j 4
|
|
||||||
cd ..
|
|
||||||
|
|
||||||
mv mupdf/build/release/libmupdf.a .
|
|
||||||
mv mupdf/build/release/libmupdf-third.a .
|
|
||||||
|
|
||||||
# openjp2
|
|
||||||
cd openjpeg
|
|
||||||
#cmake . -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_FLAGS="-O3 -march=native -DNDEBUG"
|
|
||||||
cmake . -DCMAKE_BUILD_TYPE=Release -DCMAKE_C_FLAGS="-O3"
|
|
||||||
gmake -j 4
|
|
||||||
cd ..
|
|
||||||
mv openjpeg/bin/libopenjp2.a .
|
|
||||||
|
|
||||||
# harfbuzz
|
|
||||||
cd harfbuzz
|
|
||||||
./autogen.sh
|
|
||||||
./configure --disable-shared --enable-static
|
|
||||||
gmake -j 4
|
|
||||||
cd ..
|
|
||||||
mv harfbuzz/src/.libs/libharfbuzz.a .
|
|
||||||
|
|
||||||
# ffmpeg
|
|
||||||
cd ffmpeg
|
|
||||||
./configure --disable-shared --enable-static --disable-ffmpeg --disable-ffplay \
|
|
||||||
--disable-ffprobe --disable-doc\
|
|
||||||
--disable-manpages --disable-postproc --disable-avfilter \
|
|
||||||
--disable-alsa --disable-lzma --disable-xlib --disable-debug\
|
|
||||||
--disable-vdpau --disable-vaapi --disable-sdl2 --disable-network
|
|
||||||
gmake -j 4
|
|
||||||
cd ..
|
|
||||||
|
|
||||||
mv ffmpeg/libavcodec/libavcodec.a .
|
|
||||||
mv ffmpeg/libavformat/libavformat.a .
|
|
||||||
mv ffmpeg/libavutil/libavutil.a .
|
|
||||||
mv ffmpeg/libswresample/libswresample.a .
|
|
||||||
mv ffmpeg/libswscale/libswscale.a .
|
|
||||||
|
|
||||||
#bzip2
|
|
||||||
git clone https://github.com/enthought/bzip2-1.0.6
|
|
||||||
cd bzip2-1.0.6
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
mv bzip2-1.0.6/libbz2.a .
|
|
||||||
|
|
||||||
# magic
|
|
||||||
git clone https://github.com/threatstack/libmagic
|
|
||||||
cd libmagic
|
|
||||||
./autogen.sh
|
|
||||||
./configure --enable-static --disable-shared
|
|
||||||
make -j 4
|
|
||||||
cd ..
|
|
||||||
mv libmagic/src/.libs/libmagic.a .
|
|
||||||
|
|
||||||
cd ..
|
|
||||||
@@ -1,6 +1,9 @@
|
|||||||
|
import json
|
||||||
|
|
||||||
files = [
|
files = [
|
||||||
"schema/mappings.json",
|
"schema/mappings.json",
|
||||||
"schema/settings.json",
|
"schema/settings.json",
|
||||||
|
"schema/pipeline.json",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
@@ -9,6 +12,7 @@ def clean(filepath):
|
|||||||
|
|
||||||
|
|
||||||
for file in files:
|
for file in files:
|
||||||
with open(file, "rb") as f:
|
with open(file, "r") as f:
|
||||||
data = f.read()
|
data = json.dumps(json.load(f), separators=(",", ":")).encode()
|
||||||
|
data += b'\0'
|
||||||
print("char %s[%d] = {%s};" % (clean(file), len(data), ",".join(str(int(b)) for b in data)))
|
print("char %s[%d] = {%s};" % (clean(file), len(data), ",".join(str(int(b)) for b in data)))
|
||||||
|
|||||||
@@ -2,14 +2,18 @@ application/arj, arj
|
|||||||
application/base64, mme
|
application/base64, mme
|
||||||
application/binhex, hqx
|
application/binhex, hqx
|
||||||
application/book, boo|book
|
application/book, boo|book
|
||||||
|
application/CDFV2-corrupt,
|
||||||
application/CDFV2, sdv
|
application/CDFV2, sdv
|
||||||
application/clariscad, ccad
|
application/clariscad, ccad
|
||||||
application/commonground, dp
|
application/commonground, dp
|
||||||
|
application/csv,
|
||||||
|
application/dicom, dcm
|
||||||
application/drafting, drw
|
application/drafting, drw
|
||||||
|
application/epub+zip, epub
|
||||||
application/freeloader, frl
|
application/freeloader, frl
|
||||||
application/futuresplash, spl
|
application/futuresplash, spl
|
||||||
application/groupwise, vew
|
application/groupwise, vew
|
||||||
application/gzip, gz
|
application/gzip, gz|tgz
|
||||||
application/hta, hta
|
application/hta, hta
|
||||||
application/i-deas, unv
|
application/i-deas, unv
|
||||||
application/iges, iges|igs
|
application/iges, iges|igs
|
||||||
@@ -17,8 +21,8 @@ application/inf, inf
|
|||||||
application/java-archive, jar
|
application/java-archive, jar
|
||||||
application/java, class
|
application/java, class
|
||||||
application/javascript,
|
application/javascript,
|
||||||
application/x-archive, a
|
|
||||||
application/json, json
|
application/json, json
|
||||||
|
application/ndjson, jsonl|ndjson
|
||||||
application/marc, mrc
|
application/marc, mrc
|
||||||
application/mbedlet, mbd
|
application/mbedlet, mbd
|
||||||
application/mime, aps
|
application/mime, aps
|
||||||
@@ -27,7 +31,9 @@ application/msword, doc|dot|w6w|wiz|word
|
|||||||
application/netmc, mcp
|
application/netmc, mcp
|
||||||
application/octet-stream, bin|dump|gpg
|
application/octet-stream, bin|dump|gpg
|
||||||
application/oda, oda
|
application/oda, oda
|
||||||
|
application/ogg, ogv
|
||||||
application/pdf, pdf
|
application/pdf, pdf
|
||||||
|
application/pgp-keys,
|
||||||
application/pgp-signature, pgp
|
application/pgp-signature, pgp
|
||||||
application/pkcs7-signature, p7s
|
application/pkcs7-signature, p7s
|
||||||
application/pkix-cert, cer|crt
|
application/pkix-cert, cer|crt
|
||||||
@@ -43,6 +49,10 @@ application/vda, vda
|
|||||||
application/vnd.fdf, fdf
|
application/vnd.fdf, fdf
|
||||||
application/vnd.font-fontforge-sfd, sfd
|
application/vnd.font-fontforge-sfd, sfd
|
||||||
application/vnd.hp-hpgl, hgl|hpg|hpgl
|
application/vnd.hp-hpgl, hgl|hpg|hpgl
|
||||||
|
application/vnd.iccprofile, icm
|
||||||
|
application/vnd.iccprofile, icm
|
||||||
|
application/vnd.lotus-1-2-3,
|
||||||
|
application/vnd.ms-cab-compressed, cab
|
||||||
application/vnd.ms-excel, xlb|xlc|xll|xlm|xls|xlw
|
application/vnd.ms-excel, xlb|xlc|xll|xlm|xls|xlw
|
||||||
application/vnd.ms-fontobject, eot
|
application/vnd.ms-fontobject, eot
|
||||||
application/vnd.ms-opentype, otf
|
application/vnd.ms-opentype, otf
|
||||||
@@ -54,45 +64,73 @@ application/vnd.ms-project, mpp
|
|||||||
application/vnd.oasis.opendocument.base, odb
|
application/vnd.oasis.opendocument.base, odb
|
||||||
application/vnd.oasis.opendocument.formula, odf
|
application/vnd.oasis.opendocument.formula, odf
|
||||||
application/vnd.oasis.opendocument.graphics, odg
|
application/vnd.oasis.opendocument.graphics, odg
|
||||||
|
application/vnd.oasis.opendocument.presentation, odp
|
||||||
|
application/vnd.oasis.opendocument.spreadsheet, ods
|
||||||
application/vnd.oasis.opendocument.text, odt
|
application/vnd.oasis.opendocument.text, odt
|
||||||
|
application/vnd.openxmlformats-officedocument.presentationml.presentation, pptx
|
||||||
|
application/vnd.openxmlformats-officedocument.spreadsheetml.sheet, xlsx
|
||||||
|
application/vnd.openxmlformats-officedocument.wordprocessingml.document, docx
|
||||||
|
application/vnd.symbian.install,
|
||||||
|
application/vnd.tcpdump.pcap, pcap
|
||||||
application/vnd.wap.wmlc, wmlc
|
application/vnd.wap.wmlc, wmlc
|
||||||
application/vnd.wap.wmlscriptc, wmlsc
|
application/vnd.wap.wmlscriptc, wmlsc
|
||||||
application/vnd.xara, web
|
application/vnd.xara, web
|
||||||
application/vocaltec-media-desc, vmd
|
application/vocaltec-media-desc, vmd
|
||||||
application/vocaltec-media-file, vmf
|
application/vocaltec-media-file, vmf
|
||||||
application/wordperfect6.0, w60
|
application/warc, warc
|
||||||
application/wordperfect6.1, w61
|
application/winhelp, hlp
|
||||||
application/wordperfect, wp|wp5|wp6|wpd
|
application/wordperfect, wp|wp5|wp6|wpd|w60|w61
|
||||||
application/x-123, wk1
|
application/x-123, wk1
|
||||||
|
application/x-7z-compressed, 7z
|
||||||
application/x-aim, aim
|
application/x-aim, aim
|
||||||
|
application/x-apple-diskimage,
|
||||||
|
application/x-arc,
|
||||||
|
application/x-archive, a
|
||||||
|
application/x-atari-7800-rom, a78
|
||||||
application/x-authorware-bin, aab
|
application/x-authorware-bin, aab
|
||||||
application/x-authorware-map, aam
|
application/x-authorware-map, aam
|
||||||
application/x-authorware-seg, aas
|
application/x-authorware-seg, aas
|
||||||
|
application/x-avira-qua,
|
||||||
application/x-bcpio, bcpio
|
application/x-bcpio, bcpio
|
||||||
application/x-bittorrent, torrent
|
application/x-bittorrent, torrent
|
||||||
application/x-bsh, bsh
|
application/x-bsh, bsh
|
||||||
application/x-bytecode.python, pyc
|
application/x-bytecode.python, pyc
|
||||||
application/x-bzip2, boz|bz2
|
application/x-bzip2, boz|bz2
|
||||||
application/x-bzip, bz
|
application/x-bzip, bz
|
||||||
|
application/x-cbr, cbr
|
||||||
|
application/x-cbz, cbz
|
||||||
application/x-cdlink, vcd
|
application/x-cdlink, vcd
|
||||||
application/x-chat, cha|chat
|
application/x-chat, cha|chat
|
||||||
|
application/x-chrome-extension,
|
||||||
application/x-cocoa, cco
|
application/x-cocoa, cco
|
||||||
application/x-conference, nsc
|
application/x-conference, nsc
|
||||||
|
application/x-coredump,
|
||||||
application/x-cpio, cpio
|
application/x-cpio, cpio
|
||||||
application/x-dbf, dbf
|
application/x-dbf, dbf
|
||||||
application/x-dbt,
|
application/x-dbt,
|
||||||
|
application/x-debian-package, deb
|
||||||
application/x-deepv, deepv
|
application/x-deepv, deepv
|
||||||
application/x-director, dcr|dir|dxr
|
application/x-director, dir|dxr
|
||||||
|
application/x-dmp, dmp
|
||||||
|
application/x-dosdriver,
|
||||||
application/x-dosexec, dll
|
application/x-dosexec, dll
|
||||||
application/x-dvi, dvi
|
application/x-dvi, dvi
|
||||||
application/x-elc, elc
|
application/x-elc, elc
|
||||||
|
application/x-empty,
|
||||||
application/x-envoy, env|evy
|
application/x-envoy, env|evy
|
||||||
application/x-esrehber, es
|
application/x-esrehber, es
|
||||||
application/x-excel, xla|xld|xlk|xlt|xlv
|
application/x-excel, xla|xld|xlk|xlt|xlv
|
||||||
application/x-executable, exe
|
application/x-executable, exe
|
||||||
|
application/x-font-gdos,
|
||||||
|
application/x-font-pf2, pf2
|
||||||
|
application/x-font-pfm, pfm
|
||||||
application/x-font-sfn,
|
application/x-font-sfn,
|
||||||
application/x-font-ttf, ttf|ttc
|
application/x-font-ttf, ttf|ttc
|
||||||
|
application/x-fptapplication/x-dbt,
|
||||||
application/x-freelance, pre
|
application/x-freelance, pre
|
||||||
|
application/x-gamecube-rom,
|
||||||
|
application/x-gdbm,
|
||||||
|
application/x-gettext-translation,
|
||||||
application/x-git,
|
application/x-git,
|
||||||
application/x-gsp, gsp
|
application/x-gsp, gsp
|
||||||
application/x-gss, gss
|
application/x-gss, gss
|
||||||
@@ -102,46 +140,67 @@ application/x-hdf, hdf
|
|||||||
application/x-helpfile, help
|
application/x-helpfile, help
|
||||||
application/x-httpd-imap, imap
|
application/x-httpd-imap, imap
|
||||||
application/x-ima, ima
|
application/x-ima, ima
|
||||||
|
application/x-innosetup,
|
||||||
application/x-internett-signup, ins
|
application/x-internett-signup, ins
|
||||||
application/x-inventor, iv
|
application/x-inventor, iv
|
||||||
application/x-ip2, ip
|
application/x-ip2, ip
|
||||||
application/x-java-applet,
|
application/x-java-applet,
|
||||||
application/x-java-commerce, jcm
|
application/x-java-commerce, jcm
|
||||||
application/x-java-image,
|
application/x-java-image,
|
||||||
|
application/x-java-jmod, jmod
|
||||||
application/x-java-keystore,
|
application/x-java-keystore,
|
||||||
|
application/x-kdelnk,
|
||||||
application/x-koan, skd|skm|skp|skt
|
application/x-koan, skd|skm|skp|skt
|
||||||
application/x-latex, latex|ltx
|
application/x-latex, latex|ltx
|
||||||
application/x-livescreen, ivy
|
application/x-livescreen, ivy
|
||||||
application/x-lotus, wq1
|
application/x-lotus, wq1
|
||||||
|
application/x-lz4+json, jsonlz4
|
||||||
|
application/x-lz4, lz4
|
||||||
|
application/x-lzh-compressed,
|
||||||
application/x-lzh, lzh
|
application/x-lzh, lzh
|
||||||
|
application/x-lzip, lz
|
||||||
|
application/x-lzma, lzma
|
||||||
|
application/x-lzop, lzo
|
||||||
application/x-lzx, lzx
|
application/x-lzx, lzx
|
||||||
application/x-mach-binary, jnilib|dylib
|
application/x-mach-binary, jnilib|dylib
|
||||||
application/x-mach-executable,
|
application/x-mach-executable,
|
||||||
application/x-magic-cap-package-1.0, mc$
|
application/x-magic-cap-package-1.0, mc$
|
||||||
application/x-mathcad, mcd
|
application/x-mathcad, mcd
|
||||||
|
application/x-maxis-dbpf,
|
||||||
application/x-meme, mm
|
application/x-meme, mm
|
||||||
application/x-midi, midi
|
application/x-midi, midi
|
||||||
application/x-mif, mif
|
application/x-mif, mif
|
||||||
application/x-mix-transfer, nix
|
application/x-mix-transfer, nix
|
||||||
application/xml, opf
|
application/xml, opf
|
||||||
|
application/x-mobipocket-ebook, mobi
|
||||||
|
application/vnd.amazon.mobi8-ebook, azw|azw3
|
||||||
|
application/x-msaccess, accdb
|
||||||
|
application/x-ms-compress-szdd, fon
|
||||||
application/x-ms-pdb, pdb
|
application/x-ms-pdb, pdb
|
||||||
|
application/x-ms-reader, lit
|
||||||
|
application/x-n64-rom, z64
|
||||||
application/x-navi-animation, ani
|
application/x-navi-animation, ani
|
||||||
application/x-navidoc, nvd
|
application/x-navidoc, nvd
|
||||||
application/x-navimap, map
|
application/x-navimap, map
|
||||||
application/x-navistyle, stl
|
application/x-navistyle, stl
|
||||||
|
application/x-nes-rom, nes
|
||||||
application/x-netcdf, cdf|nc
|
application/x-netcdf, cdf|nc
|
||||||
application/x-newton-compatible-pkg, pkg
|
application/x-newton-compatible-pkg, pkg
|
||||||
|
application/x-nintendo-ds-rom,
|
||||||
application/x-object, o
|
application/x-object, o
|
||||||
application/x-omcdatamaker, omcd
|
application/x-omcdatamaker, omcd
|
||||||
application/x-omc, omc
|
application/x-omc, omc
|
||||||
application/x-omcregerator, omcr
|
application/x-omcregerator, omcr
|
||||||
application/x-pagemaker, pm4|pm5
|
application/x-pagemaker, pm4|pm5
|
||||||
application/x-pcl, pcl
|
application/x-pcl, pcl
|
||||||
|
application/x-pgp-keyring,
|
||||||
application/x-pixclscript, plx
|
application/x-pixclscript, plx
|
||||||
application/x-pkcs7-certreqresp, p7r
|
application/x-pkcs7-certreqresp, p7r
|
||||||
application/x-pkcs7-signature, p7a
|
application/x-pkcs7-signature, p7a
|
||||||
application/x-project, mpc|mpt|mpv|mpx
|
application/x-project, mpc|mpt|mpv|mpx
|
||||||
application/x-qpro, wb1
|
application/x-qpro, wb1
|
||||||
|
application/x-rar, rar
|
||||||
|
application/x-rpm, rpm
|
||||||
application/x-sdp, sdp
|
application/x-sdp, sdp
|
||||||
application/x-sea, sea
|
application/x-sea, sea
|
||||||
application/x-seelogo, sl
|
application/x-seelogo, sl
|
||||||
@@ -149,12 +208,17 @@ application/x-setupscript,
|
|||||||
application/x-sharedlib, so
|
application/x-sharedlib, so
|
||||||
application/x-shar, shar
|
application/x-shar, shar
|
||||||
application/x-shockwave-flash, swf
|
application/x-shockwave-flash, swf
|
||||||
|
application/x-snappy-framed,
|
||||||
application/x-sprite, spr|sprite
|
application/x-sprite, spr|sprite
|
||||||
application/x-sqlite3,
|
application/x-sqlite3,
|
||||||
|
application/x-stargallery-thm,
|
||||||
|
application/x-stuffit, sit
|
||||||
application/x-sv4cpio, sv4cpio
|
application/x-sv4cpio, sv4cpio
|
||||||
application/x-sv4crc, sv4crc
|
application/x-sv4crc, sv4crc
|
||||||
application/x-tar, tar
|
application/x-tar, tar
|
||||||
application/x-tbook, sbk|tbk
|
application/x-tbook, sbk|tbk
|
||||||
|
application/x-terminfo,
|
||||||
|
application/x-terminfo2,
|
||||||
application/x-texinfo, texi|texinfo
|
application/x-texinfo, texi|texinfo
|
||||||
application/x-tex-tfm, tfm
|
application/x-tex-tfm, tfm
|
||||||
application/x-ustar, ustar
|
application/x-ustar, ustar
|
||||||
@@ -163,16 +227,22 @@ application/x-vnd.audioexplosion.mzz, mzz
|
|||||||
application/x-vnd.ls-xpix, xpix
|
application/x-vnd.ls-xpix, xpix
|
||||||
application/x-vrml, vrml
|
application/x-vrml, vrml
|
||||||
application/x-wais-source, src|wsrc
|
application/x-wais-source, src|wsrc
|
||||||
|
application/x-wine-extension-ini,
|
||||||
application/x-wintalk, wtk
|
application/x-wintalk, wtk
|
||||||
application/x-world, svr
|
application/x-world, svr
|
||||||
application/x-wri, wri
|
application/x-wri, wri
|
||||||
application/x-x509-ca-cert, der
|
application/x-x509-ca-cert, der
|
||||||
application/x-xz, xz
|
application/x-xz, xz
|
||||||
|
application/x-zip,
|
||||||
|
application/x-zstd, zst
|
||||||
application/zip, zip
|
application/zip, zip
|
||||||
|
application/zlib, z
|
||||||
|
!audio/basic, au
|
||||||
audio/it, it
|
audio/it, it
|
||||||
audio/make, funk|my|pfunk
|
audio/make, funk|my|pfunk
|
||||||
audio/midi, kar
|
audio/midi, kar
|
||||||
audio/mid, rmi
|
audio/mid, rmi
|
||||||
|
audio/mp4, m4b
|
||||||
audio/mpeg, m2a|mpa
|
audio/mpeg, m2a|mpa
|
||||||
audio/ogg, ogg
|
audio/ogg, ogg
|
||||||
audio/s3m, s3m
|
audio/s3m, s3m
|
||||||
@@ -180,7 +250,10 @@ audio/tsp-audio, tsi
|
|||||||
audio/tsplayer, tsp
|
audio/tsplayer, tsp
|
||||||
audio/vnd.qcelp, qcp
|
audio/vnd.qcelp, qcp
|
||||||
audio/voxware, vox
|
audio/voxware, vox
|
||||||
|
audio/x-aiff, aiff|aif
|
||||||
|
audio/x-flac, flac
|
||||||
audio/x-gsm, gsd|gsm
|
audio/x-gsm, gsd|gsm
|
||||||
|
audio/x-hx-aac-adts,
|
||||||
audio/x-jam, jam
|
audio/x-jam, jam
|
||||||
audio/x-liveaudio, lam
|
audio/x-liveaudio, lam
|
||||||
audio/x-m4a, m4a
|
audio/x-m4a, m4a
|
||||||
@@ -194,17 +267,24 @@ audio/x-nspaudio, lma
|
|||||||
audio/x-pn-realaudio, ram|rm|rmm|rmp
|
audio/x-pn-realaudio, ram|rm|rmm|rmp
|
||||||
audio/x-psid, sid
|
audio/x-psid, sid
|
||||||
audio/x-realaudio, ra
|
audio/x-realaudio, ra
|
||||||
|
audio/x-s3m,
|
||||||
audio/x-twinvq-plugin, vqe|vql
|
audio/x-twinvq-plugin, vqe|vql
|
||||||
audio/x-twinvq, vqf
|
audio/x-twinvq, vqf
|
||||||
audio/x-voc, voc
|
audio/x-voc, voc
|
||||||
audio/x-wav, wav
|
audio/x-wav, wav
|
||||||
|
!audio/x-xbox360-executable, xex
|
||||||
|
!audio/x-xbox-executable, xbe
|
||||||
font/otf,
|
font/otf,
|
||||||
font/sfnt,
|
font/sfnt,
|
||||||
|
font/woff2, woff2
|
||||||
|
font/woff, woff
|
||||||
|
image/bmp,
|
||||||
image/cmu-raster, rast
|
image/cmu-raster, rast
|
||||||
image/fif, fif
|
image/fif, fif
|
||||||
image/florian, flo|turbot
|
image/florian, flo|turbot
|
||||||
image/g3fax, g3
|
image/g3fax, g3
|
||||||
image/gif, gif
|
image/gif, gif
|
||||||
|
image/heic, heic
|
||||||
image/ief, ief|iefs
|
image/ief, ief|iefs
|
||||||
image/jpeg, jfif|jfif-tbnl|jpe|jpeg|jpg
|
image/jpeg, jfif|jfif-tbnl|jpe|jpeg|jpg
|
||||||
image/jutvision, jut
|
image/jutvision, jut
|
||||||
@@ -213,6 +293,9 @@ image/pict, pic|pict
|
|||||||
image/png, png|x-png
|
image/png, png|x-png
|
||||||
!image/svg, svg
|
!image/svg, svg
|
||||||
!image/svg+xml,
|
!image/svg+xml,
|
||||||
|
image/tiff,
|
||||||
|
!image/vnd.adobe.photoshop, psd
|
||||||
|
!image/vnd.djvu, djvu
|
||||||
image/vnd.fpx, fpx
|
image/vnd.fpx, fpx
|
||||||
image/vnd.microsoft.icon,
|
image/vnd.microsoft.icon,
|
||||||
image/vnd.rn-realflash, rf
|
image/vnd.rn-realflash, rf
|
||||||
@@ -220,9 +303,15 @@ image/vnd.rn-realpix, rp
|
|||||||
image/vnd.wap.wbmp, wbmp
|
image/vnd.wap.wbmp, wbmp
|
||||||
image/vnd.xiff, xif
|
image/vnd.xiff, xif
|
||||||
image/webp, webp
|
image/webp, webp
|
||||||
|
image/wmf,
|
||||||
|
image/x-3ds, 3ds
|
||||||
|
image/x-award-bioslogo,
|
||||||
image/x-cmu-raster, ras
|
image/x-cmu-raster, ras
|
||||||
|
image/x-cur, tga
|
||||||
image/x-dwg, dwg|dxf|svf
|
image/x-dwg, dwg|dxf|svf
|
||||||
image/x-eps,
|
image/x-eps,
|
||||||
|
image/x-exr, exr
|
||||||
|
image/x-gem,
|
||||||
image/x-icns,
|
image/x-icns,
|
||||||
!image/x-icon, ico
|
!image/x-icon, ico
|
||||||
image/x-jg, art
|
image/x-jg, art
|
||||||
@@ -236,32 +325,31 @@ image/x-portable-graymap, pgm
|
|||||||
image/x-portable-pixmap, ppm
|
image/x-portable-pixmap, ppm
|
||||||
image/x-quicktime, qif|qti|qtif
|
image/x-quicktime, qif|qti|qtif
|
||||||
image/x-rgb, rgb
|
image/x-rgb, rgb
|
||||||
|
image/x-tga,
|
||||||
image/x-tiff, tif|tiff
|
image/x-tiff, tif|tiff
|
||||||
image/tiff,
|
image/x-win-bitmap,
|
||||||
!image/x-xcf, xcf
|
!image/x-xcf, xcf
|
||||||
!image/x-xpixmap, xpm
|
!image/x-xpixmap, xpm
|
||||||
|
image/x-xwindowdump, xwd
|
||||||
|
message/news,
|
||||||
message/rfc822, mht|mhtml|mime
|
message/rfc822, mht|mhtml|mime
|
||||||
model/vnd.dwf, dwf
|
model/vnd.dwf, dwf
|
||||||
|
model/vnd.gdl, gdl
|
||||||
|
model/vnd.gs.gdl, gdsl
|
||||||
model/vrml, wrz
|
model/vrml, wrz
|
||||||
model/x-pov, pov
|
model/x-pov, pov
|
||||||
text/asp, asp
|
text/asp, asp
|
||||||
text/css, css
|
text/css, css
|
||||||
text/x-sass, sass
|
|
||||||
text/x-scss, scss
|
|
||||||
text/html, acgi|htm|html|htmls|htx|shtml
|
text/html, acgi|htm|html|htmls|htx|shtml
|
||||||
text/javascript, js
|
text/javascript, js
|
||||||
text/mcf, mcf
|
text/mcf, mcf
|
||||||
text/pascal, pas
|
text/pascal, pas
|
||||||
text/plain, com|cmd|conf|def|g|idc|list|lst|mar|sdml|text|txt|md|groovy|license|properties|desktop|ini|rst|cmake|ipynb|readme|less|lo|go|yml|d|cs|hpp|srt
|
text/PGP,
|
||||||
|
text/plain, com|cmd|conf|def|g|idc|list|lst|mar|sdml|text|txt|md|groovy|license|properties|desktop|ini|rst|cmake|ipynb|readme|less|lo|go|yml|d|cs|hpp|srt|nfo|sfv|m3u|csv|eml|make|log|markdown|yaml
|
||||||
|
application/vnd.coffeescript, coffee
|
||||||
text/richtext, rt|rtf|rtx
|
text/richtext, rt|rtf|rtx
|
||||||
text/rtf,
|
text/rtf,
|
||||||
text/scriplet, wsc
|
text/scriplet, wsc
|
||||||
text/x-awk, awk
|
|
||||||
!video/x-jng, jng
|
|
||||||
video/x-mng, mng
|
|
||||||
image/x-cur, tga
|
|
||||||
image/x-xwindowdump, xwd
|
|
||||||
!image/vnd.adobe.photoshop, psd
|
|
||||||
text/tab-separated-values, tsv
|
text/tab-separated-values, tsv
|
||||||
text/troff, man|me|ms|roff|t|tr
|
text/troff, man|me|ms|roff|t|tr
|
||||||
text/uri-list, uji|unis|uri|uris
|
text/uri-list, uji|unis|uri|uris
|
||||||
@@ -273,6 +361,7 @@ text/webviewhtml, htt
|
|||||||
text/x-Algol68,
|
text/x-Algol68,
|
||||||
text/x-asm, asm|s
|
text/x-asm, asm|s
|
||||||
text/x-audiosoft-intra, aip
|
text/x-audiosoft-intra, aip
|
||||||
|
text/x-awk, awk
|
||||||
text/x-bcpl,
|
text/x-bcpl,
|
||||||
text/x-c, c|cc|h
|
text/x-c, c|cc|h
|
||||||
text/x-c++, cpp|cxx|c++
|
text/x-c++, cpp|cxx|c++
|
||||||
@@ -287,23 +376,31 @@ text/x-makefile, am|mak
|
|||||||
text/xml, xml|pom|iml|plist
|
text/xml, xml|pom|iml|plist
|
||||||
text/x-m, m
|
text/x-m, m
|
||||||
text/x-msdos-batch, bat
|
text/x-msdos-batch, bat
|
||||||
|
text/x-ms-regedit, reg
|
||||||
|
text/x-objective-c,
|
||||||
text/x-pascal, p
|
text/x-pascal, p
|
||||||
text/x-perl, pl
|
text/x-perl, pl
|
||||||
text/x-php, php
|
text/x-php, php
|
||||||
|
text/x-po, po
|
||||||
text/x-python, py
|
text/x-python, py
|
||||||
text/x-ruby, rb
|
text/x-ruby, rb
|
||||||
|
text/x-sass, sass
|
||||||
|
text/x-scss, scss
|
||||||
text/x-server-parsed-html, ssi
|
text/x-server-parsed-html, ssi
|
||||||
text/x-setext, etx
|
text/x-setext, etx
|
||||||
text/x-sgml, sgm|sgml
|
text/x-sgml, sgm|sgml
|
||||||
text/x-shellscript, sh
|
text/x-shellscript, sh
|
||||||
text/x-speech, talk
|
text/x-speech, talk
|
||||||
|
text/x-tcl,
|
||||||
text/x-tex, tex
|
text/x-tex, tex
|
||||||
text/x-uil, uil
|
text/x-uil, uil
|
||||||
text/x-uuencode, uue
|
text/x-uuencode, uue
|
||||||
text/x-vcalendar, vcs
|
text/x-vcalendar, vcs
|
||||||
|
text/x-vcard, vcf
|
||||||
video/animaflex, afl
|
video/animaflex, afl
|
||||||
video/avi, avi
|
video/avi, avi
|
||||||
video/avs-video, avs
|
video/avs-video, avs
|
||||||
|
video/MP2T,
|
||||||
video/mp4, mp4
|
video/mp4, mp4
|
||||||
video/mpeg, m1v|m2v|mpe|mpeg|mpg
|
video/mpeg, m1v|m2v|mpe|mpeg|mpg
|
||||||
video/quicktime, moov|mov|qt
|
video/quicktime, moov|mov|qt
|
||||||
@@ -318,96 +415,36 @@ video/x-atomic3d-feature, fmf
|
|||||||
video/x-dl, dl
|
video/x-dl, dl
|
||||||
video/x-dv, dif|dv
|
video/x-dv, dif|dv
|
||||||
video/x-fli, fli
|
video/x-fli, fli
|
||||||
|
video/x-flv, flv
|
||||||
video/x-isvideo, isu
|
video/x-isvideo, isu
|
||||||
|
!video/x-jng, jng
|
||||||
|
video/x-m4v, m4v
|
||||||
|
video/x-matroska, mkv
|
||||||
|
video/x-mng, mng
|
||||||
video/x-motion-jpeg, mjpg
|
video/x-motion-jpeg, mjpg
|
||||||
video/x-ms-asf, asf|asx
|
video/x-ms-asf, asf|asx|wmv
|
||||||
|
video/x-msvideo, divx
|
||||||
video/x-qtc, qtc
|
video/x-qtc, qtc
|
||||||
video/x-sgi-movie, movie|mv
|
video/x-sgi-movie, movie|mv
|
||||||
application/x-7z-compressed, 7z
|
|
||||||
application/vnd.openxmlformats-officedocument.wordprocessingml.document, docx
|
|
||||||
text/x-po, po
|
|
||||||
application/x-rpm, rpm
|
|
||||||
application/x-debian-package, deb
|
|
||||||
application/vnd.iccprofile, icm
|
|
||||||
application/dicom, dcm
|
|
||||||
image/x-exr, exr
|
|
||||||
application/vnd.iccprofile, icm
|
|
||||||
video/x-matroska, mkv
|
|
||||||
application/x-empty,
|
|
||||||
model/vnd.gdl, gdl
|
|
||||||
model/vnd.gs.gdl, gdsl
|
|
||||||
font/woff, woff
|
|
||||||
font/woff2, woff2
|
|
||||||
application/epub+zip, epub
|
|
||||||
application/x-mobipocket-ebook, mobi
|
|
||||||
audio/x-flac, flac
|
|
||||||
application/x-rar, rar
|
|
||||||
video/x-msvideo, divx
|
|
||||||
video/x-flv, flv
|
|
||||||
application/x-kdelnk,
|
|
||||||
text/x-tcl,
|
|
||||||
application/ogg, ogv
|
|
||||||
application/vnd.openxmlformats-officedocument.spreadsheetml.sheet, xlsx
|
|
||||||
application/vnd.ms-cab-compressed, cab
|
|
||||||
audio/mp4, m4b
|
|
||||||
!image/vnd.djvu, djvu
|
|
||||||
application/x-ms-reader, lit
|
|
||||||
application/CDFV2-corrupt,
|
|
||||||
text/x-vcard, vcf
|
|
||||||
application/x-innosetup,
|
|
||||||
application/winhelp, hlp
|
|
||||||
image/x-tga,
|
|
||||||
application/x-wine-extension-ini,
|
|
||||||
application/x-cbz, cbz
|
|
||||||
application/x-cbr, cbr
|
|
||||||
application/x-ms-compress-szdd, fon
|
|
||||||
application/x-atari-7800-rom, a78
|
|
||||||
application/x-nes-rom, nes
|
|
||||||
application/x-font-pfm, pfm
|
|
||||||
application/x-gettext-translation,
|
|
||||||
image/wmf,
|
|
||||||
application/pgp-keys,
|
|
||||||
image/x-3ds, 3ds
|
|
||||||
application/x-lz4, lz4
|
|
||||||
application/vnd.openxmlformats-officedocument.presentationml.presentation, pptx
|
|
||||||
application/vnd.oasis.opendocument.presentation, odp
|
|
||||||
application/x-msaccess, accdb
|
|
||||||
application/vnd.oasis.opendocument.spreadsheet, ods
|
|
||||||
audio/x-aiff, aiff|aif
|
|
||||||
text/x-ms-regedit, reg
|
|
||||||
application/x-gamecube-rom,
|
|
||||||
application/x-nintendo-ds-rom,
|
|
||||||
text/x-objective-c,
|
|
||||||
application/x-font-gdos,
|
|
||||||
application/x-apple-diskimage,
|
|
||||||
application/x-zstd, zst
|
|
||||||
video/x-m4v, m4v
|
|
||||||
message/news,
|
|
||||||
application/vnd.symbian.install,
|
|
||||||
application/x-lzh-compressed,
|
|
||||||
application/x-dosdriver,
|
|
||||||
application/vnd.tcpdump.pcap, pcap
|
|
||||||
x-epoc/x-sisx-app,
|
x-epoc/x-sisx-app,
|
||||||
application/x-avira-qua,
|
application/x-zstd-dictionary,
|
||||||
video/MP2T,
|
application/vnd.ms-outlook, msg
|
||||||
application/x-snappy-framed,
|
image/x-olympus-orf, orf
|
||||||
application/x-lz4+json, jsonlz4
|
image/x-nikon-nef, nef
|
||||||
application/x-dmp, dmp
|
image/x-fuji-raf, raf
|
||||||
application/zlib, z
|
image/x-panasonic-raw, rw2|raw
|
||||||
application/x-pgp-keyring,
|
image/x-adobe-dng, dng
|
||||||
application/x-gdbm,
|
image/x-canon-cr2, cr2
|
||||||
application/x-font-pf2, pf2
|
image/x-canon-crw, crw
|
||||||
application/x-zip,
|
image/x-dcraw,
|
||||||
application/x-coredump,
|
image/x-kodak-dcr, dcr
|
||||||
application/x-java-jmod, jmod
|
image/x-kodak-k25, k25
|
||||||
application/x-terminfo,
|
image/x-kodak-kdc, kdc
|
||||||
application/x-terminfo2,
|
image/x-minolta-mrw, mrw
|
||||||
application/x-arc,
|
image/x-pentax-pef, pef
|
||||||
application/vnd.lotus-1-2-3,
|
image/x-sigma-x3f, xf3
|
||||||
image/x-win-bitmap,
|
image/x-sony-arw, arw
|
||||||
application/x-maxis-dbpf,
|
image/x-sony-sr2, sr2
|
||||||
text/PGP,
|
image/x-sony-srf, srf
|
||||||
audio/x-hx-aac-adts,
|
image/x-epson-erf, erf
|
||||||
application/x-chrome-extension,
|
sist2/sidecar, s2meta
|
||||||
image/heic, heic
|
|
||||||
image/x-gem,
|
|
||||||
|
@@ -3,6 +3,7 @@ noparse = set()
|
|||||||
ext_in_hash = set()
|
ext_in_hash = set()
|
||||||
|
|
||||||
major_mime = {
|
major_mime = {
|
||||||
|
"sist2": 0,
|
||||||
"model": 1,
|
"model": 1,
|
||||||
"example": 2,
|
"example": 2,
|
||||||
"message": 3,
|
"message": 3,
|
||||||
@@ -18,7 +19,6 @@ major_mime = {
|
|||||||
|
|
||||||
pdf = (
|
pdf = (
|
||||||
"application/pdf",
|
"application/pdf",
|
||||||
"application/x-cbz",
|
|
||||||
"application/epub+zip",
|
"application/epub+zip",
|
||||||
"application/vnd.ms-xpsdocument",
|
"application/vnd.ms-xpsdocument",
|
||||||
)
|
)
|
||||||
@@ -34,6 +34,68 @@ font = (
|
|||||||
"font/woff2"
|
"font/woff2"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Archive "formats"
|
||||||
|
archive = (
|
||||||
|
"application/x-tar",
|
||||||
|
"application/zip",
|
||||||
|
"application/x-rar",
|
||||||
|
"application/x-arc",
|
||||||
|
"application/x-warc",
|
||||||
|
"application/x-7z-compressed",
|
||||||
|
)
|
||||||
|
|
||||||
|
# Archive "filters"
|
||||||
|
arc_filter = (
|
||||||
|
"application/gzip",
|
||||||
|
"application/x-bzip2",
|
||||||
|
"application/x-xz",
|
||||||
|
"application/x-zstd",
|
||||||
|
"application/x-lzma",
|
||||||
|
"application/x-lz4",
|
||||||
|
"application/x-lzip",
|
||||||
|
"application/x-lzop",
|
||||||
|
)
|
||||||
|
|
||||||
|
doc = (
|
||||||
|
"application/vnd.openxmlformats-officedocument.wordprocessingml.document",
|
||||||
|
"application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
|
||||||
|
"application/vnd.openxmlformats-officedocument.presentationml.presentation"
|
||||||
|
)
|
||||||
|
|
||||||
|
mobi = (
|
||||||
|
"application/x-mobipocket-ebook",
|
||||||
|
"application/vnd.amazon.mobi8-ebook"
|
||||||
|
)
|
||||||
|
|
||||||
|
markup = (
|
||||||
|
"text/xml",
|
||||||
|
"text/html",
|
||||||
|
"text/x-sgml"
|
||||||
|
)
|
||||||
|
|
||||||
|
raw = (
|
||||||
|
"image/x-olympus-orf",
|
||||||
|
"image/x-nikon-nef",
|
||||||
|
"image/x-fuji-raf",
|
||||||
|
"image/x-panasonic-raw",
|
||||||
|
"image/x-adobe-dng",
|
||||||
|
"image/x-canon-cr2",
|
||||||
|
"image/x-canon-crw",
|
||||||
|
"image/x-dcraw",
|
||||||
|
"image/x-kodak-dcr",
|
||||||
|
"image/x-kodak-k25",
|
||||||
|
"image/x-kodak-kdc",
|
||||||
|
"image/x-minolta-mrw",
|
||||||
|
"image/x-pentax-pef",
|
||||||
|
"image/x-sigma-x3f",
|
||||||
|
"image/x-sony-arw",
|
||||||
|
"image/x-sony-sr2",
|
||||||
|
"image/x-sony-srf",
|
||||||
|
"image/x-minolta-mrw",
|
||||||
|
"image/x-pentax-pef",
|
||||||
|
"image/x-epson-erf",
|
||||||
|
)
|
||||||
|
|
||||||
cnt = 1
|
cnt = 1
|
||||||
|
|
||||||
|
|
||||||
@@ -48,8 +110,24 @@ def mime_id(mime):
|
|||||||
mime_id += " | 0x40000000"
|
mime_id += " | 0x40000000"
|
||||||
elif mime in font:
|
elif mime in font:
|
||||||
mime_id += " | 0x20000000"
|
mime_id += " | 0x20000000"
|
||||||
|
elif mime in archive:
|
||||||
|
mime_id += " | 0x10000000"
|
||||||
|
elif mime in arc_filter:
|
||||||
|
mime_id += " | 0x08000000"
|
||||||
|
elif mime in doc:
|
||||||
|
mime_id += " | 0x04000000"
|
||||||
|
elif mime in mobi:
|
||||||
|
mime_id += " | 0x02000000"
|
||||||
|
elif mime in markup:
|
||||||
|
mime_id += " | 0x01000000"
|
||||||
|
elif mime in raw:
|
||||||
|
mime_id += " | 0x00800000"
|
||||||
elif mime == "application/x-empty":
|
elif mime == "application/x-empty":
|
||||||
|
cnt -= 1
|
||||||
return "1"
|
return "1"
|
||||||
|
elif mime == "sist2/sidecar":
|
||||||
|
cnt -= 1
|
||||||
|
return "2"
|
||||||
return mime_id
|
return mime_id
|
||||||
|
|
||||||
|
|
||||||
@@ -57,7 +135,7 @@ def clean(t):
|
|||||||
return t.replace("/", "_").replace(".", "_").replace("+", "_").replace("-", "_")
|
return t.replace("/", "_").replace(".", "_").replace("+", "_").replace("-", "_")
|
||||||
|
|
||||||
|
|
||||||
with open("mime.csv") as f:
|
with open("scripts/mime.csv") as f:
|
||||||
for l in f:
|
for l in f:
|
||||||
mime, ext_list = l.split(",")
|
mime, ext_list = l.split(",")
|
||||||
if l.startswith("!"):
|
if l.startswith("!"):
|
||||||
@@ -69,7 +147,7 @@ with open("mime.csv") as f:
|
|||||||
print("// **Generated by mime.py**")
|
print("// **Generated by mime.py**")
|
||||||
print("#ifndef MIME_GENERATED_C")
|
print("#ifndef MIME_GENERATED_C")
|
||||||
print("#define MIME_GENERATED_C")
|
print("#define MIME_GENERATED_C")
|
||||||
print("#include <glib-2.0/glib.h>\n")
|
print("#include <glib.h>\n")
|
||||||
print("#include <stdlib.h>\n")
|
print("#include <stdlib.h>\n")
|
||||||
# Enum
|
# Enum
|
||||||
print("enum mime {")
|
print("enum mime {")
|
||||||
|
|||||||
6
scripts/reset.sh
Executable file
6
scripts/reset.sh
Executable file
@@ -0,0 +1,6 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
make clean
|
||||||
|
rm -rf CMakeFiles/ CMakeCache.txt Makefile \
|
||||||
|
third-party/libscan/CMakeFiles third-party/libscan/CMakeCache.txt third-party/libscan/third-party/ext_ffmpeg \
|
||||||
|
third-party/libscan/third-party/ext_libmobi third-party/libscan/Makefile
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
files = [
|
files = [
|
||||||
"web/css/bundle.css",
|
"sist2-vue/src/assets/favicon.ico",
|
||||||
"web/css/bundle_dark.css",
|
"sist2-vue/dist/css/chunk-vendors.css",
|
||||||
"web/js/bundle.js",
|
"sist2-vue/dist/css/index.css",
|
||||||
"web/img/sprite-skin-flat.png",
|
"sist2-vue/dist/js/chunk-vendors.js",
|
||||||
"web/img/sprite-skin-flat-dark.png",
|
"sist2-vue/dist/js/index.js",
|
||||||
"web/search.html",
|
"sist2-vue/dist/index.html",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
@@ -13,6 +13,10 @@ def clean(filepath):
|
|||||||
|
|
||||||
|
|
||||||
for file in files:
|
for file in files:
|
||||||
with open(file, "rb") as f:
|
try:
|
||||||
data = f.read()
|
with open(file, "rb") as f:
|
||||||
|
data = f.read()
|
||||||
|
except:
|
||||||
|
data = bytes([])
|
||||||
|
|
||||||
print("char %s[%d] = {%s};" % (clean(file), len(data), ",".join(str(int(b)) for b in data)))
|
print("char %s[%d] = {%s};" % (clean(file), len(data), ",".join(str(int(b)) for b in data)))
|
||||||
|
|||||||
23
sist2-vue/.gitignore
vendored
Normal file
23
sist2-vue/.gitignore
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
.DS_Store
|
||||||
|
node_modules
|
||||||
|
|
||||||
|
|
||||||
|
# local env files
|
||||||
|
.env.local
|
||||||
|
.env.*.local
|
||||||
|
|
||||||
|
# Log files
|
||||||
|
npm-debug.log*
|
||||||
|
yarn-debug.log*
|
||||||
|
yarn-error.log*
|
||||||
|
pnpm-debug.log*
|
||||||
|
|
||||||
|
# Editor directories and files
|
||||||
|
.idea
|
||||||
|
.vscode
|
||||||
|
*.suo
|
||||||
|
*.ntvs*
|
||||||
|
*.njsproj
|
||||||
|
*.sln
|
||||||
|
*.sw?
|
||||||
|
*.iml
|
||||||
5
sist2-vue/babel.config.js
Normal file
5
sist2-vue/babel.config.js
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
module.exports = {
|
||||||
|
"presets": [
|
||||||
|
"@vue/cli-plugin-babel/preset"
|
||||||
|
]
|
||||||
|
}
|
||||||
9
sist2-vue/dist/css/chunk-vendors.css
vendored
Normal file
9
sist2-vue/dist/css/chunk-vendors.css
vendored
Normal file
File diff suppressed because one or more lines are too long
1
sist2-vue/dist/css/index.css
vendored
Normal file
1
sist2-vue/dist/css/index.css
vendored
Normal file
File diff suppressed because one or more lines are too long
3
sist2-vue/dist/index.html
vendored
Normal file
3
sist2-vue/dist/index.html
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<!DOCTYPE html><html lang="en"><head><meta charset="utf-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1,user-scalable=no"><title>sist2</title><link href="css/chunk-vendors.css" rel="preload" as="style"><link href="css/index.css" rel="preload" as="style"><link href="js/chunk-vendors.js" rel="preload" as="script"><link href="js/index.js" rel="preload" as="script"><link href="css/chunk-vendors.css" rel="stylesheet"><link href="css/index.css" rel="stylesheet"></head><body><noscript><style>body {
|
||||||
|
height: initial;
|
||||||
|
}</style><div style="text-align: center; margin-top: 100px"><strong>We're sorry but sist2 doesn't work properly without JavaScript enabled. Please enable it to continue.</strong><br><strong>Nous sommes désolés mais sist2 ne fonctionne pas correctement si JavaScript est activé. Veuillez l'activer pour continuer.</strong></div></noscript><div id="app"></div><script src="js/chunk-vendors.js"></script><script src="js/index.js"></script></body></html>
|
||||||
146
sist2-vue/dist/js/chunk-vendors.js
vendored
Normal file
146
sist2-vue/dist/js/chunk-vendors.js
vendored
Normal file
File diff suppressed because one or more lines are too long
1
sist2-vue/dist/js/index.js
vendored
Normal file
1
sist2-vue/dist/js/index.js
vendored
Normal file
File diff suppressed because one or more lines are too long
27250
sist2-vue/package-lock.json
generated
Normal file
27250
sist2-vue/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
54
sist2-vue/package.json
Normal file
54
sist2-vue/package.json
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
{
|
||||||
|
"name": "sist2",
|
||||||
|
"version": "2.11.0",
|
||||||
|
"private": true,
|
||||||
|
"scripts": {
|
||||||
|
"serve": "vue-cli-service serve",
|
||||||
|
"build": "vue-cli-service build --mode production"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@egjs/vue-infinitegrid": "3.3.0",
|
||||||
|
"axios": "^0.21.1",
|
||||||
|
"bootstrap-vue": "^2.21.2",
|
||||||
|
"core-js": "^3.6.5",
|
||||||
|
"crypto-es": "^1.2.7",
|
||||||
|
"d3": "^5.16.0",
|
||||||
|
"date-fns": "^2.21.3",
|
||||||
|
"dom-to-image": "^2.6.0",
|
||||||
|
"fslightbox-vue": "file:../../../mnt/Hatchery/main/projects/sist2/fslightbox-vue-pro-1.3.1.tgz",
|
||||||
|
"nouislider": "^15.2.0",
|
||||||
|
"underscore": "^1.13.1",
|
||||||
|
"vue": "^2.6.12",
|
||||||
|
"vue-color": "^2.8.1",
|
||||||
|
"vue-i18n": "^8.24.4",
|
||||||
|
"vue-masonry-wall": "^0.3.2",
|
||||||
|
"vue-router": "^3.2.0",
|
||||||
|
"vue-simple-suggest": "^1.11.1",
|
||||||
|
"vuex": "^3.4.0"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@babel/polyfill": "^7.11.5",
|
||||||
|
"@vue/cli-plugin-babel": "~4.5.0",
|
||||||
|
"@vue/cli-plugin-router": "~4.5.0",
|
||||||
|
"@vue/cli-plugin-typescript": "~4.5.0",
|
||||||
|
"@vue/cli-plugin-vuex": "~4.5.0",
|
||||||
|
"@vue/cli-service": "~4.5.0",
|
||||||
|
"@vue/test-utils": "^1.0.3",
|
||||||
|
"bootstrap": "^4.5.2",
|
||||||
|
"inspire-tree": "^4.3.1",
|
||||||
|
"inspire-tree-dom": "^4.0.6",
|
||||||
|
"mutationobserver-shim": "^0.3.7",
|
||||||
|
"popper.js": "^1.16.1",
|
||||||
|
"portal-vue": "^2.1.7",
|
||||||
|
"sass": "^1.26.11",
|
||||||
|
"sass-loader": "^10.0.2",
|
||||||
|
"typescript": "~4.1.5",
|
||||||
|
"vue-cli-plugin-bootstrap-vue": "~0.7.0",
|
||||||
|
"vue-template-compiler": "^2.6.11"
|
||||||
|
},
|
||||||
|
"browserslist": [
|
||||||
|
"> 1%",
|
||||||
|
"last 2 versions",
|
||||||
|
"not dead"
|
||||||
|
]
|
||||||
|
}
|
||||||
32
sist2-vue/public/index.html
Normal file
32
sist2-vue/public/index.html
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name='viewport' content='width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no'/>
|
||||||
|
|
||||||
|
<title><%= htmlWebpackPlugin.options.title %></title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<noscript>
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
height: initial;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<div style="text-align: center; margin-top: 100px">
|
||||||
|
<strong>
|
||||||
|
We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled.
|
||||||
|
Please enable it to continue.
|
||||||
|
</strong>
|
||||||
|
<br/>
|
||||||
|
<strong>
|
||||||
|
Nous sommes désolés mais <%= htmlWebpackPlugin.options.title %> ne fonctionne pas correctement
|
||||||
|
si JavaScript est activé.
|
||||||
|
Veuillez l'activer pour continuer.
|
||||||
|
</strong>
|
||||||
|
</div>
|
||||||
|
</noscript>
|
||||||
|
<div id="app"></div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
312
sist2-vue/src/App.vue
Normal file
312
sist2-vue/src/App.vue
Normal file
@@ -0,0 +1,312 @@
|
|||||||
|
<template>
|
||||||
|
<div id="app" :class="getClass()">
|
||||||
|
<NavBar></NavBar>
|
||||||
|
<router-view v-if="!configLoading"/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import NavBar from "@/components/NavBar";
|
||||||
|
import {mapGetters} from "vuex";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
components: {NavBar},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
configLoading: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
...mapGetters(["optTheme"]),
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.$store.dispatch("loadConfiguration").then(() => {
|
||||||
|
this.$root.$i18n.locale = this.$store.state.optLang;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.$store.subscribe((mutation) => {
|
||||||
|
if (mutation.type === "setOptLang") {
|
||||||
|
this.$root.$i18n.locale = mutation.payload;
|
||||||
|
this.configLoading = true;
|
||||||
|
window.setTimeout(() => this.configLoading = false, 10);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
getClass() {
|
||||||
|
return {
|
||||||
|
"theme-light": this.optTheme === "light",
|
||||||
|
"theme-black": this.optTheme === "black",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
,
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
html, body {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#app {
|
||||||
|
/*font-family: Avenir, Helvetica, Arial, sans-serif;*/
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
/*text-align: center;*/
|
||||||
|
color: #2c3e50;
|
||||||
|
padding-bottom: 1em;
|
||||||
|
min-height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*Black theme*/
|
||||||
|
.theme-black {
|
||||||
|
background-color: #000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .card, .theme-black .modal-content {
|
||||||
|
background: #212121;
|
||||||
|
color: #e0e0e0;
|
||||||
|
border-radius: 1px;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.theme-black .table {
|
||||||
|
color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .table td, .theme-black .table th {
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .table thead th {
|
||||||
|
border-bottom: 1px solid #646464;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .custom-select {
|
||||||
|
overflow: auto;
|
||||||
|
background-color: #37474F;
|
||||||
|
border: 1px solid #616161;
|
||||||
|
color: #bdbdbd;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .custom-select:focus {
|
||||||
|
border-color: #757575;
|
||||||
|
outline: 0;
|
||||||
|
box-shadow: 0 0 0 .2rem rgba(0, 123, 255, .25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .inspire-tree .selected > .wholerow, .theme-black .inspire-tree .selected > .title-wrap:hover + .wholerow {
|
||||||
|
background: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .inspire-tree .icon-expand::before, .theme-black .inspire-tree .icon-collapse::before {
|
||||||
|
background-color: black !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .inspire-tree .title {
|
||||||
|
color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .inspire-tree {
|
||||||
|
font-weight: 400;
|
||||||
|
font-size: 14px;
|
||||||
|
font-family: Helvetica, Nueue, Verdana, sans-serif;
|
||||||
|
max-height: 350px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.inspire-tree [type=checkbox] {
|
||||||
|
left: 22px !important;
|
||||||
|
top: 7px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .form-control {
|
||||||
|
background-color: #37474F;
|
||||||
|
border: 1px solid #616161;
|
||||||
|
color: #dbdbdb !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .form-control:focus {
|
||||||
|
background-color: #546E7A;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .input-group-text, .theme-black .default-input {
|
||||||
|
background: #37474F !important;
|
||||||
|
border: 1px solid #616161 !important;
|
||||||
|
color: #dbdbdb !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black ::placeholder {
|
||||||
|
color: #BDBDBD !important;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .nav-tabs .nav-link {
|
||||||
|
color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .nav-tabs .nav-item.show .nav-link, .theme-black .nav-tabs .nav-link.active {
|
||||||
|
background-color: #212121;
|
||||||
|
border-color: #616161 #616161 #212121;
|
||||||
|
color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .nav-tabs .nav-link:focus, .theme-black .nav-tabs .nav-link:focus {
|
||||||
|
border-color: #616161 #616161 #212121;
|
||||||
|
color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .nav-tabs .nav-link:focus, .theme-black .nav-tabs .nav-link:hover {
|
||||||
|
border-color: #e0e0e0 #e0e0e0 #212121;
|
||||||
|
color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .nav-tabs {
|
||||||
|
border-bottom: #616161;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black a:hover, .theme-black .btn:hover {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .b-dropdown a:hover {
|
||||||
|
color: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .btn {
|
||||||
|
color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .modal-header .close {
|
||||||
|
color: #e0e0e0;
|
||||||
|
text-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .modal-header {
|
||||||
|
border-bottom: 1px solid #646464;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* -------------------------- */
|
||||||
|
|
||||||
|
#nav {
|
||||||
|
padding: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#nav a {
|
||||||
|
font-weight: bold;
|
||||||
|
color: #2c3e50;
|
||||||
|
}
|
||||||
|
|
||||||
|
#nav a.router-link-exact-active {
|
||||||
|
color: #42b983;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mobile {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
padding-top: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 650px) {
|
||||||
|
.mobile {
|
||||||
|
display: initial;
|
||||||
|
}
|
||||||
|
|
||||||
|
.not-mobile {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grid-single-column .fit {
|
||||||
|
max-height: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
padding-left: 0;
|
||||||
|
padding-right: 0;
|
||||||
|
padding-top: 0
|
||||||
|
}
|
||||||
|
|
||||||
|
.lightbox-caption {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-icon {
|
||||||
|
width: 1rem;
|
||||||
|
margin-right: 0.2rem;
|
||||||
|
cursor: pointer;
|
||||||
|
line-height: 1rem;
|
||||||
|
height: 1rem;
|
||||||
|
background-image: url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOnhsaW5rPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5L3hsaW5rIiB4PSIwcHgiIHk9IjBweCIKICAgICB2aWV3Qm94PSIwIDAgNDI2LjY2NyA0MjYuNjY3IiBzdHlsZT0iZW5hYmxlLWJhY2tncm91bmQ6bmV3IDAgMCA0MjYuNjY3IDQyNi42Njc7IiBmaWxsPSIjZmZmIj4KPGc+CiAgICA8Zz4KICAgICAgICA8Zz4KICAgICAgICAgICAgPHJlY3QgeD0iMTkyIiB5PSIxOTIiIHdpZHRoPSI0Mi42NjciIGhlaWdodD0iMTI4Ii8+CiAgICAgICAgICAgIDxwYXRoIGQ9Ik0yMTMuMzMzLDBDOTUuNDY3LDAsMCw5NS40NjcsMCwyMTMuMzMzczk1LjQ2NywyMTMuMzMzLDIxMy4zMzMsMjEzLjMzM1M0MjYuNjY3LDMzMS4yLDQyNi42NjcsMjEzLjMzMwogICAgICAgICAgICAgICAgUzMzMS4yLDAsMjEzLjMzMywweiBNMjEzLjMzMywzODRjLTk0LjA4LDAtMTcwLjY2Ny03Ni41ODctMTcwLjY2Ny0xNzAuNjY3UzExOS4yNTMsNDIuNjY3LDIxMy4zMzMsNDIuNjY3CiAgICAgICAgICAgICAgICBTMzg0LDExOS4yNTMsMzg0LDIxMy4zMzNTMzA3LjQxMywzODQsMjEzLjMzMywzODR6Ii8+CiAgICAgICAgICAgIDxyZWN0IHg9IjE5MiIgeT0iMTA2LjY2NyIgd2lkdGg9IjQyLjY2NyIgaGVpZ2h0PSI0Mi42NjciLz4KICAgICAgICA8L2c+CiAgICA8L2c+CjwvZz4KPC9zdmc+Cg==);
|
||||||
|
filter: brightness(45%);
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabs {
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal-title {
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (min-width: 1500px) {
|
||||||
|
.container {
|
||||||
|
max-width: 1440px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-connects {
|
||||||
|
border-radius: 1px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
mark {
|
||||||
|
background: #fff217;
|
||||||
|
border-radius: 0;
|
||||||
|
padding: 1px 0;
|
||||||
|
color: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black mark {
|
||||||
|
background: rgba(251, 191, 41, 0.25);
|
||||||
|
border-radius: 0;
|
||||||
|
padding: 1px 0;
|
||||||
|
color: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .content-div mark {
|
||||||
|
background: rgba(251, 191, 41, 0.40);
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content-div {
|
||||||
|
font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
|
||||||
|
font-size: 13px;
|
||||||
|
padding: 1em;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
border: 1px solid #ccc;
|
||||||
|
border-radius: 4px;
|
||||||
|
margin: 3px;
|
||||||
|
white-space: normal;
|
||||||
|
color: #000;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .content-div {
|
||||||
|
background-color: #37474F;
|
||||||
|
border: 1px solid #616161;
|
||||||
|
color: #E0E0E0FF;
|
||||||
|
}
|
||||||
|
|
||||||
|
.graph {
|
||||||
|
display: inline-block;
|
||||||
|
width: 40%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
383
sist2-vue/src/Sist2Api.ts
Normal file
383
sist2-vue/src/Sist2Api.ts
Normal file
@@ -0,0 +1,383 @@
|
|||||||
|
import axios from "axios";
|
||||||
|
import {ext, strUnescape, lum} from "./util";
|
||||||
|
import CryptoES from 'crypto-es';
|
||||||
|
|
||||||
|
export interface EsTag {
|
||||||
|
id: string
|
||||||
|
count: number
|
||||||
|
color: string | undefined
|
||||||
|
isLeaf: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Tag {
|
||||||
|
style: string
|
||||||
|
text: string
|
||||||
|
rawText: string
|
||||||
|
fg: string
|
||||||
|
bg: string
|
||||||
|
userTag: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Index {
|
||||||
|
name: string
|
||||||
|
version: string
|
||||||
|
id: string
|
||||||
|
idPrefix: string
|
||||||
|
timestamp: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EsHit {
|
||||||
|
_index: string
|
||||||
|
_id: string
|
||||||
|
_score: number
|
||||||
|
_path_md5: string
|
||||||
|
_type: string
|
||||||
|
_tags: Tag[]
|
||||||
|
_seq: number
|
||||||
|
_source: {
|
||||||
|
path: string
|
||||||
|
size: number
|
||||||
|
mime: string
|
||||||
|
name: string
|
||||||
|
extension: string
|
||||||
|
index: string
|
||||||
|
_depth: number
|
||||||
|
mtime: number
|
||||||
|
videoc: string
|
||||||
|
audioc: string
|
||||||
|
parent: string
|
||||||
|
width: number
|
||||||
|
height: number
|
||||||
|
duration: number
|
||||||
|
tag: string[]
|
||||||
|
checksum: string
|
||||||
|
}
|
||||||
|
_props: {
|
||||||
|
isSubDocument: boolean
|
||||||
|
isImage: boolean
|
||||||
|
isGif: boolean
|
||||||
|
isVideo: boolean
|
||||||
|
isPlayableVideo: boolean
|
||||||
|
isPlayableImage: boolean
|
||||||
|
isAudio: boolean
|
||||||
|
hasThumbnail: boolean
|
||||||
|
}
|
||||||
|
highlight: {
|
||||||
|
name: string[] | undefined,
|
||||||
|
content: string[] | undefined,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getIdPrefix(indices: Index[], id: string): string {
|
||||||
|
for (let i = 4; i < 32; i++) {
|
||||||
|
const prefix = id.slice(0, i);
|
||||||
|
|
||||||
|
if (indices.filter(idx => idx.id.slice(0, i) == prefix).length == 1) {
|
||||||
|
return prefix;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EsResult {
|
||||||
|
took: number
|
||||||
|
|
||||||
|
hits: {
|
||||||
|
// TODO: ES 6.X ?
|
||||||
|
total: {
|
||||||
|
value: number
|
||||||
|
}
|
||||||
|
hits: EsHit[]
|
||||||
|
}
|
||||||
|
|
||||||
|
aggregations: any
|
||||||
|
}
|
||||||
|
|
||||||
|
class Sist2Api {
|
||||||
|
|
||||||
|
private baseUrl: string
|
||||||
|
|
||||||
|
constructor(baseUrl: string) {
|
||||||
|
this.baseUrl = baseUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
getSist2Info(): Promise<any> {
|
||||||
|
return axios.get(`${this.baseUrl}i`).then(resp => {
|
||||||
|
const indices = resp.data.indices as Index[];
|
||||||
|
|
||||||
|
resp.data.indices = indices.map(idx => {
|
||||||
|
return {
|
||||||
|
id: idx.id,
|
||||||
|
name: idx.name,
|
||||||
|
timestamp: idx.timestamp,
|
||||||
|
version: idx.version,
|
||||||
|
idPrefix: getIdPrefix(indices, idx.id)
|
||||||
|
} as Index;
|
||||||
|
});
|
||||||
|
|
||||||
|
return resp.data;
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
setHitProps(hit: EsHit): void {
|
||||||
|
hit["_props"] = {} as any;
|
||||||
|
|
||||||
|
const mimeCategory = hit._source.mime == null ? null : hit._source.mime.split("/")[0];
|
||||||
|
|
||||||
|
if ("parent" in hit._source) {
|
||||||
|
hit._props.isSubDocument = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ("thumbnail" in hit._source) {
|
||||||
|
hit._props.hasThumbnail = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (mimeCategory) {
|
||||||
|
case "image":
|
||||||
|
if (hit._source.videoc === "gif") {
|
||||||
|
hit._props.isGif = true;
|
||||||
|
} else {
|
||||||
|
hit._props.isImage = true;
|
||||||
|
}
|
||||||
|
if ("width" in hit._source && !hit._props.isSubDocument && hit._source.videoc !== "tiff"
|
||||||
|
&& hit._source.videoc !== "raw" && hit._source.videoc !== "ppm") {
|
||||||
|
hit._props.isPlayableImage = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case "video":
|
||||||
|
if ("videoc" in hit._source) {
|
||||||
|
hit._props.isVideo = true;
|
||||||
|
}
|
||||||
|
if (hit._props.isVideo) {
|
||||||
|
const videoc = hit._source.videoc;
|
||||||
|
const mime = hit._source.mime;
|
||||||
|
|
||||||
|
hit._props.isPlayableVideo = mime != null &&
|
||||||
|
mime.startsWith("video/") &&
|
||||||
|
!hit._props.isSubDocument &&
|
||||||
|
hit._source.extension !== "mkv" &&
|
||||||
|
hit._source.extension !== "avi" &&
|
||||||
|
hit._source.extension !== "mov" &&
|
||||||
|
videoc !== "hevc" &&
|
||||||
|
videoc !== "mpeg1video" &&
|
||||||
|
videoc !== "mpeg2video" &&
|
||||||
|
videoc !== "wmv3";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case "audio":
|
||||||
|
if ("audioc" in hit._source && !hit._props.isSubDocument) {
|
||||||
|
hit._props.isAudio = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setHitTags(hit: EsHit): void {
|
||||||
|
const tags = [] as Tag[];
|
||||||
|
|
||||||
|
const mimeCategory = hit._source.mime == null ? null : hit._source.mime.split("/")[0];
|
||||||
|
|
||||||
|
switch (mimeCategory) {
|
||||||
|
case "image":
|
||||||
|
case "video":
|
||||||
|
if ("videoc" in hit._source && hit._source.videoc) {
|
||||||
|
tags.push({
|
||||||
|
style: "video",
|
||||||
|
text: hit._source.videoc.replace(" ", ""),
|
||||||
|
userTag: false
|
||||||
|
} as Tag);
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case "audio":
|
||||||
|
if ("audioc" in hit._source && hit._source.audioc) {
|
||||||
|
tags.push({
|
||||||
|
style: "audio",
|
||||||
|
text: hit._source.audioc,
|
||||||
|
userTag: false
|
||||||
|
} as Tag);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// User tags
|
||||||
|
if ("tag" in hit._source) {
|
||||||
|
hit._source.tag.forEach(tag => {
|
||||||
|
tags.push(this.createUserTag(tag));
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
hit._tags = tags;
|
||||||
|
}
|
||||||
|
|
||||||
|
createUserTag(tag: string): Tag {
|
||||||
|
const tokens = tag.split(".");
|
||||||
|
|
||||||
|
const colorToken = tokens.pop() as string;
|
||||||
|
|
||||||
|
const bg = colorToken;
|
||||||
|
const fg = lum(colorToken) > 50 ? "#000" : "#fff";
|
||||||
|
|
||||||
|
return {
|
||||||
|
style: "user",
|
||||||
|
fg: fg,
|
||||||
|
bg: bg,
|
||||||
|
text: tokens.join("."),
|
||||||
|
rawText: tag,
|
||||||
|
userTag: true,
|
||||||
|
} as Tag;
|
||||||
|
}
|
||||||
|
|
||||||
|
esQuery(query: any): Promise<EsResult> {
|
||||||
|
return axios.post(`${this.baseUrl}es`, query).then(resp => {
|
||||||
|
const res = resp.data as EsResult;
|
||||||
|
|
||||||
|
if (res.hits?.hits) {
|
||||||
|
res.hits.hits.forEach((hit: EsHit) => {
|
||||||
|
hit["_source"]["name"] = strUnescape(hit["_source"]["name"]);
|
||||||
|
hit["_source"]["path"] = strUnescape(hit["_source"]["path"]);
|
||||||
|
hit["_path_md5"] = CryptoES.MD5(
|
||||||
|
hit["_source"]["path"] +
|
||||||
|
(hit["_source"]["path"] ? "/" : "") +
|
||||||
|
hit["_source"]["name"] + ext(hit)
|
||||||
|
).toString();
|
||||||
|
|
||||||
|
this.setHitProps(hit);
|
||||||
|
this.setHitTags(hit);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
getMimeTypes() {
|
||||||
|
return this.esQuery({
|
||||||
|
aggs: {
|
||||||
|
mimeTypes: {
|
||||||
|
terms: {
|
||||||
|
field: "mime",
|
||||||
|
size: 10000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
size: 0,
|
||||||
|
}).then(resp => {
|
||||||
|
const mimeMap: any[] = [];
|
||||||
|
resp["aggregations"]["mimeTypes"]["buckets"].sort((a: any, b: any) => a.key > b.key).forEach((bucket: any) => {
|
||||||
|
const tmp = bucket["key"].split("/");
|
||||||
|
const category = tmp[0];
|
||||||
|
const mime = tmp[1];
|
||||||
|
|
||||||
|
let category_exists = false;
|
||||||
|
|
||||||
|
const child = {
|
||||||
|
"id": bucket["key"],
|
||||||
|
"text": `${mime} (${bucket["doc_count"]})`
|
||||||
|
};
|
||||||
|
|
||||||
|
mimeMap.forEach(node => {
|
||||||
|
if (node.text === category) {
|
||||||
|
node.children.push(child);
|
||||||
|
category_exists = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!category_exists) {
|
||||||
|
mimeMap.push({"text": category, children: [child]});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return mimeMap;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_createEsTag(tag: string, count: number): EsTag {
|
||||||
|
const tokens = tag.split(".");
|
||||||
|
|
||||||
|
if (/.*\.#[0-9a-f]{6}/.test(tag)) {
|
||||||
|
return {
|
||||||
|
id: tokens.slice(0, -1).join("."),
|
||||||
|
color: tokens.pop(),
|
||||||
|
isLeaf: true,
|
||||||
|
count: count
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: tag,
|
||||||
|
count: count,
|
||||||
|
isLeaf: false,
|
||||||
|
color: undefined
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
getDocInfo(docId: string) {
|
||||||
|
return axios.get(`${this.baseUrl}d/${docId}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
getTags() {
|
||||||
|
return this.esQuery({
|
||||||
|
aggs: {
|
||||||
|
tags: {
|
||||||
|
terms: {
|
||||||
|
field: "tag",
|
||||||
|
size: 10000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
size: 0,
|
||||||
|
}).then(resp => {
|
||||||
|
const seen = new Set();
|
||||||
|
|
||||||
|
const tags = resp["aggregations"]["tags"]["buckets"]
|
||||||
|
.sort((a: any, b: any) => a["key"].localeCompare(b["key"]))
|
||||||
|
.map((bucket: any) => this._createEsTag(bucket["key"], bucket["doc_count"]));
|
||||||
|
|
||||||
|
// Remove duplicates (same tag with different color)
|
||||||
|
return tags.filter((t: EsTag) => {
|
||||||
|
if (seen.has(t.id)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
seen.add(t.id);
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
saveTag(tag: string, hit: EsHit) {
|
||||||
|
return axios.post(`${this.baseUrl}tag/` + hit["_source"]["index"], {
|
||||||
|
delete: false,
|
||||||
|
name: tag,
|
||||||
|
doc_id: hit["_id"],
|
||||||
|
path_md5: hit._path_md5
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
deleteTag(tag: string, hit: EsHit) {
|
||||||
|
return axios.post(`${this.baseUrl}tag/` + hit["_source"]["index"], {
|
||||||
|
delete: true,
|
||||||
|
name: tag,
|
||||||
|
doc_id: hit["_id"],
|
||||||
|
path_md5: hit._path_md5
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
getTreemapCsvUrl(indexId: string) {
|
||||||
|
return `${this.baseUrl}s/${indexId}/1`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getMimeCsvUrl(indexId: string) {
|
||||||
|
return `${this.baseUrl}s/${indexId}/2`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getSizeCsv(indexId: string) {
|
||||||
|
return `${this.baseUrl}s/${indexId}/3`;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDateCsv(indexId: string) {
|
||||||
|
return `${this.baseUrl}s/${indexId}/4`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new Sist2Api("");
|
||||||
229
sist2-vue/src/Sist2Query.ts
Normal file
229
sist2-vue/src/Sist2Query.ts
Normal file
@@ -0,0 +1,229 @@
|
|||||||
|
import store from "./store";
|
||||||
|
import {EsHit, Index} from "@/Sist2Api";
|
||||||
|
|
||||||
|
const SORT_MODES = {
|
||||||
|
score: {
|
||||||
|
mode: [
|
||||||
|
{_score: {order: "desc"}},
|
||||||
|
{_tie: {order: "asc"}}
|
||||||
|
],
|
||||||
|
key: (hit: EsHit) => hit._score
|
||||||
|
},
|
||||||
|
random: {
|
||||||
|
mode: [
|
||||||
|
{_score: {order: "desc"}},
|
||||||
|
{_tie: {order: "asc"}}
|
||||||
|
],
|
||||||
|
key: (hit: EsHit) => hit._score
|
||||||
|
},
|
||||||
|
dateAsc: {
|
||||||
|
mode: [
|
||||||
|
{mtime: {order: "asc"}},
|
||||||
|
{_tie: {order: "asc"}}
|
||||||
|
],
|
||||||
|
key: (hit: EsHit) => hit._source.mtime
|
||||||
|
},
|
||||||
|
dateDesc: {
|
||||||
|
mode: [
|
||||||
|
{mtime: {order: "desc"}},
|
||||||
|
{_tie: {order: "asc"}}
|
||||||
|
],
|
||||||
|
key: (hit: EsHit) => hit._source.mtime
|
||||||
|
},
|
||||||
|
sizeAsc: {
|
||||||
|
mode: [
|
||||||
|
{size: {order: "asc"}},
|
||||||
|
{_tie: {order: "asc"}}
|
||||||
|
],
|
||||||
|
key: (hit: EsHit) => hit._source.size
|
||||||
|
},
|
||||||
|
sizeDesc: {
|
||||||
|
mode: [
|
||||||
|
{size: {order: "desc"}},
|
||||||
|
{_tie: {order: "asc"}}
|
||||||
|
],
|
||||||
|
key: (hit: EsHit) => hit._source.size
|
||||||
|
}
|
||||||
|
} as any;
|
||||||
|
|
||||||
|
interface SortMode {
|
||||||
|
text: string
|
||||||
|
mode: any[]
|
||||||
|
key: (hit: EsHit) => any
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class Sist2Query {
|
||||||
|
|
||||||
|
searchQuery(): any {
|
||||||
|
|
||||||
|
const getters = store.getters;
|
||||||
|
|
||||||
|
const searchText = getters.searchText;
|
||||||
|
const pathText = getters.pathText;
|
||||||
|
const empty = searchText === "";
|
||||||
|
const sizeMin = getters.sizeMin;
|
||||||
|
const sizeMax = getters.sizeMax;
|
||||||
|
const dateMin = getters.dateMin;
|
||||||
|
const dateMax = getters.dateMax;
|
||||||
|
const fuzzy = getters.fuzzy;
|
||||||
|
const size = getters.size;
|
||||||
|
const after = getters.lastDoc;
|
||||||
|
const selectedIndexIds = getters.selectedIndices.map((idx: Index) => idx.id)
|
||||||
|
const selectedMimeTypes = getters.selectedMimeTypes;
|
||||||
|
const selectedTags = getters.selectedTags;
|
||||||
|
|
||||||
|
const filters = [
|
||||||
|
{terms: {index: selectedIndexIds}}
|
||||||
|
] as any[];
|
||||||
|
|
||||||
|
if (sizeMin && sizeMax) {
|
||||||
|
filters.push({range: {size: {gte: sizeMin, lte: sizeMax}}})
|
||||||
|
} else if (sizeMin) {
|
||||||
|
filters.push({range: {size: {gte: sizeMin}}})
|
||||||
|
} else if (sizeMax) {
|
||||||
|
filters.push({range: {size: {lte: sizeMax}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dateMin && dateMax) {
|
||||||
|
filters.push({range: {mtime: {gte: dateMin, lte: dateMax}}})
|
||||||
|
} else if (dateMin) {
|
||||||
|
filters.push({range: {mtime: {gte: dateMin}}})
|
||||||
|
} else if (dateMax) {
|
||||||
|
filters.push({range: {mtime: {lte: dateMax}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
const fields = [
|
||||||
|
"name^8",
|
||||||
|
"content^3",
|
||||||
|
"album^8", "artist^8", "title^8", "genre^2", "album_artist^8",
|
||||||
|
"font_name^6"
|
||||||
|
];
|
||||||
|
|
||||||
|
if (getters.optSearchInPath) {
|
||||||
|
fields.push("path.text^5");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fuzzy) {
|
||||||
|
fields.push("content.nGram");
|
||||||
|
if (getters.optSearchInPath) {
|
||||||
|
fields.push("path.nGram");
|
||||||
|
}
|
||||||
|
fields.push("name.nGram^3");
|
||||||
|
}
|
||||||
|
|
||||||
|
const path = pathText.replace(/\/$/, "").toLowerCase(); //remove trailing slashes
|
||||||
|
if (path !== "") {
|
||||||
|
filters.push({term: {path: path}})
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedMimeTypes.length > 0) {
|
||||||
|
filters.push({terms: {"mime": selectedMimeTypes}});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedTags.length > 0) {
|
||||||
|
if (getters.optTagOrOperator) {
|
||||||
|
filters.push({terms: {"tag": selectedTags}});
|
||||||
|
} else {
|
||||||
|
selectedTags.forEach((tag: string) => filters.push({term: {"tag": tag}}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let query;
|
||||||
|
if (getters.optQueryMode === "simple") {
|
||||||
|
query = {
|
||||||
|
simple_query_string: {
|
||||||
|
query: searchText,
|
||||||
|
fields: fields,
|
||||||
|
default_operator: "and"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
query = {
|
||||||
|
query_string: {
|
||||||
|
query: searchText,
|
||||||
|
default_field: "name",
|
||||||
|
default_operator: "and"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const q = {
|
||||||
|
_source: {
|
||||||
|
excludes: ["content", "_tie"]
|
||||||
|
},
|
||||||
|
query: {
|
||||||
|
bool: {
|
||||||
|
filter: filters,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
sort: SORT_MODES[getters.sortMode].mode,
|
||||||
|
aggs:
|
||||||
|
{
|
||||||
|
total_size: {"sum": {"field": "size"}},
|
||||||
|
total_count: {"value_count": {"field": "size"}}
|
||||||
|
},
|
||||||
|
size: size,
|
||||||
|
} as any;
|
||||||
|
|
||||||
|
if (!empty) {
|
||||||
|
q.query.bool.must = query;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (after) {
|
||||||
|
q.search_after = [SORT_MODES[getters.sortMode].key(after), after["_id"]];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getters.optHighlight) {
|
||||||
|
q.highlight = {
|
||||||
|
pre_tags: ["<mark>"],
|
||||||
|
post_tags: ["</mark>"],
|
||||||
|
fragment_size: getters.optFragmentSize,
|
||||||
|
number_of_fragments: 1,
|
||||||
|
order: "score",
|
||||||
|
fields: {
|
||||||
|
content: {},
|
||||||
|
name: {},
|
||||||
|
"name.nGram": {},
|
||||||
|
"content.nGram": {},
|
||||||
|
font_name: {},
|
||||||
|
},
|
||||||
|
max_analyzed_offset: 9_999_999
|
||||||
|
};
|
||||||
|
if (getters.optSearchInPath) {
|
||||||
|
q.highlight.fields["path.text"] = {};
|
||||||
|
q.highlight.fields["path.nGram"] = {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getters.sortMode === "random") {
|
||||||
|
q.query = {
|
||||||
|
function_score: {
|
||||||
|
query: {
|
||||||
|
bool: {
|
||||||
|
must: filters,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
functions: [
|
||||||
|
{
|
||||||
|
random_score: {
|
||||||
|
seed: getters.seed,
|
||||||
|
field: "_seq_no",
|
||||||
|
},
|
||||||
|
weight: 1000
|
||||||
|
}
|
||||||
|
],
|
||||||
|
boost_mode: "sum"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty) {
|
||||||
|
q.query.function_score.query.bool.must.push(query);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return q;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new Sist2Query();
|
||||||
0
sist2-vue/src/assets/.gitkeep
Normal file
0
sist2-vue/src/assets/.gitkeep
Normal file
BIN
sist2-vue/src/assets/favicon.ico
Normal file
BIN
sist2-vue/src/assets/favicon.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 15 KiB |
28
sist2-vue/src/components/ContentDiv.vue
Normal file
28
sist2-vue/src/components/ContentDiv.vue
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<template>
|
||||||
|
<div class="content-div" v-html="content()" v-if="content()"></div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "ContentDiv",
|
||||||
|
props: ["doc"],
|
||||||
|
methods: {
|
||||||
|
content() {
|
||||||
|
if (!this.doc.highlight) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.doc.highlight["content.nGram"]) {
|
||||||
|
return this.doc.highlight["content.nGram"][0];
|
||||||
|
}
|
||||||
|
if (this.doc.highlight.content) {
|
||||||
|
return this.doc.highlight.content[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
129
sist2-vue/src/components/D3DateHistogram.vue
Normal file
129
sist2-vue/src/components/D3DateHistogram.vue
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
<template>
|
||||||
|
<div class="graph">
|
||||||
|
<svg id="date-histogram"></svg>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import * as d3 from "d3";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
|
||||||
|
const formatSI = d3.format("~s");
|
||||||
|
|
||||||
|
|
||||||
|
function dateHistogram(data, svg, title) {
|
||||||
|
let bins = data.map(d => {
|
||||||
|
return {
|
||||||
|
length: Number(d.count),
|
||||||
|
x0: Number(d.bucket),
|
||||||
|
x1: Number(d.bucket) + 2629800
|
||||||
|
}
|
||||||
|
});
|
||||||
|
bins.sort((a, b) => a.length - b.length);
|
||||||
|
|
||||||
|
const margin = {
|
||||||
|
top: 50,
|
||||||
|
right: 20,
|
||||||
|
bottom: 70,
|
||||||
|
left: 40
|
||||||
|
};
|
||||||
|
|
||||||
|
const thresh = d3.quantile(bins, 0.9, d => d.length);
|
||||||
|
bins = bins.filter(d => d.length > thresh);
|
||||||
|
|
||||||
|
const width = 550;
|
||||||
|
const height = 450;
|
||||||
|
|
||||||
|
svg.selectAll("*").remove();
|
||||||
|
svg.attr("viewBox", [0, 0, width, height]);
|
||||||
|
|
||||||
|
const y = d3.scaleLinear()
|
||||||
|
.domain([0, d3.max(bins, d => d.length)]).nice()
|
||||||
|
.range([height - margin.bottom, margin.top]);
|
||||||
|
|
||||||
|
const x = d3.scaleLinear()
|
||||||
|
.domain(d3.extent(bins, d => d.x0)).nice()
|
||||||
|
.range([margin.left, width - margin.right]);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("fill", "steelblue")
|
||||||
|
.selectAll("rect")
|
||||||
|
.data(bins)
|
||||||
|
.join("rect")
|
||||||
|
.attr("x", d => x(d.x0) + 1)
|
||||||
|
.attr("width", d => Math.max(1, x(d.x1) - x(d.x0) - 1))
|
||||||
|
.attr("y", d => y(d.length))
|
||||||
|
.attr("height", d => y(0) - y(d.length))
|
||||||
|
.call(g => g
|
||||||
|
.append("title")
|
||||||
|
.text(d => d.length)
|
||||||
|
);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(0,${height - margin.bottom})`)
|
||||||
|
.call(
|
||||||
|
d3.axisBottom(x)
|
||||||
|
.ticks(width / 30)
|
||||||
|
.tickSizeOuter(0)
|
||||||
|
.tickFormat(t => d3.timeFormat("%Y-%m-%d")(d3.utcParse("%s")(t)))
|
||||||
|
)
|
||||||
|
.call(g => g
|
||||||
|
.selectAll("text")
|
||||||
|
.style("text-anchor", "end")
|
||||||
|
.attr("dx", "-.8em")
|
||||||
|
.attr("dy", ".15em")
|
||||||
|
.attr("transform", "rotate(-65)")
|
||||||
|
)
|
||||||
|
.call(g => g.append("text")
|
||||||
|
.attr("x", width - margin.right)
|
||||||
|
.attr("y", -4)
|
||||||
|
.attr("fill", "currentColor")
|
||||||
|
.attr("font-weight", "bold")
|
||||||
|
.attr("text-anchor", "end")
|
||||||
|
.text("mtime")
|
||||||
|
);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(${margin.left},0)`)
|
||||||
|
.call(
|
||||||
|
d3.axisLeft(y)
|
||||||
|
.ticks(height / 40)
|
||||||
|
.tickFormat(t => formatSI(t))
|
||||||
|
)
|
||||||
|
.call(g => g.select(".domain").remove())
|
||||||
|
.call(g => g.select(".tick:last-of-type text").clone()
|
||||||
|
.attr("x", 4)
|
||||||
|
.attr("text-anchor", "start")
|
||||||
|
.attr("font-weight", "bold")
|
||||||
|
.text("File count"));
|
||||||
|
|
||||||
|
svg.append("text")
|
||||||
|
.attr("x", (width / 2))
|
||||||
|
.attr("y", (margin.top / 2))
|
||||||
|
.attr("text-anchor", "middle")
|
||||||
|
.style("font-size", "16px")
|
||||||
|
.text(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "D3DateHistogram",
|
||||||
|
props: ["indexId"],
|
||||||
|
mounted() {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
indexId: function () {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
update(indexId) {
|
||||||
|
const svg = d3.select("#date-histogram");
|
||||||
|
|
||||||
|
d3.csv(Sist2Api.getDateCsv(indexId)).then(tabularData => {
|
||||||
|
dateHistogram(tabularData.slice(), svg, this.$t("d3.dateHistogram"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
100
sist2-vue/src/components/D3MimeBarCount.vue
Normal file
100
sist2-vue/src/components/D3MimeBarCount.vue
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
<template>
|
||||||
|
<div class="graph">
|
||||||
|
<svg id="agg-mime-count"></svg>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import * as d3 from "d3";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
|
||||||
|
const formatSI = d3.format("~s");
|
||||||
|
const barHeight = 20;
|
||||||
|
const ordinalColor = d3.scaleOrdinal(d3.schemeCategory10);
|
||||||
|
|
||||||
|
function mimeBarCount(data, svg, fillOpacity, title) {
|
||||||
|
|
||||||
|
const margin = {
|
||||||
|
top: 50,
|
||||||
|
right: 0,
|
||||||
|
bottom: 10,
|
||||||
|
left: Math.max(
|
||||||
|
d3.max(data.sort((a, b) => b.count - a.count).slice(0, 15), d => d.mime.length) * 6,
|
||||||
|
d3.max(data.sort((a, b) => b.size - a.size).slice(0, 15), d => d.mime.length) * 6,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
data.forEach(d => {
|
||||||
|
d.name = d.mime;
|
||||||
|
d.value = Number(d.count);
|
||||||
|
});
|
||||||
|
|
||||||
|
data = data.sort((a, b) => b.value - a.value).slice(0, 15);
|
||||||
|
|
||||||
|
const width = 550;
|
||||||
|
const height = Math.ceil((data.length + 0.1) * barHeight) + margin.top + margin.bottom;
|
||||||
|
|
||||||
|
svg.selectAll("*").remove();
|
||||||
|
svg.attr("viewBox", [0, 0, width, height]);
|
||||||
|
|
||||||
|
const y = d3.scaleBand()
|
||||||
|
.domain(d3.range(data.length))
|
||||||
|
.rangeRound([margin.top, height - margin.bottom]);
|
||||||
|
|
||||||
|
const x = d3.scaleLinear()
|
||||||
|
.domain([0, d3.max(data, d => d.value)])
|
||||||
|
.range([margin.left, width - margin.right]);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("fill-opacity", fillOpacity)
|
||||||
|
.selectAll("rect")
|
||||||
|
.data(data)
|
||||||
|
.join("rect")
|
||||||
|
.attr("fill", d => ordinalColor(d.name))
|
||||||
|
.attr("x", x(0))
|
||||||
|
.attr("y", (d, i) => y(i))
|
||||||
|
.attr("width", d => x(d.value) - x(0))
|
||||||
|
.attr("height", y.bandwidth())
|
||||||
|
.append("title")
|
||||||
|
.text(d => d3.format(",")(d.value));
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(0,${margin.top})`)
|
||||||
|
.call(d3.axisTop(x).ticks(width / 80, data.format).tickFormat(formatSI))
|
||||||
|
.call(g => g.select(".domain").remove());
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(${margin.left},0)`)
|
||||||
|
.call(d3.axisLeft(y).tickFormat(i => data[i].name).tickSizeOuter(0));
|
||||||
|
|
||||||
|
svg.append("text")
|
||||||
|
.attr("x", (width / 2))
|
||||||
|
.attr("y", (margin.top / 2))
|
||||||
|
.attr("text-anchor", "middle")
|
||||||
|
.style("font-size", "16px")
|
||||||
|
.text(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "D3MimeBarSize",
|
||||||
|
props: ["indexId"],
|
||||||
|
mounted() {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
indexId: function () {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
update(indexId) {
|
||||||
|
const mimeSvgCount = d3.select("#agg-mime-count");
|
||||||
|
const fillOpacity = this.$store.state.optTheme === "black" ? 0.9 : 0.6;
|
||||||
|
|
||||||
|
d3.csv(Sist2Api.getMimeCsvUrl(indexId)).then(tabularData => {
|
||||||
|
mimeBarCount(tabularData.slice(), mimeSvgCount, fillOpacity, this.$t("d3.mimeCount"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
99
sist2-vue/src/components/D3MimeBarSize.vue
Normal file
99
sist2-vue/src/components/D3MimeBarSize.vue
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
<template>
|
||||||
|
<div class="graph">
|
||||||
|
<svg id="agg-mime-size"></svg>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import * as d3 from "d3";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
|
||||||
|
const formatSI = d3.format("~s");
|
||||||
|
const barHeight = 20;
|
||||||
|
const ordinalColor = d3.scaleOrdinal(d3.schemeCategory10);
|
||||||
|
|
||||||
|
function mimeBarSize(data, svg, fillOpacity, title) {
|
||||||
|
|
||||||
|
const margin = {
|
||||||
|
top: 50,
|
||||||
|
right: 0,
|
||||||
|
bottom: 10,
|
||||||
|
left: Math.max(
|
||||||
|
d3.max(data.sort((a, b) => b.count - a.count).slice(0, 15), d => d.mime.length) * 6,
|
||||||
|
d3.max(data.sort((a, b) => b.size - a.size).slice(0, 15), d => d.mime.length) * 6,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
data.forEach(d => {
|
||||||
|
d.name = d.mime;
|
||||||
|
d.value = Number(d.size);
|
||||||
|
});
|
||||||
|
data = data.sort((a, b) => b.value - a.value).slice(0, 15);
|
||||||
|
|
||||||
|
const width = 550;
|
||||||
|
const height = Math.ceil((data.length + 0.1) * barHeight) + margin.top + margin.bottom;
|
||||||
|
|
||||||
|
svg.selectAll("*").remove();
|
||||||
|
svg.attr("viewBox", [0, 0, width, height]);
|
||||||
|
|
||||||
|
const y = d3.scaleBand()
|
||||||
|
.domain(d3.range(data.length))
|
||||||
|
.rangeRound([margin.top, height - margin.bottom]);
|
||||||
|
|
||||||
|
const x = d3.scaleLinear()
|
||||||
|
.domain([0, d3.max(data, d => d.value)])
|
||||||
|
.range([margin.left, width - margin.right]);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("fill-opacity", fillOpacity)
|
||||||
|
.selectAll("rect")
|
||||||
|
.data(data)
|
||||||
|
.join("rect")
|
||||||
|
.attr("fill", d => ordinalColor(d.name))
|
||||||
|
.attr("x", x(0))
|
||||||
|
.attr("y", (d, i) => y(i))
|
||||||
|
.attr("width", d => x(d.value) - x(0))
|
||||||
|
.attr("height", y.bandwidth())
|
||||||
|
.append("title")
|
||||||
|
.text(d => formatSI(d.value));
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(0,${margin.top})`)
|
||||||
|
.call(d3.axisTop(x).ticks(width / 80, data.format).tickFormat(formatSI))
|
||||||
|
.call(g => g.select(".domain").remove());
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(${margin.left},0)`)
|
||||||
|
.call(d3.axisLeft(y).tickFormat(i => data[i].name).tickSizeOuter(0));
|
||||||
|
|
||||||
|
svg.append("text")
|
||||||
|
.attr("x", (width / 2))
|
||||||
|
.attr("y", (margin.top / 2))
|
||||||
|
.attr("text-anchor", "middle")
|
||||||
|
.style("font-size", "16px")
|
||||||
|
.text(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "D3MimeBarSize",
|
||||||
|
props: ["indexId"],
|
||||||
|
mounted() {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
indexId: function () {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
update(indexId) {
|
||||||
|
const mimeSvgSize = d3.select("#agg-mime-size");
|
||||||
|
const fillOpacity = this.$store.state.optTheme === "black" ? 0.9 : 0.6;
|
||||||
|
|
||||||
|
d3.csv(Sist2Api.getMimeCsvUrl(indexId)).then(tabularData => {
|
||||||
|
mimeBarSize(tabularData.slice(), mimeSvgSize, fillOpacity, this.$t("d3.mimeSize"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
126
sist2-vue/src/components/D3SizeHistogram.vue
Normal file
126
sist2-vue/src/components/D3SizeHistogram.vue
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
<template>
|
||||||
|
<div class="graph">
|
||||||
|
<svg id="size-histogram"></svg>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import * as d3 from "d3";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
|
||||||
|
const formatSI = d3.format("~s");
|
||||||
|
|
||||||
|
function sizeHistogram(data, svg, title) {
|
||||||
|
|
||||||
|
let bins = data.map(d => {
|
||||||
|
return {
|
||||||
|
length: Number(d.count),
|
||||||
|
x0: Number(d.bucket),
|
||||||
|
x1: Number(d.bucket) + (5 * 1024 * 1024)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
bins = bins.sort((a, b) => b.length - a.length).slice(0, 25);
|
||||||
|
|
||||||
|
const margin = {
|
||||||
|
top: 50,
|
||||||
|
right: 20,
|
||||||
|
bottom: 70,
|
||||||
|
left: 40
|
||||||
|
};
|
||||||
|
|
||||||
|
const width = 550;
|
||||||
|
const height = 450;
|
||||||
|
|
||||||
|
svg.selectAll("*").remove();
|
||||||
|
svg.attr("viewBox", [0, 0, width, height]);
|
||||||
|
|
||||||
|
const y = d3.scaleLinear()
|
||||||
|
.domain([0, d3.max(bins, d => d.length)])
|
||||||
|
.range([height - margin.bottom, margin.top]);
|
||||||
|
|
||||||
|
const x = d3.scaleLinear()
|
||||||
|
.domain(d3.extent(bins, d => d.x0)).nice()
|
||||||
|
.range([margin.left, width - margin.right]);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("fill", "steelblue")
|
||||||
|
.selectAll("rect")
|
||||||
|
.data(bins)
|
||||||
|
.join("rect")
|
||||||
|
.attr("x", d => x(d.x0) + 1)
|
||||||
|
.attr("width", d => Math.max(1, x(d.x1) - x(d.x0) - 1))
|
||||||
|
.attr("y", d => y(d.length))
|
||||||
|
.attr("height", d => y(0) - y(d.length))
|
||||||
|
.call(g => g
|
||||||
|
.append("title")
|
||||||
|
.text(d => d.length)
|
||||||
|
);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(0,${height - margin.bottom})`)
|
||||||
|
.call(
|
||||||
|
d3.axisBottom(x)
|
||||||
|
.ticks(width / 30)
|
||||||
|
.tickSizeOuter(0)
|
||||||
|
.tickFormat(formatSI)
|
||||||
|
)
|
||||||
|
.call(g => g
|
||||||
|
.selectAll("text")
|
||||||
|
.style("text-anchor", "end")
|
||||||
|
.attr("dx", "-.8em")
|
||||||
|
.attr("dy", ".15em")
|
||||||
|
.attr("transform", "rotate(-65)")
|
||||||
|
)
|
||||||
|
.call(g => g.append("text")
|
||||||
|
.attr("x", width - margin.right)
|
||||||
|
.attr("y", -4)
|
||||||
|
.attr("fill", "currentColor")
|
||||||
|
.attr("font-weight", "bold")
|
||||||
|
.attr("text-anchor", "end")
|
||||||
|
.text("size (bytes)")
|
||||||
|
);
|
||||||
|
|
||||||
|
svg.append("g")
|
||||||
|
.attr("transform", `translate(${margin.left},0)`)
|
||||||
|
.call(
|
||||||
|
d3.axisLeft(y)
|
||||||
|
.ticks(height / 40)
|
||||||
|
.tickFormat(t => formatSI(t))
|
||||||
|
)
|
||||||
|
.call(g => g.select(".domain").remove())
|
||||||
|
.call(g => g.select(".tick:last-of-type text").clone()
|
||||||
|
.attr("x", 4)
|
||||||
|
.attr("text-anchor", "start")
|
||||||
|
.attr("font-weight", "bold")
|
||||||
|
.text("File count"));
|
||||||
|
|
||||||
|
svg.append("text")
|
||||||
|
.attr("x", (width / 2))
|
||||||
|
.attr("y", (margin.top / 2))
|
||||||
|
.attr("text-anchor", "middle")
|
||||||
|
.style("font-size", "16px")
|
||||||
|
.text(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "D3SizeHistogram",
|
||||||
|
props: ["indexId"],
|
||||||
|
mounted() {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
indexId: function () {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
update(indexId) {
|
||||||
|
const svg = d3.select("#size-histogram");
|
||||||
|
|
||||||
|
d3.csv(Sist2Api.getSizeCsv(indexId)).then(tabularData => {
|
||||||
|
sizeHistogram(tabularData.slice(), svg, this.$t("d3.sizeHistogram"));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
267
sist2-vue/src/components/D3Treemap.vue
Normal file
267
sist2-vue/src/components/D3Treemap.vue
Normal file
@@ -0,0 +1,267 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<b-btn style="float:right;margin-bottom: 10px" @click="downloadTreemap()" variant="primary">
|
||||||
|
{{ $t("download") }}
|
||||||
|
</b-btn>
|
||||||
|
<svg id="treemap"></svg>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import * as d3 from "d3";
|
||||||
|
import {burrow} from "@/util-js"
|
||||||
|
import {humanFileSize} from "@/util";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
import domtoimage from "dom-to-image";
|
||||||
|
|
||||||
|
|
||||||
|
const TILING_MODES = {
|
||||||
|
"squarify": d3.treemapSquarify,
|
||||||
|
"binary": d3.treemapBinary,
|
||||||
|
"sliceDice": d3.treemapSliceDice,
|
||||||
|
"slice": d3.treemapSlice,
|
||||||
|
"dice": d3.treemapDice,
|
||||||
|
};
|
||||||
|
|
||||||
|
const COLORS = {
|
||||||
|
"PuBuGn": d3.interpolatePuBuGn,
|
||||||
|
"PuRd": d3.interpolatePuRd,
|
||||||
|
"PuBu": d3.interpolatePuBu,
|
||||||
|
"YlOrBr": d3.interpolateYlOrBr,
|
||||||
|
"YlOrRd": d3.interpolateYlOrRd,
|
||||||
|
"YlGn": d3.interpolateYlGn,
|
||||||
|
"YlGnBu": d3.interpolateYlGnBu,
|
||||||
|
"Plasma": d3.interpolatePlasma,
|
||||||
|
"Magma": d3.interpolateMagma,
|
||||||
|
"Inferno": d3.interpolateInferno,
|
||||||
|
"Viridis": d3.interpolateViridis,
|
||||||
|
"Turbo": d3.interpolateTurbo,
|
||||||
|
};
|
||||||
|
|
||||||
|
const SIZES = {
|
||||||
|
"small": [800, 600],
|
||||||
|
"medium": [1300, 750],
|
||||||
|
"large": [1900, 900],
|
||||||
|
"x-large": [2800, 1700],
|
||||||
|
"xx-large": [3600, 2000],
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const uids = {};
|
||||||
|
|
||||||
|
function uid(name) {
|
||||||
|
let id = uids[name] || 0;
|
||||||
|
uids[name] = id + 1;
|
||||||
|
return name + id;
|
||||||
|
}
|
||||||
|
|
||||||
|
function cascade(root, offset) {
|
||||||
|
const x = new Map;
|
||||||
|
const y = new Map;
|
||||||
|
return root.eachAfter(d => {
|
||||||
|
if (d.children && d.children.length !== 0) {
|
||||||
|
x.set(d, 1 + d3.max(d.children, c => c.x1 === d.x1 - offset ? x.get(c) : NaN));
|
||||||
|
y.set(d, 1 + d3.max(d.children, c => c.y1 === d.y1 - offset ? y.get(c) : NaN));
|
||||||
|
} else {
|
||||||
|
x.set(d, 0);
|
||||||
|
y.set(d, 0);
|
||||||
|
}
|
||||||
|
}).eachBefore(d => {
|
||||||
|
d.x1 -= 2 * offset * x.get(d);
|
||||||
|
d.y1 -= 2 * offset * y.get(d);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function cascadeTreemap(data, svg, width, height, tilingMode, treemapColor) {
|
||||||
|
const root = cascade(
|
||||||
|
d3.treemap()
|
||||||
|
.size([width, height])
|
||||||
|
.tile(TILING_MODES[tilingMode])
|
||||||
|
.paddingOuter(3)
|
||||||
|
.paddingTop(16)
|
||||||
|
.paddingInner(1)
|
||||||
|
.round(true)(
|
||||||
|
d3.hierarchy(data)
|
||||||
|
.sum(d => d.value)
|
||||||
|
.sort((a, b) => b.value - a.value)
|
||||||
|
),
|
||||||
|
3 // treemap.paddingOuter
|
||||||
|
);
|
||||||
|
|
||||||
|
const maxDepth = Math.max(...root.descendants().map(d => d.depth));
|
||||||
|
const color = d3.scaleSequential([maxDepth, -1], COLORS[treemapColor]);
|
||||||
|
|
||||||
|
svg.append("filter")
|
||||||
|
.attr("id", "shadow")
|
||||||
|
.append("feDropShadow")
|
||||||
|
.attr("flood-opacity", 0.3)
|
||||||
|
.attr("dx", 0)
|
||||||
|
.attr("stdDeviation", 3);
|
||||||
|
|
||||||
|
const node = svg.selectAll("g")
|
||||||
|
.data(
|
||||||
|
d3.nest()
|
||||||
|
.key(d => d.depth).sortKeys(d3.ascending)
|
||||||
|
.entries(root.descendants())
|
||||||
|
)
|
||||||
|
.join("g")
|
||||||
|
.attr("filter", "url(#shadow)")
|
||||||
|
.selectAll("g")
|
||||||
|
.data(d => d.values)
|
||||||
|
.join("g")
|
||||||
|
.attr("transform", d => `translate(${d.x0},${d.y0})`);
|
||||||
|
|
||||||
|
node.append("title")
|
||||||
|
.text(d => `${d.ancestors().reverse().splice(1).map(d => d.data.name).join("/")}\n${humanFileSize(d.value)}`);
|
||||||
|
|
||||||
|
node.append("rect")
|
||||||
|
.attr("id", d => (d.nodeUid = uid("node")))
|
||||||
|
.attr("fill", d => color(d.depth))
|
||||||
|
.attr("width", d => d.x1 - d.x0)
|
||||||
|
.attr("height", d => d.y1 - d.y0);
|
||||||
|
|
||||||
|
node.append("clipPath")
|
||||||
|
.attr("id", d => (d.clipUid = uid("clip")))
|
||||||
|
.append("use")
|
||||||
|
.attr("href", d => `#${d.nodeUid}`);
|
||||||
|
|
||||||
|
node.append("text")
|
||||||
|
.attr("fill", d => d3.hsl(color(d.depth)).l > .5 ? "#333" : "#eee")
|
||||||
|
.attr("clip-path", d => `url(#${d.clipUid})`)
|
||||||
|
.selectAll("tspan")
|
||||||
|
.data(d => [d.data.name, humanFileSize(d.value)])
|
||||||
|
.join("tspan")
|
||||||
|
.text(d => d);
|
||||||
|
|
||||||
|
node.filter(d => d.children).selectAll("tspan")
|
||||||
|
.attr("dx", 3)
|
||||||
|
.attr("y", 13);
|
||||||
|
|
||||||
|
node.filter(d => !d.children).selectAll("tspan")
|
||||||
|
.attr("x", 3)
|
||||||
|
.attr("y", (d, i) => `${i === 0 ? 1.1 : 2.3}em`);
|
||||||
|
}
|
||||||
|
|
||||||
|
function flatTreemap(data, svg, width, height, groupingDepth, tilingMode, fillOpacity) {
|
||||||
|
const ordinalColor = d3.scaleOrdinal(d3.schemeCategory10);
|
||||||
|
|
||||||
|
const root = d3.treemap()
|
||||||
|
.tile(TILING_MODES[tilingMode])
|
||||||
|
.size([width, height])
|
||||||
|
.padding(1)
|
||||||
|
.round(true)(
|
||||||
|
d3.hierarchy(data)
|
||||||
|
.sum(d => d.value)
|
||||||
|
.sort((a, b) => b.value - a.value)
|
||||||
|
);
|
||||||
|
|
||||||
|
const leaf = svg.selectAll("g")
|
||||||
|
.data(root.leaves())
|
||||||
|
.join("g")
|
||||||
|
.attr("transform", d => `translate(${d.x0},${d.y0})`);
|
||||||
|
|
||||||
|
leaf.append("title")
|
||||||
|
.text(d => `${d.ancestors().reverse().map(d => d.data.name).join("/")}\n${humanFileSize(d.value)}`);
|
||||||
|
|
||||||
|
leaf.append("rect")
|
||||||
|
.attr("id", d => (d.leafUid = uid("leaf")))
|
||||||
|
.attr("fill", d => {
|
||||||
|
while (d.depth > groupingDepth) d = d.parent;
|
||||||
|
return ordinalColor(d.data.name);
|
||||||
|
})
|
||||||
|
.attr("fill-opacity", fillOpacity)
|
||||||
|
.attr("width", d => d.x1 - d.x0)
|
||||||
|
.attr("height", d => d.y1 - d.y0);
|
||||||
|
|
||||||
|
leaf.append("clipPath")
|
||||||
|
.attr("id", d => (d.clipUid = uid("clip")))
|
||||||
|
.append("use")
|
||||||
|
.attr("href", d => `#${d.leafUid}`);
|
||||||
|
|
||||||
|
leaf.append("text")
|
||||||
|
.attr("clip-path", d => `url(#${d.clipUid})`)
|
||||||
|
.selectAll("tspan")
|
||||||
|
.data(d => {
|
||||||
|
if (d.data.name === ".") {
|
||||||
|
d = d.parent;
|
||||||
|
}
|
||||||
|
return [d.data.name, humanFileSize(d.value)]
|
||||||
|
})
|
||||||
|
.join("tspan")
|
||||||
|
.attr("x", 2)
|
||||||
|
.attr("y", (d, i) => `${i === 0 ? 1.1 : 2.3}em`)
|
||||||
|
.text(d => d);
|
||||||
|
}
|
||||||
|
|
||||||
|
function exportTreemap(indexName, width, height) {
|
||||||
|
domtoimage.toBlob(document.getElementById("treemap"), {width: width, height: height})
|
||||||
|
.then(function (blob) {
|
||||||
|
let a = document.createElement("a");
|
||||||
|
let url = URL.createObjectURL(blob);
|
||||||
|
|
||||||
|
a.href = url;
|
||||||
|
a.download = `${indexName}_treemap.png`;
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
setTimeout(function () {
|
||||||
|
document.body.removeChild(a);
|
||||||
|
window.URL.revokeObjectURL(url);
|
||||||
|
}, 0);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "D3Treemap",
|
||||||
|
props: ["indexId"],
|
||||||
|
watch: {
|
||||||
|
indexId: function () {
|
||||||
|
this.update(this.indexId);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.update(this.indexId);
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
update(indexId) {
|
||||||
|
const width = SIZES[this.$store.state.optTreemapSize][0];
|
||||||
|
const height = SIZES[this.$store.state.optTreemapSize][1];
|
||||||
|
const tilingMode = this.$store.state.optTreemapTiling;
|
||||||
|
const groupingDepth = this.$store.state.optTreemapColorGroupingDepth;
|
||||||
|
const treemapColor = this.$store.state.optTreemapColor;
|
||||||
|
const treemapType = this.$store.state.optTreemapType;
|
||||||
|
|
||||||
|
const treemapSvg = d3.select("#treemap");
|
||||||
|
|
||||||
|
treemapSvg.selectAll("*").remove();
|
||||||
|
treemapSvg.attr("viewBox", [0, 0, width, height])
|
||||||
|
.attr("xmlns", "http://www.w3.org/2000/svg")
|
||||||
|
.attr("xmlns:xlink", "http://www.w3.org/1999/xlink")
|
||||||
|
.attr("version", "1.1")
|
||||||
|
.style("overflow", "visible")
|
||||||
|
.style("font", "10px sans-serif");
|
||||||
|
|
||||||
|
d3.csv(Sist2Api.getTreemapCsvUrl(indexId)).then(tabularData => {
|
||||||
|
tabularData.forEach(row => {
|
||||||
|
row.taxonomy = row.path.split("/");
|
||||||
|
row.size = Number(row.size);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (treemapType === "cascaded") {
|
||||||
|
const data = burrow(tabularData, false);
|
||||||
|
cascadeTreemap(data, treemapSvg, width, height, tilingMode, treemapColor);
|
||||||
|
} else {
|
||||||
|
const data = burrow(tabularData.sort((a, b) => b.taxonomy.length - a.taxonomy.length), true);
|
||||||
|
const fillOpacity = this.$store.state.optTheme === "black" ? 0.9 : 0.6;
|
||||||
|
flatTreemap(data, treemapSvg, width, height, groupingDepth, tilingMode, fillOpacity);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
downloadTreemap() {
|
||||||
|
const width = SIZES[this.$store.state.optTreemapSize][0];
|
||||||
|
const height = SIZES[this.$store.state.optTreemapSize][1];
|
||||||
|
|
||||||
|
exportTreemap(this.indexId, width, height);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
66
sist2-vue/src/components/DateSlider.vue
Normal file
66
sist2-vue/src/components/DateSlider.vue
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
<template>
|
||||||
|
<div id="dateSlider"></div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import noUiSlider from 'nouislider';
|
||||||
|
import 'nouislider/dist/nouislider.css';
|
||||||
|
import {humanDate} from "@/util";
|
||||||
|
import {mergeTooltips} from "@/util-js";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "DateSlider",
|
||||||
|
mounted() {
|
||||||
|
this.$store.subscribe((mutation) => {
|
||||||
|
if (mutation.type === "setDateBoundsMax") {
|
||||||
|
const elem = document.getElementById("dateSlider");
|
||||||
|
|
||||||
|
if (elem.children.length > 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const dateMax = this.$store.state.dateBoundsMax;
|
||||||
|
const dateMin = this.$store.state.dateBoundsMin;
|
||||||
|
|
||||||
|
const slider = noUiSlider.create(elem, {
|
||||||
|
start: [
|
||||||
|
this.$store.state.dateMin ? this.$store.state.dateMin : dateMin,
|
||||||
|
this.$store.state.dateMax ? this.$store.state.dateMax : dateMax
|
||||||
|
],
|
||||||
|
|
||||||
|
tooltips: [true, true],
|
||||||
|
behaviour: "drag-tap",
|
||||||
|
connect: true,
|
||||||
|
range: {
|
||||||
|
"min": dateMin,
|
||||||
|
"max": dateMax,
|
||||||
|
},
|
||||||
|
format: {
|
||||||
|
to: x => humanDate(x),
|
||||||
|
from: x => x
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
mergeTooltips(elem, 10, " - ", true)
|
||||||
|
|
||||||
|
elem.querySelectorAll('.noUi-connect')[0].classList.add("slider-color0")
|
||||||
|
|
||||||
|
slider.on("set", (values, handle, unencoded) => {
|
||||||
|
if (handle === 0) {
|
||||||
|
this.$store.commit("setDateMin", unencoded[0] === dateMin ? undefined : Math.round(unencoded[0]));
|
||||||
|
} else {
|
||||||
|
this.$store.commit("setDateMax", unencoded[1] >= dateMax ? undefined : Math.round(unencoded[1]));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
#dateSlider {
|
||||||
|
margin-top: 34px;
|
||||||
|
height: 12px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
22
sist2-vue/src/components/DebugIcon.vue
Normal file
22
sist2-vue/src/components/DebugIcon.vue
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
<template>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 309.998 309.998" fill="currentColor">
|
||||||
|
<path
|
||||||
|
d="M294.998,155.03H250v-48.82l39.714-39.715c5.858-5.857,5.858-15.356,0-21.213c-5.857-5.857-15.355-5.857-21.213,0 l-23.7,23.701c-12.885-37.2-48.274-63.984-89.802-63.984c-41.528,0-76.913,26.787-89.797,63.989L41.497,45.282 c-5.856-5.859-15.354-5.857-21.213,0s-5.858,15.355,0,21.213L60,106.212v48.818H15c-8.284,0-15,6.716-15,15c0,8.284,6.716,15,15,15 h45.134c0.855,16.314,5.849,31.551,13.944,44.68l-49.685,49.683c-5.858,5.857-5.858,15.354,0,21.213 c2.929,2.93,6.768,4.394,10.607,4.394c3.838,0,7.678-1.465,10.606-4.394l48.095-48.093c16.558,14.018,37.957,22.486,61.301,22.486 c0.019,0,0.037-0.001,0.057-0.001c0.011,0,0.022,0.002,0.033,0.002c0.019,0,0.037-0.003,0.056-0.003 c23.285-0.035,44.629-8.494,61.15-22.483l48.094,48.092c2.929,2.929,6.768,4.394,10.606,4.394c3.839,0,7.678-1.465,10.607-4.394 c5.858-5.858,5.858-15.355,0-21.213l-49.683-49.681c8.096-13.131,13.089-28.366,13.944-44.682h45.132c8.284,0,15-6.716,15-15 C309.998,161.746,303.282,155.03,294.998,155.03z M154.999,34.999c30.681,0,56.465,21.365,63.254,50H91.747 C98.535,56.364,124.318,34.999,154.999,34.999z M90,179.999v-9.272c0.011-0.232,0.035-0.462,0.035-0.696 c0-0.234-0.024-0.464-0.035-0.695v-54.336h50.092v128.254C111.415,236.494,90,210.708,90,179.999z M170.092,243.212V114.999H220 v54.297c-0.012,0.244-0.037,0.486-0.037,0.734c0,0.248,0.025,0.49,0.037,0.734v9.234C220,210.645,198.676,236.388,170.092,243.212z"/>
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "DebugIcon"
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
svg {
|
||||||
|
display: inline-block;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
39
sist2-vue/src/components/DebugInfo.vue
Normal file
39
sist2-vue/src/components/DebugInfo.vue
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
<template>
|
||||||
|
<b-card class="mb-4 mt-4">
|
||||||
|
<b-card-title><DebugIcon class="mr-1"></DebugIcon>{{ $t("debug") }}</b-card-title>
|
||||||
|
<p v-html="$t('debugDescription')"></p>
|
||||||
|
|
||||||
|
<b-card-body>
|
||||||
|
|
||||||
|
<!-- TODO: ES connectivity, Link to GH page -->
|
||||||
|
<b-table :items="tableItems" small borderless responsive="md" thead-class="hidden" class="mb-0"></b-table>
|
||||||
|
|
||||||
|
<hr />
|
||||||
|
<IndexDebugInfo v-for="idx of $store.state.sist2Info.indices" :key="idx.id" :index="idx" class="mt-2"></IndexDebugInfo>
|
||||||
|
</b-card-body>
|
||||||
|
</b-card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import IndexDebugInfo from "@/components/IndexDebugInfo";
|
||||||
|
import DebugIcon from "@/components/DebugIcon";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "DebugInfo.vue",
|
||||||
|
components: {DebugIcon, IndexDebugInfo},
|
||||||
|
computed: {
|
||||||
|
tableItems() {
|
||||||
|
return [
|
||||||
|
{key: "version", value: this.$store.state.sist2Info.version},
|
||||||
|
{key: "platform", value: this.$store.state.sist2Info.platform},
|
||||||
|
{key: "debugBinary", value: this.$store.state.sist2Info.debug},
|
||||||
|
{key: "sist2CommitHash", value: this.$store.state.sist2Info.sist2Hash},
|
||||||
|
{key: "libscanCommitHash", value: this.$store.state.sist2Info.libscanHash},
|
||||||
|
{key: "esIndex", value: this.$store.state.sist2Info.esIndex},
|
||||||
|
{key: "tagline", value: this.$store.state.sist2Info.tagline},
|
||||||
|
{key: "dev", value: this.$store.state.sist2Info.dev},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
37
sist2-vue/src/components/DisplayModeToggle.vue
Normal file
37
sist2-vue/src/components/DisplayModeToggle.vue
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
<template>
|
||||||
|
<b-button-group>
|
||||||
|
<b-button variant="primary" :title="$t('displayMode.list')" :pressed="optDisplay==='list'"
|
||||||
|
@click="setOptDisplay('list')">
|
||||||
|
<svg width="20px" height="20px" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M80 368H16a16 16 0 0 0-16 16v64a16 16 0 0 0 16 16h64a16 16 0 0 0 16-16v-64a16 16 0 0 0-16-16zm0-320H16A16 16 0 0 0 0 64v64a16 16 0 0 0 16 16h64a16 16 0 0 0 16-16V64a16 16 0 0 0-16-16zm0 160H16a16 16 0 0 0-16 16v64a16 16 0 0 0 16 16h64a16 16 0 0 0 16-16v-64a16 16 0 0 0-16-16zm416 176H176a16 16 0 0 0-16 16v32a16 16 0 0 0 16 16h320a16 16 0 0 0 16-16v-32a16 16 0 0 0-16-16zm0-320H176a16 16 0 0 0-16 16v32a16 16 0 0 0 16 16h320a16 16 0 0 0 16-16V80a16 16 0 0 0-16-16zm0 160H176a16 16 0 0 0-16 16v32a16 16 0 0 0 16 16h320a16 16 0 0 0 16-16v-32a16 16 0 0 0-16-16z"></path>
|
||||||
|
</svg>
|
||||||
|
</b-button>
|
||||||
|
|
||||||
|
<b-button variant="primary" :title="$t('displayMode.grid')" :pressed="optDisplay==='grid'"
|
||||||
|
@click="setOptDisplay('grid')">
|
||||||
|
<svg width="20px" height="20px" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M149.333 56v80c0 13.255-10.745 24-24 24H24c-13.255 0-24-10.745-24-24V56c0-13.255 10.745-24 24-24h101.333c13.255 0 24 10.745 24 24zm181.334 240v-80c0-13.255-10.745-24-24-24H205.333c-13.255 0-24 10.745-24 24v80c0 13.255 10.745 24 24 24h101.333c13.256 0 24.001-10.745 24.001-24zm32-240v80c0 13.255 10.745 24 24 24H488c13.255 0 24-10.745 24-24V56c0-13.255-10.745-24-24-24H386.667c-13.255 0-24 10.745-24 24zm-32 80V56c0-13.255-10.745-24-24-24H205.333c-13.255 0-24 10.745-24 24v80c0 13.255 10.745 24 24 24h101.333c13.256 0 24.001-10.745 24.001-24zm-205.334 56H24c-13.255 0-24 10.745-24 24v80c0 13.255 10.745 24 24 24h101.333c13.255 0 24-10.745 24-24v-80c0-13.255-10.745-24-24-24zM0 376v80c0 13.255 10.745 24 24 24h101.333c13.255 0 24-10.745 24-24v-80c0-13.255-10.745-24-24-24H24c-13.255 0-24 10.745-24 24zm386.667-56H488c13.255 0 24-10.745 24-24v-80c0-13.255-10.745-24-24-24H386.667c-13.255 0-24 10.745-24 24v80c0 13.255 10.745 24 24 24zm0 160H488c13.255 0 24-10.745 24-24v-80c0-13.255-10.745-24-24-24H386.667c-13.255 0-24 10.745-24 24v80c0 13.255 10.745 24 24 24zM181.333 376v80c0 13.255 10.745 24 24 24h101.333c13.255 0 24-10.745 24-24v-80c0-13.255-10.745-24-24-24H205.333c-13.255 0-24 10.745-24 24z"></path>
|
||||||
|
</svg>
|
||||||
|
</b-button>
|
||||||
|
</b-button-group>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {mapGetters, mapMutations} from "vuex";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "DisplayModeToggle",
|
||||||
|
computed: {
|
||||||
|
...mapGetters(["optDisplay"])
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
...mapMutations(["setOptDisplay"])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
232
sist2-vue/src/components/DocCard.vue
Normal file
232
sist2-vue/src/components/DocCard.vue
Normal file
@@ -0,0 +1,232 @@
|
|||||||
|
<template>
|
||||||
|
<div class="doc-card" :class="{'sub-document': doc._props.isSubDocument}" :style="`width: ${width}px`">
|
||||||
|
<b-card
|
||||||
|
no-body
|
||||||
|
img-top
|
||||||
|
>
|
||||||
|
<!-- Info modal-->
|
||||||
|
<DocInfoModal :show="showInfo" :doc="doc" @close="showInfo = false"></DocInfoModal>
|
||||||
|
|
||||||
|
<ContentDiv :doc="doc"></ContentDiv>
|
||||||
|
|
||||||
|
<!-- Thumbnail-->
|
||||||
|
<div v-if="doc._props.hasThumbnail" class="img-wrapper" @mouseenter="onTnEnter()" @mouseleave="onTnLeave()">
|
||||||
|
<div v-if="doc._props.isAudio" class="card-img-overlay" :class="{'small-badge': smallBadge}">
|
||||||
|
<span class="badge badge-resolution">{{ humanTime(doc._source.duration) }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="doc._props.isImage && !hover" class="card-img-overlay" :class="{'small-badge': smallBadge}">
|
||||||
|
<span class="badge badge-resolution">{{ `${doc._source.width}x${doc._source.height}` }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="(doc._props.isVideo || doc._props.isGif) && doc._source.duration > 0 && !hover" class="card-img-overlay"
|
||||||
|
:class="{'small-badge': smallBadge}">
|
||||||
|
<span class="badge badge-resolution">{{ humanTime(doc._source.duration) }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="doc._props.isPlayableVideo" class="play">
|
||||||
|
<svg viewBox="0 0 494.942 494.942" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="m35.353 0 424.236 247.471-424.236 247.471z"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<img v-if="doc._props.isPlayableImage || doc._props.isPlayableVideo"
|
||||||
|
:src="(doc._props.isGif && hover) ? `f/${doc._id}` : `t/${doc._source.index}/${doc._id}`"
|
||||||
|
alt=""
|
||||||
|
class="pointer fit card-img-top" @click="onThumbnailClick()">
|
||||||
|
<img v-else :src="`t/${doc._source.index}/${doc._id}`" alt=""
|
||||||
|
class="fit card-img-top">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Audio player-->
|
||||||
|
<audio v-if="doc._props.isAudio" ref="audio" preload="none" class="audio-fit fit" controls :type="doc._source.mime"
|
||||||
|
:src="`f/${doc._id}`"
|
||||||
|
@play="onAudioPlay()"></audio>
|
||||||
|
|
||||||
|
<b-card-body class="padding-03">
|
||||||
|
|
||||||
|
<!-- Title line -->
|
||||||
|
<div style="display: flex">
|
||||||
|
<span class="info-icon" @click="onInfoClick()"></span>
|
||||||
|
<DocFileTitle :doc="doc"></DocFileTitle>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Tags -->
|
||||||
|
<div class="card-text">
|
||||||
|
<TagContainer :hit="doc"></TagContainer>
|
||||||
|
</div>
|
||||||
|
</b-card-body>
|
||||||
|
</b-card>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {ext, humanFileSize, humanTime} from "@/util";
|
||||||
|
import TagContainer from "@/components/TagContainer.vue";
|
||||||
|
import DocFileTitle from "@/components/DocFileTitle.vue";
|
||||||
|
import DocInfoModal from "@/components/DocInfoModal.vue";
|
||||||
|
import ContentDiv from "@/components/ContentDiv.vue";
|
||||||
|
|
||||||
|
|
||||||
|
export default {
|
||||||
|
components: {ContentDiv, DocInfoModal, DocFileTitle, TagContainer},
|
||||||
|
props: ["doc", "width"],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
ext: ext,
|
||||||
|
showInfo: false,
|
||||||
|
hover: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
placeHolderStyle() {
|
||||||
|
|
||||||
|
const tokens = this.doc._source.thumbnail.split(",");
|
||||||
|
const w = Number(tokens[0]);
|
||||||
|
const h = Number(tokens[1]);
|
||||||
|
|
||||||
|
const MAX_HEIGHT = 400;
|
||||||
|
|
||||||
|
return {
|
||||||
|
height: `${Math.min((h / w) * this.width, MAX_HEIGHT)}px`,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
smallBadge() {
|
||||||
|
return this.width < 150;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
humanFileSize: humanFileSize,
|
||||||
|
humanTime: humanTime,
|
||||||
|
onInfoClick() {
|
||||||
|
this.showInfo = true;
|
||||||
|
},
|
||||||
|
async onThumbnailClick() {
|
||||||
|
this.$store.commit("setUiLightboxSlide", this.doc._seq);
|
||||||
|
await this.$store.dispatch("showLightbox");
|
||||||
|
},
|
||||||
|
onAudioPlay() {
|
||||||
|
document.getElementsByTagName("audio").forEach((el) => {
|
||||||
|
if (el !== this.$refs["audio"]) {
|
||||||
|
el.pause();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onTnEnter() {
|
||||||
|
this.hover = true;
|
||||||
|
},
|
||||||
|
onTnLeave() {
|
||||||
|
this.hover = false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.img-wrapper {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-wrapper:hover svg {
|
||||||
|
fill: rgba(0, 0, 0, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.pointer {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fit {
|
||||||
|
display: block;
|
||||||
|
min-width: 64px;
|
||||||
|
max-width: 100%;
|
||||||
|
/*max-height: 400px;*/
|
||||||
|
margin: 0 auto 0;
|
||||||
|
width: auto;
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
.card-img-top {
|
||||||
|
border-top-left-radius: 0;
|
||||||
|
border-top-right-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.padding-03 {
|
||||||
|
padding: 0.3rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card {
|
||||||
|
margin-top: 1em;
|
||||||
|
margin-left: 0;
|
||||||
|
margin-right: 0;
|
||||||
|
box-shadow: 0 .125rem .25rem rgba(0, 0, 0, .08) !important;
|
||||||
|
border-radius: 0;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-body {
|
||||||
|
padding: 0.3rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.thumbnail-placeholder {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-img-overlay {
|
||||||
|
pointer-events: none;
|
||||||
|
padding: 0.75rem;
|
||||||
|
bottom: unset;
|
||||||
|
top: 0;
|
||||||
|
left: unset;
|
||||||
|
right: unset;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-resolution {
|
||||||
|
color: #212529;
|
||||||
|
background-color: #FFC107;
|
||||||
|
}
|
||||||
|
|
||||||
|
.play {
|
||||||
|
position: absolute;
|
||||||
|
width: 25px;
|
||||||
|
height: 25px;
|
||||||
|
left: 50%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.play svg {
|
||||||
|
fill: rgba(0, 0, 0, 0.7);
|
||||||
|
}
|
||||||
|
|
||||||
|
.doc-card {
|
||||||
|
padding-left: 3px;
|
||||||
|
padding-right: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.small-badge {
|
||||||
|
padding: 1px 3px;
|
||||||
|
font-size: 70%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.audio-fit {
|
||||||
|
height: 39px;
|
||||||
|
vertical-align: bottom;
|
||||||
|
display: inline;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sub-document .card {
|
||||||
|
background: #AB47BC1F !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .sub-document .card {
|
||||||
|
background: #37474F !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sub-document .fit {
|
||||||
|
padding: 4px 4px 0 4px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
69
sist2-vue/src/components/DocCardWall.vue
Normal file
69
sist2-vue/src/components/DocCardWall.vue
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
<template>
|
||||||
|
<GridLayout
|
||||||
|
ref="grid-layout"
|
||||||
|
:options="gridOptions"
|
||||||
|
@append="append"
|
||||||
|
@layout-complete="$emit('layout-complete')"
|
||||||
|
>
|
||||||
|
<DocCard v-for="doc in docs" :key="doc._id" :doc="doc" :width="width"></DocCard>
|
||||||
|
</GridLayout>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import Vue from "vue";
|
||||||
|
import DocCard from "@/components/DocCard";
|
||||||
|
|
||||||
|
import VueInfiniteGrid from "@egjs/vue-infinitegrid";
|
||||||
|
|
||||||
|
Vue.use(VueInfiniteGrid);
|
||||||
|
|
||||||
|
export default Vue.extend({
|
||||||
|
components: {
|
||||||
|
DocCard,
|
||||||
|
},
|
||||||
|
props: ["docs", "append"],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
width: 0,
|
||||||
|
gridOptions: {
|
||||||
|
align: "center",
|
||||||
|
margin: 0,
|
||||||
|
transitionDuration: 0,
|
||||||
|
isOverflowScroll: false,
|
||||||
|
isConstantSize: false,
|
||||||
|
useFit: false,
|
||||||
|
// Indicates whether keep the number of DOMs is maintained. If the useRecycle value is 'true', keep the number
|
||||||
|
// of DOMs is maintained. If the useRecycle value is 'false', the number of DOMs will increase as card elements
|
||||||
|
// are added.
|
||||||
|
useRecycle: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
colCount() {
|
||||||
|
const columns = this.$store.getters["optColumns"];
|
||||||
|
|
||||||
|
if (columns === "auto") {
|
||||||
|
return Math.round(this.$refs["grid-layout"].$el.scrollWidth / 300)
|
||||||
|
}
|
||||||
|
return columns;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.width = this.$refs["grid-layout"].$el.scrollWidth / this.colCount;
|
||||||
|
|
||||||
|
if (this.colCount === 1) {
|
||||||
|
this.$refs["grid-layout"].$el.classList.add("grid-single-column");
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$store.subscribe((mutation) => {
|
||||||
|
if (mutation.type === "busUpdateWallItems" && this.$refs["grid-layout"]) {
|
||||||
|
this.$refs["grid-layout"].updateItems();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
</style>
|
||||||
63
sist2-vue/src/components/DocFileTitle.vue
Normal file
63
sist2-vue/src/components/DocFileTitle.vue
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
<template>
|
||||||
|
<a :href="`f/${doc._id}`" class="file-title-anchor" target="_blank">
|
||||||
|
<div class="file-title" :title="doc._source.path + '/' + doc._source.name + ext(doc)"
|
||||||
|
v-html="fileName() + ext(doc)"></div>
|
||||||
|
</a>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {ext} from "@/util";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "DocFileTitle",
|
||||||
|
props: ["doc"],
|
||||||
|
methods: {
|
||||||
|
ext: ext,
|
||||||
|
fileName() {
|
||||||
|
if (!this.doc.highlight) {
|
||||||
|
return this.doc._source.name;
|
||||||
|
}
|
||||||
|
if (this.doc.highlight["name.nGram"]) {
|
||||||
|
return this.doc.highlight["name.nGram"];
|
||||||
|
}
|
||||||
|
if (this.doc.highlight.name) {
|
||||||
|
return this.doc.highlight.name;
|
||||||
|
}
|
||||||
|
return this.doc._source.name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.file-title-anchor {
|
||||||
|
max-width: calc(100% - 1.2rem);
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-title {
|
||||||
|
width: 100%;
|
||||||
|
line-height: 1rem;
|
||||||
|
height: 1.1rem;
|
||||||
|
white-space: nowrap;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
overflow: hidden;
|
||||||
|
font-size: 16px;
|
||||||
|
font-family: "Source Sans Pro", sans-serif;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .file-title {
|
||||||
|
color: #ddd;
|
||||||
|
}
|
||||||
|
.theme-black .file-title:hover {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-light .file-title {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
.doc-card .file-title {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
33
sist2-vue/src/components/DocInfoModal.vue
Normal file
33
sist2-vue/src/components/DocInfoModal.vue
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
<template>
|
||||||
|
<b-modal :visible="show" size="lg" :hide-footer="true" static lazy @close="$emit('close')" @hide="$emit('close')"
|
||||||
|
>
|
||||||
|
<template #modal-title>
|
||||||
|
<h5 class="modal-title" :title="doc._source.name + ext(doc)">{{ doc._source.name + ext(doc) }}</h5>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<img v-if="doc._props.hasThumbnail" :src="`t/${doc._source.index}/${doc._id}`" alt="" class="fit card-img-top">
|
||||||
|
|
||||||
|
<InfoTable :doc="doc"></InfoTable>
|
||||||
|
|
||||||
|
<LazyContentDiv :doc-id="doc._id"></LazyContentDiv>
|
||||||
|
</b-modal>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {ext} from "@/util";
|
||||||
|
import InfoTable from "@/components/InfoTable";
|
||||||
|
import LazyContentDiv from "@/components/LazyContentDiv";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "DocInfoModal",
|
||||||
|
components: {LazyContentDiv, InfoTable},
|
||||||
|
props: ["doc", "show"],
|
||||||
|
methods: {
|
||||||
|
ext: ext,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
45
sist2-vue/src/components/DocList.vue
Normal file
45
sist2-vue/src/components/DocList.vue
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<template>
|
||||||
|
<b-list-group class="mt-3">
|
||||||
|
<DocListItem v-for="doc in docs" :key="doc._id" :doc="doc"></DocListItem>
|
||||||
|
</b-list-group>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import DocListItem from "@/components/DocListItem.vue";
|
||||||
|
import Vue from "vue";
|
||||||
|
|
||||||
|
export default Vue.extend({
|
||||||
|
name: "DocList",
|
||||||
|
components: {DocListItem},
|
||||||
|
props: ["docs", "append"],
|
||||||
|
mounted() {
|
||||||
|
window.addEventListener("scroll", () => {
|
||||||
|
const threshold = 400;
|
||||||
|
const app = document.getElementById("app");
|
||||||
|
|
||||||
|
if ((window.innerHeight + window.scrollY) >= app.offsetHeight - threshold) {
|
||||||
|
this.append();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
|
||||||
|
.theme-black .list-group-item {
|
||||||
|
background: #212121;
|
||||||
|
color: #e0e0e0;
|
||||||
|
|
||||||
|
border-bottom: none;
|
||||||
|
border-left: none;
|
||||||
|
border-right: none;
|
||||||
|
border-radius: 0;
|
||||||
|
padding: .25rem 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .list-group-item:first-child {
|
||||||
|
border-top: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
180
sist2-vue/src/components/DocListItem.vue
Normal file
180
sist2-vue/src/components/DocListItem.vue
Normal file
@@ -0,0 +1,180 @@
|
|||||||
|
<template>
|
||||||
|
<b-list-group-item class="flex-column align-items-start mb-2" :class="{'sub-document': doc._props.isSubDocument}">
|
||||||
|
|
||||||
|
<!-- Info modal-->
|
||||||
|
<DocInfoModal :show="showInfo" :doc="doc" @close="showInfo = false"></DocInfoModal>
|
||||||
|
|
||||||
|
<div class="media ml-2">
|
||||||
|
<div v-if="doc._props.hasThumbnail" class="align-self-start mr-2 wrapper-sm">
|
||||||
|
<div class="img-wrapper">
|
||||||
|
<div v-if="doc._props.isPlayableVideo" class="play">
|
||||||
|
<svg viewBox="0 0 494.942 494.942" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="m35.353 0 424.236 247.471-424.236 247.471z"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<img v-if="doc._props.isPlayableImage || doc._props.isPlayableVideo"
|
||||||
|
:src="(doc._props.isGif && hover) ? `f/${doc._id}` : `t/${doc._source.index}/${doc._id}`"
|
||||||
|
alt=""
|
||||||
|
class="pointer fit-sm" @click="onThumbnailClick()">
|
||||||
|
<img v-else :src="`t/${doc._source.index}/${doc._id}`" alt=""
|
||||||
|
class="fit-sm">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div v-else class="file-icon-wrapper" style="">
|
||||||
|
<FileIcon></FileIcon>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="doc-line ml-3">
|
||||||
|
<div style="display: flex">
|
||||||
|
<span class="info-icon" @click="showInfo = true"></span>
|
||||||
|
<DocFileTitle :doc="doc"></DocFileTitle>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Content highlight -->
|
||||||
|
<ContentDiv :doc="doc"></ContentDiv>
|
||||||
|
|
||||||
|
<div class="path-row">
|
||||||
|
<div class="path-line" v-html="path()"></div>
|
||||||
|
<TagContainer :hit="doc"></TagContainer>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="doc._source.pages || doc._source.author" class="path-row text-muted">
|
||||||
|
<span v-if="doc._source.pages">{{ doc._source.pages }} {{
|
||||||
|
doc._source.pages > 1 ? $t("pages") : $t("page")
|
||||||
|
}}</span>
|
||||||
|
<span v-if="doc._source.author && doc._source.pages" class="mx-1">-</span>
|
||||||
|
<span v-if="doc._source.author">{{ doc._source.author }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</b-list-group-item>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import TagContainer from "@/components/TagContainer";
|
||||||
|
import DocFileTitle from "@/components/DocFileTitle";
|
||||||
|
import DocInfoModal from "@/components/DocInfoModal";
|
||||||
|
import ContentDiv from "@/components/ContentDiv";
|
||||||
|
import FileIcon from "@/components/FileIcon";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "DocListItem",
|
||||||
|
components: {FileIcon, ContentDiv, DocInfoModal, DocFileTitle, TagContainer},
|
||||||
|
props: ["doc"],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
hover: false,
|
||||||
|
showInfo: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
async onThumbnailClick() {
|
||||||
|
this.$store.commit("setUiLightboxSlide", this.doc._seq);
|
||||||
|
await this.$store.dispatch("showLightbox");
|
||||||
|
},
|
||||||
|
path() {
|
||||||
|
if (!this.doc.highlight) {
|
||||||
|
return this.doc._source.path + "/"
|
||||||
|
}
|
||||||
|
if (this.doc.highlight["path.text"]) {
|
||||||
|
return this.doc.highlight["path.text"] + "/"
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.doc.highlight["path.nGram"]) {
|
||||||
|
return this.doc.highlight["path.nGram"] + "/"
|
||||||
|
}
|
||||||
|
return this.doc._source.path + "/"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.sub-document {
|
||||||
|
background: #AB47BC1F !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .sub-document {
|
||||||
|
background: #37474F !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-group {
|
||||||
|
margin-top: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-group-item {
|
||||||
|
padding: .25rem 0.5rem;
|
||||||
|
|
||||||
|
box-shadow: 0 0.125rem 0.25rem rgb(0 0 0 / 8%) !important;
|
||||||
|
border-radius: 0;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.path-row {
|
||||||
|
display: -ms-flexbox;
|
||||||
|
display: flex;
|
||||||
|
-ms-flex-align: start;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.path-line {
|
||||||
|
color: #808080;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
margin-right: 0.3em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .path-line {
|
||||||
|
color: #bbb;
|
||||||
|
}
|
||||||
|
|
||||||
|
.play {
|
||||||
|
position: absolute;
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
left: 50%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.play svg {
|
||||||
|
fill: rgba(0, 0, 0, 0.7);
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-group-item .img-wrapper {
|
||||||
|
width: 88px;
|
||||||
|
height: 88px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fit-sm {
|
||||||
|
max-height: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
width: auto;
|
||||||
|
height: auto;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
margin: auto;
|
||||||
|
|
||||||
|
/*box-shadow: 2px 2px 4px rgba(0, 0, 0, 0.12);*/
|
||||||
|
}
|
||||||
|
|
||||||
|
.doc-line {
|
||||||
|
max-width: calc(100% - 88px - 1.5rem);
|
||||||
|
flex: 1;
|
||||||
|
vertical-align: middle;
|
||||||
|
margin-top: auto;
|
||||||
|
margin-bottom: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-icon-wrapper {
|
||||||
|
width: calc(88px + .5rem);
|
||||||
|
height: 88px;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
33
sist2-vue/src/components/FileIcon.vue
Normal file
33
sist2-vue/src/components/FileIcon.vue
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
<template>
|
||||||
|
<svg class="file-icon" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 50 50">
|
||||||
|
<path
|
||||||
|
fill="currentColor"
|
||||||
|
d="M 7 2 L 7 48 L 43 48 L 43 14.59375 L 42.71875 14.28125 L 30.71875 2.28125 L 30.40625 2 Z M 9 4 L 29 4 L 29 16 L 41 16 L 41 46 L 9 46 Z M 31 5.4375 L 39.5625 14 L 31 14 Z"/>
|
||||||
|
</svg>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "FileIcon"
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.file-icon {
|
||||||
|
position: absolute;
|
||||||
|
max-height: 100%;
|
||||||
|
max-width: 100%;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .file-icon {
|
||||||
|
color: #ffffff50;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-light .file-icon {
|
||||||
|
color: #000000a0;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
23
sist2-vue/src/components/GearIcon.vue
Normal file
23
sist2-vue/src/components/GearIcon.vue
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
<template>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1000 1000" fill="currentColor">
|
||||||
|
<g transform="translate(0.000000,512.000000) scale(0.100000,-0.100000)">
|
||||||
|
<path
|
||||||
|
d="M4568.5,5011c-73.2-7.7-154-25-177.1-36.6c-84.7-46.2-102-119.4-159.8-689.2s-65.5-610.3-159.8-670c-23.1-15.4-125.1-55.8-225.3-90.5c-100.1-32.7-290.7-111.7-423.6-175.2c-319.6-152.1-315.8-152.1-619.9,94.3c-718.1,583.3-650.7,535.2-747,535.2c-77,0-104-11.6-184.8-77c-157.9-127.1-410.1-375.4-567.9-558.3c-155.9-177.1-190.6-250.3-159.8-344.6c9.6-27,165.6-227.2,344.6-446.7c181-219.5,342.7-425.5,360-458.2c52-88.6,42.3-150.2-50.1-335c-73.2-148.3-144.4-325.4-252.2-623.8c-17.3-50-57.8-113.6-88.6-138.6c-63.5-53.9-59.7-53.9-695-117.4c-527.5-52-577.6-65.5-627.6-179c-46.2-105.9-46.2-1057,0-1162.9c50-113.6,98.2-127.1,646.9-181c271.5-25,523.7-52,560.2-57.8c111.7-17.3,179.1-107.8,259.9-344.6c38.5-115.5,119.4-310,177.1-431.3c57.8-119.4,104-240.7,104-269.5c0-78.9-42.4-140.5-394.7-568c-179-219.5-335-419.7-344.6-446.6c-30.8-94.3,3.9-167.5,159.8-344.6c157.9-181,410.1-429.3,564.1-554.5c96.3-78.9,188.7-105.9,265.7-75.1c26.9,11.6,234.9,173.3,462.1,360c227.2,188.7,433.2,348.5,458.2,358.1c82.8,30.8,136.7,17.3,354.3-86.6c119.4-57.8,308-136.7,419.7-175.2c111.7-38.5,221.4-82.8,244.5-98.2c94.3-59.7,102-100.1,159.8-670c61.6-606.5,73.2-648.8,188.7-700.8c105.9-46.2,1057-46.2,1162.9,0c115.5,52,127.1,94.3,188.7,700.8c57.8,569.9,65.4,610.3,159.8,670c23.1,15.4,132.9,59.7,244.5,98.2s300.3,117.4,417.8,175.2c219.5,104,273.4,117.5,356.2,86.6c25-9.6,231-169.4,458.2-358.1c227.2-186.8,435.1-348.5,462.1-360c77-28.9,169.4-3.9,265.7,75.1c152.1,121.3,442.8,410.1,583.4,577.6c140.6,163.6,173.3,242.6,136.7,333.1c-11.6,27-173.3,234.9-360,462.1c-188.7,227.2-348.5,433.2-358.1,458.2c-30.8,82.8-17.3,136.7,86.6,356.2c57.8,117.4,138.6,311.9,177.1,427.4c80.9,236.8,148.3,327.3,259.9,344.6c36.6,5.8,288.8,32.7,562.2,59.7c308,28.9,517.9,59.7,550.6,77c30.8,15.4,71.2,59.7,90.5,100.1c32.8,65.4,36.6,123.2,34.7,573.7c0,562.2-11.5,627.6-115.5,687.3c-46.2,27-188.7,48.1-612.2,90.5c-573.7,59.7-614.2,67.4-673.8,161.7c-15.4,23.1-59.7,132.9-98.2,244.5s-117.4,300.3-175.2,417.8c-57.8,119.4-104,240.7-104,271.5c0,80.9,40.4,138.6,394.7,569.9c181,219.5,335,419.7,344.6,446.7c30.8,94.3-3.9,167.5-159.8,344.6c-157.9,181-410.1,429.3-564.1,554.5c-96.3,78.9-188.7,104-265.7,75.1c-27-11.6-234.9-173.3-462.1-360c-227.2-188.7-433.2-348.5-458.2-358.1c-80.9-30.8-130.9-19.2-371.6,96.3c-130.9,61.6-325.4,142.5-431.3,177.1c-217.5,71.2-308,140.5-325.4,250.3c-5.8,36.6-32.7,288.8-57.8,560.3c-53.9,550.6-67.4,596.8-181,645C5502.3,5018.7,4807.3,5036,4568.5,5011z M5463.8,1897.8c502.5-127.1,954.9-494.8,1184-960.7c446.7-914.5,78.9-2011.9-824-2460.5c-1053.1-521.8-2308.4,52-2604.9,1189.8c-71.2,277.2-71.2,629.6,0,904.9c192.5,737.4,814.4,1284.2,1569.1,1376.6C4974.8,1971,5255.9,1949.8,5463.8,1897.8z"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "GearIcon"
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
svg {
|
||||||
|
display: inline-block;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
margin-top: -4px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
72
sist2-vue/src/components/HelpDialog.vue
Normal file
72
sist2-vue/src/components/HelpDialog.vue
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
<template>
|
||||||
|
<b-modal :visible="show" size="lg" :hide-footer="true" static :title="$t('help.help')"
|
||||||
|
@close="$emit('close')"
|
||||||
|
@hide="$emit('close')"
|
||||||
|
>
|
||||||
|
<h2>{{$t("help.simpleSearch")}}</h2>
|
||||||
|
|
||||||
|
<table class="table">
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td><code>+</code></td>
|
||||||
|
<td>{{$t("help.and")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>|</code></td>
|
||||||
|
<td>{{$t("help.or")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>-</code></td>
|
||||||
|
<td>{{$t("help.not")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>""</code></td>
|
||||||
|
<td>{{$t("help.quotes")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>{{$t("help.term")}}*</code></td>
|
||||||
|
<td>{{$t("help.prefix")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>(</code> {{$t("and")}} <code>)</code></td>
|
||||||
|
<td>{{$t("help.parens")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>{{$t("help.term")}}~N</code></td>
|
||||||
|
<td>{{$t("help.tildeTerm")}}</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>"..."~N</code></td>
|
||||||
|
<td>{{$t("help.tildePhrase")}}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<p v-html="$t('help.example1')"></p>
|
||||||
|
|
||||||
|
<p v-html="$t('help.defaultOperator')"></p>
|
||||||
|
|
||||||
|
<p v-html="$t('help.fuzzy')"></p>
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
<p v-html="$t('help.moreInfoSimple')"></p>
|
||||||
|
|
||||||
|
<p></p>
|
||||||
|
|
||||||
|
<h2>{{$t("help.advancedSearch")}}</h2>
|
||||||
|
<p v-html="$t('help.moreInfoAdvanced')"></p>
|
||||||
|
|
||||||
|
</b-modal>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "HelpDialog",
|
||||||
|
props: ["show"]
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
30
sist2-vue/src/components/IndexDebugInfo.vue
Normal file
30
sist2-vue/src/components/IndexDebugInfo.vue
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<h4>[{{ index.name }}]</h4>
|
||||||
|
<b-table :items="tableItems" small borderless responsive="md" thead-class="hidden" class="mb-0"></b-table>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {humanDate} from "@/util";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "IndexDebugInfo",
|
||||||
|
props: ["index"],
|
||||||
|
computed: {
|
||||||
|
tableItems() {
|
||||||
|
return [
|
||||||
|
{key: this.$t("name"), value: this.index.name},
|
||||||
|
{key: this.$t("id"), value: this.index.id},
|
||||||
|
{key: this.$t("indexVersion"), value: this.index.version},
|
||||||
|
{key: this.$t("rewriteUrl"), value: this.index.rewriteUrl},
|
||||||
|
{key: this.$t("timestamp"), value: humanDate(this.index.timestamp)},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
95
sist2-vue/src/components/IndexPicker.vue
Normal file
95
sist2-vue/src/components/IndexPicker.vue
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
<template>
|
||||||
|
<div v-if="isMobile">
|
||||||
|
<b-form-select
|
||||||
|
:value="selectedIndicesIds"
|
||||||
|
@change="onSelect($event)"
|
||||||
|
:options="indices" multiple :select-size="6" text-field="name"
|
||||||
|
value-field="id"></b-form-select>
|
||||||
|
</div>
|
||||||
|
<div v-else>
|
||||||
|
<b-list-group id="index-picker-desktop">
|
||||||
|
<b-list-group-item
|
||||||
|
v-for="idx in indices"
|
||||||
|
@click="toggleIndex(idx)"
|
||||||
|
class="d-flex justify-content-between align-items-center list-group-item-action pointer">
|
||||||
|
<div class="d-flex">
|
||||||
|
<b-checkbox @change="toggleIndex(idx)" :checked="isSelected(idx)"></b-checkbox>
|
||||||
|
{{ idx.name }}
|
||||||
|
<span class="text-muted timestamp-text ml-2">{{ formatIdxDate(idx.timestamp) }}</span>
|
||||||
|
</div>
|
||||||
|
<b-badge class="version-badge">v{{ idx.version }}</b-badge>
|
||||||
|
</b-list-group-item>
|
||||||
|
</b-list-group>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import SmallBadge from "./SmallBadge.vue"
|
||||||
|
import {mapActions, mapGetters} from "vuex";
|
||||||
|
import Vue from "vue";
|
||||||
|
import {format} from "date-fns";
|
||||||
|
|
||||||
|
export default Vue.extend({
|
||||||
|
components: {
|
||||||
|
SmallBadge
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
loading: true,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
...mapGetters([
|
||||||
|
"indices", "selectedIndices"
|
||||||
|
]),
|
||||||
|
selectedIndicesIds() {
|
||||||
|
return this.selectedIndices.map(idx => idx.id)
|
||||||
|
},
|
||||||
|
isMobile() {
|
||||||
|
return window.innerWidth <= 650;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
...mapActions({
|
||||||
|
setSelectedIndices: "setSelectedIndices"
|
||||||
|
}),
|
||||||
|
onSelect(value) {
|
||||||
|
this.setSelectedIndices(this.indices.filter(idx => value.includes(idx.id)));
|
||||||
|
},
|
||||||
|
formatIdxDate(timestamp: number): string {
|
||||||
|
return format(new Date(timestamp * 1000), "yyyy-MM-dd");
|
||||||
|
},
|
||||||
|
toggleIndex(index) {
|
||||||
|
if (this.isSelected(index)) {
|
||||||
|
this.setSelectedIndices(this.selectedIndices.filter(idx => idx.id != index.id));
|
||||||
|
} else {
|
||||||
|
this.setSelectedIndices([index, ...this.selectedIndices]);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isSelected(index) {
|
||||||
|
return this.selectedIndices.find(idx => idx.id == index.id) != null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.timestamp-text {
|
||||||
|
line-height: 24px;
|
||||||
|
font-size: 80%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.version-badge {
|
||||||
|
color: #222 !important;
|
||||||
|
background: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-group-item {
|
||||||
|
padding: 0.2em 0.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
#index-picker-desktop {
|
||||||
|
overflow-y: auto;
|
||||||
|
max-height: 132px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
94
sist2-vue/src/components/InfoTable.vue
Normal file
94
sist2-vue/src/components/InfoTable.vue
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
<template>
|
||||||
|
<b-table :items="tableItems" small borderless responsive="md" thead-class="hidden" class="mb-0 mt-4">
|
||||||
|
|
||||||
|
<template #cell(value)="data">
|
||||||
|
<span v-if="'html' in data.item" v-html="data.item.html"></span>
|
||||||
|
<span v-else>{{ data.value }}</span>
|
||||||
|
</template>
|
||||||
|
</b-table>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {humanDate, humanFileSize} from "@/util";
|
||||||
|
|
||||||
|
function makeGpsLink(latitude, longitude) {
|
||||||
|
|
||||||
|
if (isNaN(latitude) || isNaN(longitude)) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return `<a target="_blank" href="https://maps.google.com/?q=${latitude},${longitude}&ll=${latitude},${longitude}&t=k&z=17">${latitude}, ${longitude}</a>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function dmsToDecimal(dms, ref) {
|
||||||
|
const tokens = dms.split(",")
|
||||||
|
|
||||||
|
const d = Number(tokens[0].trim().split(":")[0]) / Number(tokens[0].trim().split(":")[1])
|
||||||
|
const m = Number(tokens[1].trim().split(":")[0]) / Number(tokens[1].trim().split(":")[1])
|
||||||
|
const s = Number(tokens[2].trim().split(":")[0]) / Number(tokens[2].trim().split(":")[1])
|
||||||
|
|
||||||
|
return (d + (m / 60) + (s / 3600)) * (ref === "S" || ref === "W" ? -1 : 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "InfoTable",
|
||||||
|
props: ["doc"],
|
||||||
|
computed: {
|
||||||
|
tableItems() {
|
||||||
|
const src = this.doc._source;
|
||||||
|
|
||||||
|
const items = [
|
||||||
|
{key: "index", value: `[${this.$store.getters.indexMap[src.index].name}]`},
|
||||||
|
{key: "mtime", value: humanDate(src.mtime)},
|
||||||
|
{key: "mime", value: src.mime},
|
||||||
|
{key: "size", value: humanFileSize(src.size)},
|
||||||
|
{key: "path", value: src.path},
|
||||||
|
];
|
||||||
|
|
||||||
|
if ("width" in this.doc._source) {
|
||||||
|
items.push({
|
||||||
|
key: "image size",
|
||||||
|
value: `${src.width}x${src.height}`
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const fields = [
|
||||||
|
"title", "duration", "audioc", "videoc",
|
||||||
|
"bitrate", "artist", "album", "album_artist", "genre", "font_name", "author",
|
||||||
|
"modified_by", "pages", "tag",
|
||||||
|
"exif_make", "exif_software", "exif_exposure_time", "exif_fnumber", "exif_focal_length",
|
||||||
|
"exif_user_comment", "exif_iso_speed_ratings", "exif_model", "exif_datetime",
|
||||||
|
"checksum"
|
||||||
|
];
|
||||||
|
|
||||||
|
fields.forEach(field => {
|
||||||
|
if (field in src) {
|
||||||
|
items.push({key: field, value: src[field]});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Exif GPS
|
||||||
|
if ("exif_gps_longitude_dec" in src) {
|
||||||
|
items.push({
|
||||||
|
key: "Exif GPS",
|
||||||
|
html: makeGpsLink(src["exif_gps_latitude_dec"], src["exif_gps_longitude_dec"]),
|
||||||
|
});
|
||||||
|
} else if ("exif_gps_longitude_dms" in src) {
|
||||||
|
items.push({
|
||||||
|
key: "Exif GPS",
|
||||||
|
html: makeGpsLink(
|
||||||
|
dmsToDecimal(src["exif_gps_latitude_dms"], src["exif_gps_latitude_ref"]),
|
||||||
|
dmsToDecimal(src["exif_gps_longitude_dms"], src["exif_gps_longitude_ref"]),
|
||||||
|
),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
</style>
|
||||||
30
sist2-vue/src/components/LazyContentDiv.vue
Normal file
30
sist2-vue/src/components/LazyContentDiv.vue
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<template>
|
||||||
|
<Preloader v-if="loading"></Preloader>
|
||||||
|
<div v-else-if="content" class="content-div">{{ content }}</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
import Preloader from "@/components/Preloader";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "LazyContentDiv",
|
||||||
|
components: {Preloader},
|
||||||
|
props: ["docId"],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
content: "",
|
||||||
|
loading: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
Sist2Api.getDocInfo(this.docId).then(src => {
|
||||||
|
this.content = src.data.content;
|
||||||
|
this.loading = false;
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
</style>
|
||||||
129
sist2-vue/src/components/Lightbox.vue
Normal file
129
sist2-vue/src/components/Lightbox.vue
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<!-- TODO: Set slideshowTime as a configurable option-->
|
||||||
|
<FsLightbox
|
||||||
|
:key="lightboxKey"
|
||||||
|
:toggler="showLightbox"
|
||||||
|
:sources="lightboxSources"
|
||||||
|
:thumbs="lightboxThumbs"
|
||||||
|
:captions="lightboxCaptions"
|
||||||
|
:types="lightboxTypes"
|
||||||
|
:source-index="lightboxSlide"
|
||||||
|
:custom-toolbar-buttons="customButtons"
|
||||||
|
:slideshow-time="1000 * 10"
|
||||||
|
:zoom-increment="0.5"
|
||||||
|
:load-only-current-source="$store.getters.optLightboxLoadOnlyCurrent"
|
||||||
|
:on-close="onClose"
|
||||||
|
:on-open="onShow"
|
||||||
|
:on-slide-change="onSlideChange"
|
||||||
|
></FsLightbox>
|
||||||
|
|
||||||
|
<a id="lightbox-download" style="display: none"></a>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import FsLightbox from "fslightbox-vue";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "Lightbox",
|
||||||
|
components: {FsLightbox},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
customButtons: [
|
||||||
|
{
|
||||||
|
viewBox: "0 0 384.928 384.928",
|
||||||
|
d: "M321.339,245.334c-4.74-4.692-12.439-4.704-17.179,0l-99.551,98.564V12.03 c0-6.641-5.438-12.03-12.151-12.03s-12.151,5.39-12.151,12.03v331.868l-99.551-98.552c-4.74-4.704-12.439-4.704-17.179,0 s-4.74,12.319,0,17.011l120.291,119.088c4.692,4.644,12.499,4.644,17.191,0l120.291-119.088 C326.091,257.653,326.091,250.038,321.339,245.334C316.599,240.642,326.091,250.038,321.339,245.334z",
|
||||||
|
width: "17px",
|
||||||
|
height: "17px",
|
||||||
|
title: "Download",
|
||||||
|
onClick: this.onDownloadClick
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
showLightbox() {
|
||||||
|
return this.$store.getters["uiShowLightbox"];
|
||||||
|
},
|
||||||
|
lightboxSources() {
|
||||||
|
return this.$store.getters["uiLightboxSources"];
|
||||||
|
},
|
||||||
|
lightboxThumbs() {
|
||||||
|
return this.$store.getters["uiLightboxThumbs"];
|
||||||
|
},
|
||||||
|
lightboxKey() {
|
||||||
|
return this.$store.getters["uiLightboxKey"];
|
||||||
|
},
|
||||||
|
lightboxSlide() {
|
||||||
|
return this.$store.getters["uiLightboxSlide"];
|
||||||
|
},
|
||||||
|
lightboxCaptions() {
|
||||||
|
return this.$store.getters["uiLightboxCaptions"];
|
||||||
|
},
|
||||||
|
lightboxTypes() {
|
||||||
|
return this.$store.getters["uiLightboxTypes"];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
onDownloadClick() {
|
||||||
|
const url = this.lightboxSources[this.lightboxSlide];
|
||||||
|
|
||||||
|
const a = document.getElementById("lightbox-download");
|
||||||
|
a.setAttribute("href", url);
|
||||||
|
a.setAttribute("download", "");
|
||||||
|
a.click();
|
||||||
|
},
|
||||||
|
onShow() {
|
||||||
|
this.$store.commit("setUiLightboxIsOpen", true);
|
||||||
|
},
|
||||||
|
onClose() {
|
||||||
|
this.$store.commit("setUiLightboxIsOpen", false);
|
||||||
|
},
|
||||||
|
onSlideChange() {
|
||||||
|
// Pause all videos when changing slide
|
||||||
|
document.getElementsByTagName("video").forEach((el) => {
|
||||||
|
el.pause();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.fslightbox-toolbar-button:nth-child(2) {
|
||||||
|
order: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fslightbox-toolbar-button:nth-child(1) {
|
||||||
|
order: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fslightbox-toolbar-button:nth-child(3) {
|
||||||
|
order: 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fslightbox-toolbar-button:nth-child(4) {
|
||||||
|
order: 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fslightbox-toolbar-button:nth-child(5) {
|
||||||
|
order: 5;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 650px) {
|
||||||
|
/* Disable fullscreen on mobile because it's buggy */
|
||||||
|
.fslightbox-toolbar-button:nth-child(6) {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.fslightbox-toolbar-button:nth-child(6) {
|
||||||
|
order: 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fslightbox-toolbar-button:nth-child(7) {
|
||||||
|
order: 7;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
26
sist2-vue/src/components/LightboxCaption.vue
Normal file
26
sist2-vue/src/components/LightboxCaption.vue
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<template>
|
||||||
|
<div class="lightbox-caption">
|
||||||
|
<p>
|
||||||
|
<b>{{
|
||||||
|
`[${$store.getters.indices.find(i => i.id === hit._source.index).name}]`
|
||||||
|
}}</b>{{ `/${hit._source.path}/${hit._source.name}${ext(hit)}` }}
|
||||||
|
</p>
|
||||||
|
<p style="margin-top: -1em">
|
||||||
|
<span v-if="hit._source.width">{{ `${hit._source.width}x${hit._source.height}`}}</span>
|
||||||
|
{{ ` (${humanFileSize(hit._source.size)})` }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {ext, humanFileSize} from "@/util";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "LightboxCaption",
|
||||||
|
props: ["hit"],
|
||||||
|
methods: {
|
||||||
|
humanFileSize: humanFileSize,
|
||||||
|
ext: ext
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
61
sist2-vue/src/components/MimePicker.vue
Normal file
61
sist2-vue/src/components/MimePicker.vue
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<template>
|
||||||
|
<div id="mimeTree"></div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import InspireTree from "inspire-tree";
|
||||||
|
import InspireTreeDOM from "inspire-tree-dom";
|
||||||
|
|
||||||
|
import "inspire-tree-dom/dist/inspire-tree-light.min.css";
|
||||||
|
import {getSelectedTreeNodes} from "@/util";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "MimePicker",
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
mimeTree: null,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.$store.subscribe((mutation) => {
|
||||||
|
if (mutation.type === "setUiMimeMap") {
|
||||||
|
const mimeMap = mutation.payload.slice();
|
||||||
|
|
||||||
|
this.mimeTree = new InspireTree({
|
||||||
|
selection: {
|
||||||
|
mode: 'checkbox'
|
||||||
|
},
|
||||||
|
data: mimeMap
|
||||||
|
});
|
||||||
|
new InspireTreeDOM(this.mimeTree, {
|
||||||
|
target: '#mimeTree'
|
||||||
|
});
|
||||||
|
this.mimeTree.on("node.state.changed", this.handleTreeClick);
|
||||||
|
this.mimeTree.deselect();
|
||||||
|
|
||||||
|
if (this.$store.state._onLoadSelectedMimeTypes.length > 0) {
|
||||||
|
this.$store.state._onLoadSelectedMimeTypes.forEach(mime => {
|
||||||
|
this.mimeTree.node(mime).select();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
handleTreeClick(node, e) {
|
||||||
|
if (e === "indeterminate" || e === "collapsed" || e === 'rendered' || e === "focused") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$store.commit("setSelectedMimeTypes", getSelectedTreeNodes(this.mimeTree));
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
#mimeTree {
|
||||||
|
max-height: 350px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
101
sist2-vue/src/components/NavBar.vue
Normal file
101
sist2-vue/src/components/NavBar.vue
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
<template>
|
||||||
|
<b-navbar>
|
||||||
|
<b-navbar-brand v-if="$route.path !== '/'" to="/">
|
||||||
|
<Sist2Icon></Sist2Icon>
|
||||||
|
</b-navbar-brand>
|
||||||
|
<b-navbar-brand v-else href=".">
|
||||||
|
<Sist2Icon></Sist2Icon>
|
||||||
|
</b-navbar-brand>
|
||||||
|
|
||||||
|
<span class="badge badge-pill version" v-if="$store && $store.state.sist2Info">
|
||||||
|
v{{ sist2Version() }}<span v-if="isDebug()">-dbg</span>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span v-if="$store && $store.state.sist2Info" class="tagline" v-html="tagline()"></span>
|
||||||
|
|
||||||
|
<b-button class="ml-auto" to="stats" variant="link">{{ $t("stats") }}</b-button>
|
||||||
|
<b-button to="config" variant="link">{{ $t("config") }}</b-button>
|
||||||
|
</b-navbar>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import Sist2Icon from "@/components/Sist2Icon";
|
||||||
|
export default {
|
||||||
|
name: "NavBar",
|
||||||
|
components: {Sist2Icon},
|
||||||
|
methods: {
|
||||||
|
tagline() {
|
||||||
|
return this.$store.state.sist2Info.tagline;
|
||||||
|
},
|
||||||
|
sist2Version() {
|
||||||
|
return this.$store.state.sist2Info.version;
|
||||||
|
},
|
||||||
|
isDebug() {
|
||||||
|
return this.$store.state.sist2Info.debug;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.navbar {
|
||||||
|
box-shadow: 0 0.125rem 0.25rem rgb(0 0 0 / 8%) !important;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .navbar {
|
||||||
|
background: #546b7a30;
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-brand {
|
||||||
|
color: #222 !important;
|
||||||
|
font-size: 1.75rem;
|
||||||
|
padding: 0;
|
||||||
|
font-family: Hack;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-brand:hover {
|
||||||
|
color: #000 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.version {
|
||||||
|
color: #222 !important;
|
||||||
|
margin-left: -18px;
|
||||||
|
margin-top: -14px;
|
||||||
|
font-size: 11px;
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .version {
|
||||||
|
color: #f5f5f5 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .navbar-brand {
|
||||||
|
font-size: 1.75rem;
|
||||||
|
padding: 0;
|
||||||
|
color: #f5f5f5 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black a:hover, .theme-black .btn:hover {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .navbar span {
|
||||||
|
color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 650px) {
|
||||||
|
.tagline {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.version {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-light .btn-link{
|
||||||
|
color: #222;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
245
sist2-vue/src/components/PathTree.vue
Normal file
245
sist2-vue/src/components/PathTree.vue
Normal file
@@ -0,0 +1,245 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div class="input-group" style="margin-bottom: 0.5em; margin-top: 1em">
|
||||||
|
<div class="input-group-prepend">
|
||||||
|
|
||||||
|
<b-button variant="outline-secondary" @click="$refs['path-modal'].show()">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 512" width="20px">
|
||||||
|
<path
|
||||||
|
fill="currentColor"
|
||||||
|
d="M288 224h224a32 32 0 0 0 32-32V64a32 32 0 0 0-32-32H400L368 0h-80a32 32 0 0 0-32 32v64H64V8a8 8 0 0 0-8-8H40a8 8 0 0 0-8 8v392a16 16 0 0 0 16 16h208v64a32 32 0 0 0 32 32h224a32 32 0 0 0 32-32V352a32 32 0 0 0-32-32H400l-32-32h-80a32 32 0 0 0-32 32v64H64V128h192v64a32 32 0 0 0 32 32zm0 96h66.74l32 32H512v128H288zm0-288h66.74l32 32H512v128H288z"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
</b-button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<VueSimpleSuggest
|
||||||
|
class="form-control-fix-flex"
|
||||||
|
@input="setPathText"
|
||||||
|
:value="getPathText"
|
||||||
|
:list="suggestPath"
|
||||||
|
:max-suggestions="0"
|
||||||
|
:placeholder="$t('pathBar.placeholder')"
|
||||||
|
>
|
||||||
|
<!-- Suggestion item template-->
|
||||||
|
<div slot="suggestion-item" slot-scope="{ suggestion, query }">
|
||||||
|
<div class="suggestion-line" :title="suggestion">
|
||||||
|
<strong>{{ query }}</strong>{{ getSuggestionWithoutQueryPrefix(suggestion, query) }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</VueSimpleSuggest>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<b-modal ref="path-modal" :title="$t('pathBar.modalTitle')" size="lg" :hide-footer="true" static>
|
||||||
|
<div id="pathTree"></div>
|
||||||
|
</b-modal>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import InspireTree from "inspire-tree";
|
||||||
|
import InspireTreeDOM from "inspire-tree-dom";
|
||||||
|
|
||||||
|
import "inspire-tree-dom/dist/inspire-tree-light.min.css";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
import {mapGetters, mapMutations} from "vuex";
|
||||||
|
import VueSimpleSuggest from 'vue-simple-suggest'
|
||||||
|
import 'vue-simple-suggest/dist/styles.css' // Optional CSS
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "PathTree",
|
||||||
|
components: {
|
||||||
|
VueSimpleSuggest
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
mimeTree: null,
|
||||||
|
pathItems: [],
|
||||||
|
tmpPath: ""
|
||||||
|
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
...mapGetters(["getPathText"])
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.$store.subscribe((mutation) => {
|
||||||
|
// Wait until indices are loaded to get the root paths
|
||||||
|
if (mutation.type === "setIndices") {
|
||||||
|
let pathTree = new InspireTree({
|
||||||
|
data: (node, resolve, reject) => {
|
||||||
|
return this.getNextDepth(node);
|
||||||
|
},
|
||||||
|
sort: "text"
|
||||||
|
});
|
||||||
|
|
||||||
|
this.$store.state.indices.forEach(idx => {
|
||||||
|
pathTree.addNode({
|
||||||
|
id: "/" + idx.id,
|
||||||
|
values: ["/" + idx.id],
|
||||||
|
text: `/[${idx.name}]`,
|
||||||
|
index: idx.id,
|
||||||
|
depth: 0,
|
||||||
|
children: true
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
new InspireTreeDOM(pathTree, {
|
||||||
|
target: "#pathTree"
|
||||||
|
});
|
||||||
|
|
||||||
|
pathTree.on("node.click", this.handleTreeClick);
|
||||||
|
pathTree.expand();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
...mapMutations(["setPathText"]),
|
||||||
|
getSuggestionWithoutQueryPrefix(suggestion, query) {
|
||||||
|
return suggestion.slice(query.length)
|
||||||
|
},
|
||||||
|
async getPathChoices() {
|
||||||
|
return new Promise(getPaths => {
|
||||||
|
const q = {
|
||||||
|
suggest: {
|
||||||
|
path: {
|
||||||
|
prefix: this.getPathText,
|
||||||
|
completion: {
|
||||||
|
field: "suggest-path",
|
||||||
|
skip_duplicates: true,
|
||||||
|
size: 10000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Sist2Api.esQuery(q)
|
||||||
|
.then(resp => getPaths(resp["suggest"]["path"][0]["options"].map(opt => opt["_source"]["path"])));
|
||||||
|
})
|
||||||
|
},
|
||||||
|
async suggestPath(term) {
|
||||||
|
if (!this.$store.state.optSuggestPath) {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
|
||||||
|
term = term.toLowerCase();
|
||||||
|
|
||||||
|
const choices = await this.getPathChoices();
|
||||||
|
|
||||||
|
let matches = [];
|
||||||
|
for (let i = 0; i < choices.length; i++) {
|
||||||
|
if (~choices[i].toLowerCase().indexOf(term)) {
|
||||||
|
matches.push(choices[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return matches.sort((a, b) => a.length - b.length);
|
||||||
|
},
|
||||||
|
getNextDepth(node) {
|
||||||
|
const q = {
|
||||||
|
query: {
|
||||||
|
bool: {
|
||||||
|
filter: [
|
||||||
|
{term: {index: node.index}},
|
||||||
|
{range: {_depth: {gte: node.depth + 1, lte: node.depth + 3}}},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
aggs: {
|
||||||
|
paths: {
|
||||||
|
terms: {
|
||||||
|
field: "path",
|
||||||
|
size: 10000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
size: 0
|
||||||
|
};
|
||||||
|
|
||||||
|
if (node.depth > 0) {
|
||||||
|
q.query.bool.must = {
|
||||||
|
prefix: {
|
||||||
|
path: node.id,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return Sist2Api.esQuery(q).then(resp => {
|
||||||
|
const buckets = resp["aggregations"]["paths"]["buckets"];
|
||||||
|
if (!buckets) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const paths = [];
|
||||||
|
|
||||||
|
return buckets
|
||||||
|
.filter(bucket => bucket.key.length > node.id.length || node.id.startsWith("/"))
|
||||||
|
.sort((a, b) => a.key > b.key)
|
||||||
|
.map(bucket => {
|
||||||
|
|
||||||
|
if (paths.some(n => bucket.key.startsWith(n))) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const name = node.id.startsWith("/") ? bucket.key : bucket.key.slice(node.id.length + 1);
|
||||||
|
|
||||||
|
paths.push(bucket.key);
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: bucket.key,
|
||||||
|
text: `${name}/ (${bucket.doc_count})`,
|
||||||
|
depth: node.depth + 1,
|
||||||
|
index: node.index,
|
||||||
|
values: [bucket.key],
|
||||||
|
children: true,
|
||||||
|
}
|
||||||
|
}).filter(x => x !== null)
|
||||||
|
});
|
||||||
|
},
|
||||||
|
handleTreeClick(e, node, handler) {
|
||||||
|
if (node.depth !== 0) {
|
||||||
|
this.setPathText(node.id);
|
||||||
|
this.$refs['path-modal'].hide()
|
||||||
|
|
||||||
|
this.$emit("search");
|
||||||
|
}
|
||||||
|
|
||||||
|
handler();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
#mimeTree {
|
||||||
|
max-height: 350px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-control-fix-flex {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
width: 1%;
|
||||||
|
min-width: 0;
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestion-line {
|
||||||
|
max-width: 100%;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
line-height: 1.1;
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.suggestions {
|
||||||
|
max-height: 250px;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .suggestions {
|
||||||
|
color: black
|
||||||
|
}
|
||||||
|
</style>
|
||||||
3
sist2-vue/src/components/Preloader.vue
Normal file
3
sist2-vue/src/components/Preloader.vue
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<template>
|
||||||
|
<b-progress value="1" max="1" animated></b-progress>
|
||||||
|
</template>
|
||||||
76
sist2-vue/src/components/ResultsCard.vue
Normal file
76
sist2-vue/src/components/ResultsCard.vue
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
<template>
|
||||||
|
<b-card v-if="lastResultsLoaded" id="results">
|
||||||
|
<span>{{ hitCount }} {{ hitCount === 1 ? $t("hit") : $t("hits") }}</span>
|
||||||
|
|
||||||
|
<div style="float: right">
|
||||||
|
<b-button v-b-toggle.collapse-1 variant="primary" class="not-mobile">{{ $t("details") }}</b-button>
|
||||||
|
|
||||||
|
<SortSelect class="ml-2"></SortSelect>
|
||||||
|
|
||||||
|
<DisplayModeToggle class="ml-2"></DisplayModeToggle>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<b-collapse id="collapse-1" class="pt-2" style="clear:both;">
|
||||||
|
<b-card>
|
||||||
|
<b-table :items="tableItems" small borderless thead-class="hidden" class="mb-0"></b-table>
|
||||||
|
</b-card>
|
||||||
|
</b-collapse>
|
||||||
|
</b-card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import {EsResult} from "@/Sist2Api";
|
||||||
|
import Vue from "vue";
|
||||||
|
import {humanFileSize, humanTime} from "@/util";
|
||||||
|
import DisplayModeToggle from "@/components/DisplayModeToggle.vue";
|
||||||
|
import SortSelect from "@/components/SortSelect.vue";
|
||||||
|
|
||||||
|
export default Vue.extend({
|
||||||
|
name: "ResultsCard",
|
||||||
|
components: {SortSelect, DisplayModeToggle},
|
||||||
|
computed: {
|
||||||
|
lastResultsLoaded() {
|
||||||
|
return this.$store.state.lastQueryResults != null;
|
||||||
|
},
|
||||||
|
hitCount() {
|
||||||
|
return (this.$store.state.lastQueryResults as EsResult).aggregations.total_count.value;
|
||||||
|
},
|
||||||
|
tableItems() {
|
||||||
|
const items = [];
|
||||||
|
|
||||||
|
|
||||||
|
items.push({key: this.$t("queryTime"), value: this.took()});
|
||||||
|
items.push({key: this.$t("totalSize"), value: this.totalSize()});
|
||||||
|
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
took() {
|
||||||
|
return (this.$store.state.lastQueryResults as EsResult).took + "ms";
|
||||||
|
},
|
||||||
|
totalSize() {
|
||||||
|
return humanFileSize((this.$store.state.lastQueryResults as EsResult).aggregations.total_size.value);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
#results {
|
||||||
|
margin-top: 1em;
|
||||||
|
|
||||||
|
box-shadow: 0 .125rem .25rem rgba(0, 0, 0, .08) !important;
|
||||||
|
border-radius: 0;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#results .card-body {
|
||||||
|
padding: 0.7em 1.25em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hidden {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
46
sist2-vue/src/components/SearchBar.vue
Normal file
46
sist2-vue/src/components/SearchBar.vue
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<b-input-group>
|
||||||
|
<b-form-input :value="searchText"
|
||||||
|
:placeholder="advanced() ? $t('searchBar.advanced') : $t('searchBar.simple')"
|
||||||
|
@input="setSearchText($event)"></b-form-input>
|
||||||
|
|
||||||
|
<template #prepend>
|
||||||
|
<b-input-group-text>
|
||||||
|
<b-form-checkbox :checked="fuzzy" title="Toggle fuzzy searching" @change="setFuzzy($event)">
|
||||||
|
{{ $t("searchBar.fuzzy") }}
|
||||||
|
</b-form-checkbox>
|
||||||
|
</b-input-group-text>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #append>
|
||||||
|
<b-button variant="outline-secondary" @click="$emit('show-help')">{{$t("help.help")}}</b-button>
|
||||||
|
</template>
|
||||||
|
</b-input-group>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {mapGetters, mapMutations} from "vuex";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
computed: {
|
||||||
|
...mapGetters({
|
||||||
|
optQueryMode: "optQueryMode",
|
||||||
|
searchText: "searchText",
|
||||||
|
fuzzy: "fuzzy",
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
...mapMutations({
|
||||||
|
setSearchText: "setSearchText",
|
||||||
|
setFuzzy: "setFuzzy"
|
||||||
|
}),
|
||||||
|
advanced() {
|
||||||
|
return this.optQueryMode === "advanced"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
</style>
|
||||||
40
sist2-vue/src/components/Sist2Icon.vue
Normal file
40
sist2-vue/src/components/Sist2Icon.vue
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<template>
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="27.868069mm"
|
||||||
|
height="7.6446671mm"
|
||||||
|
viewBox="0 0 27.868069 7.6446671"
|
||||||
|
>
|
||||||
|
<g transform="translate(-4.5018313,-4.1849793)">
|
||||||
|
<g
|
||||||
|
style="fill: currentColor;fill-opacity:1;stroke:none;stroke-width:0.26458332">
|
||||||
|
<path
|
||||||
|
d="m 6.3153296,11.829646 q -0.7717014,0 -1.8134983,-0.337619 v -0.916395 q 1.0128581,0.511252 1.803852,0.511252 0.5643067,0 0.901926,-0.236334 0.3376194,-0.236333 0.3376194,-0.63183 0,-0.3424428 -0.2845649,-0.5498376 Q 6.980922,9.4566645 6.3635609,9.3264399 L 5.9921796,9.2492698 Q 5.2301245,9.0949295 4.8732126,8.7428407 4.5211238,8.3859288 4.5211238,7.7733908 q 0,-0.7765245 0.5305447,-1.1961372 0.5305447,-0.4196126 1.5096409,-0.4196126 0.829579,0 1.6061036,0.3183268 V 7.3441319 Q 7.4101809,6.9004036 6.5854251,6.9004036 q -1.1671984,0 -1.1671984,0.7958171 0,0.2604492 0.1012858,0.4147895 0.1012858,0.1495171 0.3858507,0.2556261 0.2845649,0.1012858 0.8392253,0.2122179 l 0.3569119,0.067524 q 1.3408312,0.2652724 1.3408312,1.4614098 0,0.80064 -0.5691298,1.263661 -0.5691298,0.458197 -1.5578722,0.458197 z"
|
||||||
|
style="stroke-width:0.26458332"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="m 11.943927,5.3087694 q -0.144694,0 -0.144694,-0.144694 V 4.3296733 q 0,-0.144694 0.144694,-0.144694 h 0.694531 q 0.144694,0 0.144694,0.144694 v 0.8344021 q 0,0.144694 -0.144694,0.144694 z M 13.5645,11.728361 q -0.795817,0 -1.234722,-0.511253 -0.434082,-0.516075 -0.434082,-1.4469398 V 6.9823969 H 10.714028 V 6.2878656 h 2.069124 v 3.4823026 q 0,0.5884228 0.221864,0.8971028 0.221865,0.308681 0.6463,0.308681 h 1.036974 v 0.752409 z"
|
||||||
|
style="stroke-width:0.26458332"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="m 18.209178,11.829646 q -0.771701,0 -1.813498,-0.337619 v -0.916395 q 1.012858,0.511252 1.803852,0.511252 0.564306,0 0.901926,-0.236334 0.337619,-0.236333 0.337619,-0.63183 0,-0.3424428 -0.284565,-0.5498376 Q 18.87477,9.4566645 18.257409,9.3264399 l -0.371381,-0.07717 Q 17.123973,9.0949295 16.767061,8.7428407 16.414972,8.3859288 16.414972,7.7733908 q 0,-0.7765245 0.530545,-1.1961372 0.530545,-0.4196126 1.509641,-0.4196126 0.829579,0 1.606103,0.3183268 v 0.8681641 q -0.757232,-0.4437283 -1.581988,-0.4437283 -1.167198,0 -1.167198,0.7958171 0,0.2604492 0.101286,0.4147895 0.101286,0.1495171 0.385851,0.2556261 0.284565,0.1012858 0.839225,0.2122179 l 0.356912,0.067524 q 1.340831,0.2652724 1.340831,1.4614098 0,0.80064 -0.56913,1.263661 -0.56913,0.458197 -1.557872,0.458197 z"
|
||||||
|
style="stroke-width:0.26458332"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="m 25.207545,11.709068 q -0.993565,0 -1.408355,-0.40032 -0.409966,-0.405143 -0.409966,-1.3794164 V 6.9775737 H 21.947107 V 6.2878656 h 1.442117 V 4.8746874 l 0.887457,-0.3858507 v 1.7990289 h 2.016069 v 0.6897081 h -2.016069 v 2.9517579 q 0,0.5932454 0.226687,0.8344024 0.226687,0.236333 0.790994,0.236333 h 0.998388 v 0.709001 z"
|
||||||
|
style="stroke-width:0.26458332"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="m 27.995317,11.043476 q 0,-0.178456 0.120578,-0.299035 0.274919,-0.289388 0.651123,-0.684885 0.376205,-0.4003199 0.805464,-0.8681638 0.327973,-0.356912 0.491959,-0.5353679 0.16881,-0.1832791 0.255626,-0.2845649 0.09164,-0.1012858 0.178456,-0.2073948 0.255626,-0.3086805 0.405144,-0.5257215 0.15434,-0.2170411 0.250803,-0.4292589 0.168809,-0.3762045 0.168809,-0.7524089 0,-0.5980686 -0.352089,-0.935688 -0.356911,-0.3424425 -0.979096,-0.3424425 -0.863341,0 -1.938899,0.6414768 V 4.8361023 q 0.491959,-0.2363335 0.979096,-0.3569119 0.47749,-0.1205783 0.945334,-0.1205783 0.501606,0 0.940511,0.1350477 0.438905,0.1350478 0.766878,0.4244358 0.289388,0.2556261 0.463021,0.6270074 0.173633,0.3665582 0.173633,0.829579 0,0.4726671 -0.212218,0.9501574 -0.106109,0.2411567 -0.274919,0.4726671 -0.163986,0.2266873 -0.424435,0.540191 Q 31.270225,8.501684 31.077299,8.718725 30.884374,8.9357661 30.628748,9.2106847 30.445469,9.4084332 30.286305,9.5675966 30.131965,9.72676 29.958332,9.9003928 29.7847,10.069203 29.558012,10.300713 29.336148,10.5274 29.012998,10.869843 h 3.356901 v 0.819932 h -4.374582 z"
|
||||||
|
style="stroke-width:0.26458332"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "Sist2Icon"
|
||||||
|
}
|
||||||
|
</script>
|
||||||
135
sist2-vue/src/components/SizeSlider.vue
Normal file
135
sist2-vue/src/components/SizeSlider.vue
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
<template>
|
||||||
|
<div id="sizeSlider"></div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import noUiSlider from 'nouislider';
|
||||||
|
import 'nouislider/dist/nouislider.css';
|
||||||
|
import {humanFileSize} from "@/util";
|
||||||
|
import {mergeTooltips} from "@/util-js";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "SizeSlider",
|
||||||
|
mounted() {
|
||||||
|
const elem = document.getElementById("sizeSlider");
|
||||||
|
|
||||||
|
const slider = noUiSlider.create(elem, {
|
||||||
|
start: [
|
||||||
|
this.$store.state.sizeMin ? this.$store.state.sizeMin : 0,
|
||||||
|
this.$store.state.sizeMax ? this.$store.state.sizeMax: 1000 * 1000 * 50000
|
||||||
|
],
|
||||||
|
|
||||||
|
tooltips: [true, true],
|
||||||
|
behaviour: "drag-tap",
|
||||||
|
connect: true,
|
||||||
|
range: {
|
||||||
|
'min': 0,
|
||||||
|
"10%": 1000 * 1000,
|
||||||
|
"20%": 1000 * 1000 * 10,
|
||||||
|
"50%": 1000 * 1000 * 5000,
|
||||||
|
"max": 1000 * 1000 * 50000,
|
||||||
|
},
|
||||||
|
format: {
|
||||||
|
to: x => x >= 1000 * 1000 * 50000 ? "50G+" : humanFileSize(Math.round(x)),
|
||||||
|
from: x => x
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
mergeTooltips(elem, 10, " - ")
|
||||||
|
|
||||||
|
elem.querySelectorAll('.noUi-connect')[0].classList.add("slider-color0")
|
||||||
|
|
||||||
|
slider.on("set", (values, handle, unencoded) => {
|
||||||
|
|
||||||
|
if (handle === 0) {
|
||||||
|
this.$store.commit("setSizeMin", unencoded[0] === 0 ? undefined : Math.round(unencoded[0]))
|
||||||
|
} else {
|
||||||
|
this.$store.commit("setSizeMax", unencoded[1] >= 1000 * 1000 * 50000 ? undefined : Math.round(unencoded[1]))
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
#sizeSlider {
|
||||||
|
margin-top: 34px;
|
||||||
|
height: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.slider-color0 {
|
||||||
|
background: #2196f3;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .slider-color0 {
|
||||||
|
background: #00bcd4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-horizontal .noUi-handle {
|
||||||
|
width: 2px;
|
||||||
|
height: 18px;
|
||||||
|
right: -1px;
|
||||||
|
top: -3px;
|
||||||
|
border: none;
|
||||||
|
background-color: #1976d2;
|
||||||
|
box-shadow: none;
|
||||||
|
cursor: ew-resize;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .noUi-horizontal .noUi-handle {
|
||||||
|
background-color: #2168ac;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-handle:before {
|
||||||
|
top: -6px;
|
||||||
|
left: 3px;
|
||||||
|
|
||||||
|
width: 10px;
|
||||||
|
height: 28px;
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-handle:after {
|
||||||
|
top: -6px;
|
||||||
|
left: -10px;
|
||||||
|
|
||||||
|
width: 10px;
|
||||||
|
height: 28px;
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-draggable {
|
||||||
|
cursor: move;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-tooltip {
|
||||||
|
color: #fff;
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 1.333;
|
||||||
|
text-shadow: none;
|
||||||
|
padding: 0 2px;
|
||||||
|
background: #2196F3;
|
||||||
|
border-radius: 4px;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .noUi-tooltip {
|
||||||
|
background: #00bcd4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.theme-black .noUi-connects {
|
||||||
|
background-color: #37474f;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-horizontal .noUi-origin > .noUi-tooltip {
|
||||||
|
bottom: 7px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noUi-target {
|
||||||
|
background-color: #e1e4e9;
|
||||||
|
border: none;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
21
sist2-vue/src/components/SmallBadge.vue
Normal file
21
sist2-vue/src/components/SmallBadge.vue
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<template>
|
||||||
|
<b-badge variant="secondary" :pill="pill">{{ text }}</b-badge>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import Vue from "vue";
|
||||||
|
|
||||||
|
export default Vue.extend({
|
||||||
|
props: {
|
||||||
|
text: String,
|
||||||
|
pill: Boolean
|
||||||
|
}
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.badge-pill {
|
||||||
|
padding: 0.3em .4em 0.1em;
|
||||||
|
border-radius: 6rem;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
59
sist2-vue/src/components/SortSelect.vue
Normal file
59
sist2-vue/src/components/SortSelect.vue
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
<template>
|
||||||
|
<b-dropdown variant="primary">
|
||||||
|
<b-dropdown-item :class="{'dropdown-active': sort === 'score'}" @click="onSelect('score')">{{
|
||||||
|
$t("sort.relevance")
|
||||||
|
}}
|
||||||
|
</b-dropdown-item>
|
||||||
|
<b-dropdown-item :class="{'dropdown-active': sort === 'dateAsc'}" @click="onSelect('dateAsc')">{{
|
||||||
|
$t("sort.dateAsc")
|
||||||
|
}}
|
||||||
|
</b-dropdown-item>
|
||||||
|
<b-dropdown-item :class="{'dropdown-active': sort === 'dateDesc'}" @click="onSelect('dateDesc')">
|
||||||
|
{{ $t("sort.dateDesc") }}
|
||||||
|
</b-dropdown-item>
|
||||||
|
<b-dropdown-item :class="{'dropdown-active': sort === 'sizeAsc'}" @click="onSelect('sizeAsc')">{{
|
||||||
|
$t("sort.sizeAsc")
|
||||||
|
}}
|
||||||
|
</b-dropdown-item>
|
||||||
|
<b-dropdown-item :class="{'dropdown-active': sort === 'sizeDesc'}" @click="onSelect('sizeDesc')">
|
||||||
|
{{ $t("sort.sizeDesc") }}
|
||||||
|
</b-dropdown-item>
|
||||||
|
|
||||||
|
<b-dropdown-item :class="{'dropdown-active': sort === 'random'}" @click="onSelect('random')">
|
||||||
|
{{ $t("sort.random") }}
|
||||||
|
</b-dropdown-item>
|
||||||
|
|
||||||
|
<template #button-content>
|
||||||
|
<svg aria-hidden="true" width="20px" height="20px" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 320 512">
|
||||||
|
<path
|
||||||
|
fill="currentColor"
|
||||||
|
d="M41 288h238c21.4 0 32.1 25.9 17 41L177 448c-9.4 9.4-24.6 9.4-33.9 0L24 329c-15.1-15.1-4.4-41 17-41zm255-105L177 64c-9.4-9.4-24.6-9.4-33.9 0L24 183c-15.1 15.1-4.4 41 17 41h238c21.4 0 32.1-25.9 17-41z"></path>
|
||||||
|
</svg>
|
||||||
|
</template>
|
||||||
|
</b-dropdown>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: "SortSelect",
|
||||||
|
computed: {
|
||||||
|
sort() {
|
||||||
|
return this.$store.state.sortMode;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
onSelect(sortMode) {
|
||||||
|
if (sortMode === "random") {
|
||||||
|
this.$store.commit("setSeed", Math.round(Math.random() * 100000));
|
||||||
|
}
|
||||||
|
this.$store.commit("setSortMode", sortMode);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.dropdown-active a {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
337
sist2-vue/src/components/TagContainer.vue
Normal file
337
sist2-vue/src/components/TagContainer.vue
Normal file
@@ -0,0 +1,337 @@
|
|||||||
|
<template>
|
||||||
|
<div @mouseenter="showAddButton = true" @mouseleave="showAddButton = false">
|
||||||
|
|
||||||
|
<b-modal v-model="showModal" :title="$t('saveTagModalTitle')" hide-footer no-fade centered size="lg" static lazy>
|
||||||
|
<b-row>
|
||||||
|
<b-col style="flex-grow: 2" sm>
|
||||||
|
<VueSimpleSuggest
|
||||||
|
ref="suggest"
|
||||||
|
:value="tagText"
|
||||||
|
@select="setTagText($event)"
|
||||||
|
@input="setTagText($event)"
|
||||||
|
class="form-control-fix-flex"
|
||||||
|
style="margin-top: 17px"
|
||||||
|
:list="suggestTag"
|
||||||
|
:max-suggestions="0"
|
||||||
|
:placeholder="$t('saveTagPlaceholder')"
|
||||||
|
>
|
||||||
|
<!-- Suggestion item template-->
|
||||||
|
<div slot="suggestion-item" slot-scope="{ suggestion, query}"
|
||||||
|
>
|
||||||
|
<div class="suggestion-line">
|
||||||
|
<span
|
||||||
|
class="badge badge-suggestion"
|
||||||
|
:style="{background: getBg(suggestion), color: getFg(suggestion)}"
|
||||||
|
>
|
||||||
|
<strong>{{ query }}</strong>{{ getSuggestionWithoutQueryPrefix(suggestion, query) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</VueSimpleSuggest>
|
||||||
|
</b-col>
|
||||||
|
<b-col class="mt-4">
|
||||||
|
<TwitterColorPicker v-model="color" triangle="hide" :width="252" class="mr-auto ml-auto"></TwitterColorPicker>
|
||||||
|
</b-col>
|
||||||
|
</b-row>
|
||||||
|
|
||||||
|
<b-button variant="primary" style="float: right" class="mt-2" @click="saveTag()">{{ $t("confirm") }}
|
||||||
|
</b-button>
|
||||||
|
</b-modal>
|
||||||
|
|
||||||
|
|
||||||
|
<template v-for="tag in hit._tags">
|
||||||
|
<div v-if="tag.userTag" :key="tag.rawText" style="display: inline-block">
|
||||||
|
<span
|
||||||
|
:id="hit._id+tag.rawText"
|
||||||
|
:title="tag.text"
|
||||||
|
tabindex="-1"
|
||||||
|
class="badge pointer"
|
||||||
|
:style="badgeStyle(tag)" :class="badgeClass(tag)"
|
||||||
|
@click.right="onTagRightClick(tag, $event)"
|
||||||
|
>{{ tag.text.split(".").pop() }}</span>
|
||||||
|
|
||||||
|
<b-popover :target="hit._id+tag.rawText" triggers="focus blur" placement="top">
|
||||||
|
<b-button variant="danger" @click="onTagDeleteClick(tag, $event)">Delete</b-button>
|
||||||
|
</b-popover>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span
|
||||||
|
v-else :key="tag.text"
|
||||||
|
class="badge"
|
||||||
|
:style="badgeStyle(tag)" :class="badgeClass(tag)"
|
||||||
|
>{{ tag.text.split(".").pop() }}</span>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<!-- Add button -->
|
||||||
|
<small v-if="showAddButton" class="badge add-tag-button" @click="tagAdd()">Add</small>
|
||||||
|
|
||||||
|
<!-- Size tag-->
|
||||||
|
<small v-else class="text-muted badge-size">{{
|
||||||
|
humanFileSize(hit._source.size)
|
||||||
|
}}</small>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {humanFileSize, lum} from "@/util";
|
||||||
|
import Vue from "vue";
|
||||||
|
import {Twitter} from 'vue-color'
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
import VueSimpleSuggest from 'vue-simple-suggest'
|
||||||
|
|
||||||
|
export default Vue.extend({
|
||||||
|
components: {
|
||||||
|
"TwitterColorPicker": Twitter,
|
||||||
|
VueSimpleSuggest
|
||||||
|
},
|
||||||
|
props: ["hit"],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
showAddButton: false,
|
||||||
|
showModal: false,
|
||||||
|
tagText: null,
|
||||||
|
color: {
|
||||||
|
hex: "#e0e0e0",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
tagHover() {
|
||||||
|
return this.$store.getters["uiTagHover"];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
humanFileSize: humanFileSize,
|
||||||
|
getSuggestionWithoutQueryPrefix(suggestion, query) {
|
||||||
|
return suggestion.id.slice(query.length, -8)
|
||||||
|
},
|
||||||
|
getBg(suggestion) {
|
||||||
|
return suggestion.id.slice(-7);
|
||||||
|
},
|
||||||
|
getFg(suggestion) {
|
||||||
|
return lum(suggestion.id.slice(-7)) > 50 ? "#000" : "#fff";
|
||||||
|
},
|
||||||
|
setTagText(value) {
|
||||||
|
this.$refs.suggest.clearSuggestions();
|
||||||
|
|
||||||
|
if (typeof value === "string") {
|
||||||
|
this.tagText = {
|
||||||
|
id: value,
|
||||||
|
title: value
|
||||||
|
};
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.color = {
|
||||||
|
hex: "#" + value.id.split("#")[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
this.tagText = value;
|
||||||
|
},
|
||||||
|
badgeClass(tag) {
|
||||||
|
return `badge-${tag.style}`;
|
||||||
|
},
|
||||||
|
badgeStyle(tag) {
|
||||||
|
return {
|
||||||
|
background: tag.bg,
|
||||||
|
color: tag.fg,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
onTagHover(tag) {
|
||||||
|
if (tag.userTag) {
|
||||||
|
this.$store.commit("setUiTagHover", tag);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onTagLeave() {
|
||||||
|
this.$store.commit("setUiTagHover", null);
|
||||||
|
},
|
||||||
|
onTagDeleteClick(tag, e) {
|
||||||
|
this.hit._tags = this.hit._tags.filter(t => t !== tag);
|
||||||
|
|
||||||
|
Sist2Api.deleteTag(tag.rawText, this.hit).then(() => {
|
||||||
|
//toast
|
||||||
|
this.$store.commit("busUpdateWallItems");
|
||||||
|
this.$store.commit("busUpdateTags");
|
||||||
|
});
|
||||||
|
},
|
||||||
|
tagAdd() {
|
||||||
|
this.showModal = true;
|
||||||
|
},
|
||||||
|
saveTag() {
|
||||||
|
if (this.tagText.id.includes("#")) {
|
||||||
|
this.$bvToast.toast(
|
||||||
|
this.$t("toast.invalidTag"),
|
||||||
|
{
|
||||||
|
title: this.$t("toast.invalidTagTitle"),
|
||||||
|
noAutoHide: true,
|
||||||
|
toaster: "b-toaster-bottom-right",
|
||||||
|
headerClass: "toast-header-error",
|
||||||
|
bodyClass: "toast-body-error",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let tag = this.tagText.id + this.color.hex.replace("#", ".#");
|
||||||
|
const userTags = this.hit._tags.filter(t => t.userTag);
|
||||||
|
|
||||||
|
if (userTags.find(t => t.rawText === tag) != null) {
|
||||||
|
this.$bvToast.toast(
|
||||||
|
this.$t("toast.dupeTag"),
|
||||||
|
{
|
||||||
|
title: this.$t("toast.dupeTagTitle"),
|
||||||
|
noAutoHide: true,
|
||||||
|
toaster: "b-toaster-bottom-right",
|
||||||
|
headerClass: "toast-header-error",
|
||||||
|
bodyClass: "toast-body-error",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.hit._tags.push(Sist2Api.createUserTag(tag));
|
||||||
|
|
||||||
|
Sist2Api.saveTag(tag, this.hit).then(() => {
|
||||||
|
this.tagText = null;
|
||||||
|
this.showModal = false;
|
||||||
|
this.$store.commit("busUpdateWallItems");
|
||||||
|
this.$store.commit("busUpdateTags");
|
||||||
|
// TODO: toast
|
||||||
|
});
|
||||||
|
},
|
||||||
|
async suggestTag(term) {
|
||||||
|
term = term.toLowerCase();
|
||||||
|
|
||||||
|
const choices = await this.getTagChoices(term);
|
||||||
|
|
||||||
|
let matches = [];
|
||||||
|
for (let i = 0; i < choices.length; i++) {
|
||||||
|
if (~choices[i].toLowerCase().indexOf(term)) {
|
||||||
|
matches.push(choices[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return matches.sort().map(match => {
|
||||||
|
return {
|
||||||
|
title: match.split(".").slice(0,-1).join("."),
|
||||||
|
id: match
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
getTagChoices(prefix) {
|
||||||
|
return new Promise(getPaths => {
|
||||||
|
Sist2Api.esQuery({
|
||||||
|
suggest: {
|
||||||
|
tag: {
|
||||||
|
prefix: prefix,
|
||||||
|
completion: {
|
||||||
|
field: "suggest-tag",
|
||||||
|
skip_duplicates: true,
|
||||||
|
size: 10000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}).then(resp => {
|
||||||
|
const result = [];
|
||||||
|
resp["suggest"]["tag"][0]["options"].map(opt => opt["_source"]["tag"]).forEach(tags => {
|
||||||
|
tags.forEach(tag => {
|
||||||
|
const t = tag.slice(0, -8);
|
||||||
|
if (!result.find(x => x.slice(0, -8) === t)) {
|
||||||
|
result.push(tag);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
getPaths(result);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
|
||||||
|
|
||||||
|
.badge-video {
|
||||||
|
color: #FFFFFF;
|
||||||
|
background-color: #F27761;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-image {
|
||||||
|
color: #FFFFFF;
|
||||||
|
background-color: #AA99C9;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-audio {
|
||||||
|
color: #FFFFFF;
|
||||||
|
background-color: #00ADEF;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-user {
|
||||||
|
color: #212529;
|
||||||
|
background-color: #e0e0e0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-user:hover, .add-tag-button:hover {
|
||||||
|
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.12), 0 1px 2px rgba(0, 0, 0, 0.24);
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-text {
|
||||||
|
color: #FFFFFF;
|
||||||
|
background-color: #FAAB3C;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge {
|
||||||
|
margin-right: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-delete {
|
||||||
|
margin-right: -2px;
|
||||||
|
margin-left: 2px;
|
||||||
|
margin-top: -1px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 90%;
|
||||||
|
background: rgba(0, 0, 0, 0.2);
|
||||||
|
padding: 0.1em 0.4em;
|
||||||
|
color: white;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-size {
|
||||||
|
width: 50px;
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-tag-button {
|
||||||
|
cursor: pointer;
|
||||||
|
color: #212529;
|
||||||
|
background-color: #e0e0e0;
|
||||||
|
width: 50px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge {
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge-suggestion {
|
||||||
|
font-size: 90%;
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.vc-twitter-body {
|
||||||
|
padding: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-twitter {
|
||||||
|
box-shadow: none !important;
|
||||||
|
background: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip {
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toast {
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
185
sist2-vue/src/components/TagPicker.vue
Normal file
185
sist2-vue/src/components/TagPicker.vue
Normal file
@@ -0,0 +1,185 @@
|
|||||||
|
<template>
|
||||||
|
<div id="tagTree"></div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import InspireTree from "inspire-tree";
|
||||||
|
import InspireTreeDOM from "inspire-tree-dom";
|
||||||
|
|
||||||
|
import "inspire-tree-dom/dist/inspire-tree-light.min.css";
|
||||||
|
import {getSelectedTreeNodes} from "@/util";
|
||||||
|
import Sist2Api from "@/Sist2Api";
|
||||||
|
|
||||||
|
function resetState(node) {
|
||||||
|
node._tree.defaultState.forEach(function (val, prop) {
|
||||||
|
node.state(prop, val);
|
||||||
|
});
|
||||||
|
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
function baseStateChange(prop, value, verb, node, deep) {
|
||||||
|
if (node.state(prop) !== value) {
|
||||||
|
node._tree.batch();
|
||||||
|
|
||||||
|
if (node._tree.config.nodes.resetStateOnRestore && verb === 'restored') {
|
||||||
|
resetState(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
node.state(prop, value);
|
||||||
|
|
||||||
|
node._tree.emit('node.' + verb, node, false);
|
||||||
|
|
||||||
|
if (deep && node.hasChildren()) {
|
||||||
|
node.children.recurseDown(function (child) {
|
||||||
|
baseStateChange(prop, value, verb, child);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
node.markDirty();
|
||||||
|
node._tree.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
function addTag(map, tag, id, count) {
|
||||||
|
const tags = tag.split(".");
|
||||||
|
|
||||||
|
const child = {
|
||||||
|
id: id,
|
||||||
|
count: count,
|
||||||
|
text: tags.length !== 1 ? tags[0] : `${tags[0]} (${count})`,
|
||||||
|
name: tags[0],
|
||||||
|
children: [],
|
||||||
|
// Overwrite base functions
|
||||||
|
blur: function () {
|
||||||
|
// noop
|
||||||
|
},
|
||||||
|
select: function () {
|
||||||
|
this.state("selected", true);
|
||||||
|
return this.check()
|
||||||
|
},
|
||||||
|
deselect: function () {
|
||||||
|
this.state("selected", false);
|
||||||
|
return this.uncheck()
|
||||||
|
},
|
||||||
|
uncheck: function () {
|
||||||
|
baseStateChange('checked', false, 'unchecked', this, false);
|
||||||
|
this.state('indeterminate', false);
|
||||||
|
|
||||||
|
if (this.hasParent()) {
|
||||||
|
this.getParent().refreshIndeterminateState();
|
||||||
|
}
|
||||||
|
|
||||||
|
this._tree.end();
|
||||||
|
return this;
|
||||||
|
},
|
||||||
|
check: function () {
|
||||||
|
baseStateChange('checked', true, 'checked', this, false);
|
||||||
|
|
||||||
|
if (this.hasParent()) {
|
||||||
|
this.getParent().refreshIndeterminateState();
|
||||||
|
}
|
||||||
|
|
||||||
|
this._tree.end();
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let found = false;
|
||||||
|
map.forEach(node => {
|
||||||
|
if (node.name === child.name) {
|
||||||
|
found = true;
|
||||||
|
if (tags.length !== 1) {
|
||||||
|
addTag(node.children, tags.slice(1).join("."), id, count);
|
||||||
|
} else {
|
||||||
|
// Same name, different color
|
||||||
|
console.error("FIXME: Duplicate tag?")
|
||||||
|
console.trace(node)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (!found) {
|
||||||
|
if (tags.length !== 1) {
|
||||||
|
addTag(child.children, tags.slice(1).join("."), id, count);
|
||||||
|
map.push(child);
|
||||||
|
} else {
|
||||||
|
map.push(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: "TagPicker",
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
tagTree: null,
|
||||||
|
loadedFromArgs: false,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.$store.subscribe((mutation) => {
|
||||||
|
if (mutation.type === "setUiMimeMap") {
|
||||||
|
this.initializeTree();
|
||||||
|
this.updateTree();
|
||||||
|
} else if (mutation.type === "busUpdateTags") {
|
||||||
|
window.setTimeout(this.updateTree, 2000);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
initializeTree() {
|
||||||
|
const tagMap = [];
|
||||||
|
this.tagTree = new InspireTree({
|
||||||
|
selection: {
|
||||||
|
mode: "checkbox",
|
||||||
|
autoDeselect: false,
|
||||||
|
},
|
||||||
|
checkbox: {
|
||||||
|
autoCheckChildren: false,
|
||||||
|
},
|
||||||
|
data: tagMap
|
||||||
|
});
|
||||||
|
new InspireTreeDOM(this.tagTree, {
|
||||||
|
target: '#tagTree'
|
||||||
|
});
|
||||||
|
this.tagTree.on("node.state.changed", this.handleTreeClick);
|
||||||
|
},
|
||||||
|
updateTree() {
|
||||||
|
const tagMap = [];
|
||||||
|
Sist2Api.getTags().then(tags => {
|
||||||
|
tags.forEach(tag => addTag(tagMap, tag.id, tag.id, tag.count));
|
||||||
|
this.tagTree.removeAll();
|
||||||
|
this.tagTree.addNodes(tagMap);
|
||||||
|
|
||||||
|
if (this.$store.state._onLoadSelectedTags.length > 0 && !this.loadedFromArgs) {
|
||||||
|
this.$store.state._onLoadSelectedTags.forEach(mime => {
|
||||||
|
this.tagTree.node(mime).select();
|
||||||
|
this.loadedFromArgs = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
handleTreeClick(node, e) {
|
||||||
|
if (e === "indeterminate" || e === "collapsed" || e === 'rendered' || e === "focused") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$store.commit("setSelectedTags", getSelectedTreeNodes(this.tagTree));
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
#mimeTree {
|
||||||
|
max-height: 350px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<style>
|
||||||
|
.inspire-tree .focused>.wholerow {
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
298
sist2-vue/src/i18n/messages.ts
Normal file
298
sist2-vue/src/i18n/messages.ts
Normal file
@@ -0,0 +1,298 @@
|
|||||||
|
export default {
|
||||||
|
en: {
|
||||||
|
searchBar: {
|
||||||
|
simple: "Search",
|
||||||
|
advanced: "Advanced search",
|
||||||
|
fuzzy: "Fuzzy"
|
||||||
|
},
|
||||||
|
download: "Download",
|
||||||
|
and: "and",
|
||||||
|
page: "page",
|
||||||
|
pages: "pages",
|
||||||
|
mimeTypes: "Media types",
|
||||||
|
tags: "Tags",
|
||||||
|
help: {
|
||||||
|
simpleSearch: "Simple search",
|
||||||
|
advancedSearch: "Advanced search",
|
||||||
|
help: "Help",
|
||||||
|
term: "<TERM>",
|
||||||
|
and: "AND operator",
|
||||||
|
or: "OR operator",
|
||||||
|
not: "negates a single term",
|
||||||
|
quotes: "will match the enclosed sequence of terms in that specific order",
|
||||||
|
prefix: "will match any term with a given prefix when used at the end of a word",
|
||||||
|
parens: "used to group expressions",
|
||||||
|
tildeTerm: "match a term with a given edit distance",
|
||||||
|
tildePhrase: "match a phrase with a given number of allowed intervening unmatched words",
|
||||||
|
example1:
|
||||||
|
"For example: <code>\"fried eggs\" +(eggplant | potato) -frittata</code> will match the " +
|
||||||
|
"phrase <i>fried eggs</i> and either <i>eggplant</i> or <i>potato</i>, but will ignore results " +
|
||||||
|
"containing <i>frittata</i>.",
|
||||||
|
defaultOperator:
|
||||||
|
"When neither <code>+</code> or <code>|</code> is specified, the default operator is " +
|
||||||
|
"<code>+</code> (and).",
|
||||||
|
fuzzy:
|
||||||
|
"When the <b>Fuzzy</b> option is checked, partial matches based on 3-grams are also returned.",
|
||||||
|
moreInfoSimple: "For more information, see <a target=\"_blank\" " +
|
||||||
|
"rel=\"noreferrer\" href=\"//www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl-simple-query-string-query.html\">Elasticsearch documentation</a>",
|
||||||
|
moreInfoAdvanced: "For documentation about the advanced search mode, see <a target=\"_blank\" rel=\"noreferrer\" href=\"//www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl-query-string-query.html#query-string-syntax\">Elasticsearch documentation</a>"
|
||||||
|
},
|
||||||
|
config: "Configuration",
|
||||||
|
configDescription: "Configuration is saved in real time for this browser.",
|
||||||
|
configReset: "Reset configuration",
|
||||||
|
searchOptions: "Search options",
|
||||||
|
treemapOptions: "Treemap options",
|
||||||
|
displayOptions: "Display options",
|
||||||
|
opt: {
|
||||||
|
lang: "Language",
|
||||||
|
highlight: "Enable highlighting",
|
||||||
|
fuzzy: "Set fuzzy search by default",
|
||||||
|
searchInPath: "Enable matching query against document path",
|
||||||
|
suggestPath: "Enable auto-complete in path filter bar",
|
||||||
|
fragmentSize: "Highlight context size in characters",
|
||||||
|
queryMode: "Search mode",
|
||||||
|
displayMode: "Display",
|
||||||
|
columns: "Column count",
|
||||||
|
treemapType: "Treemap type",
|
||||||
|
treemapTiling: "Treemap tiling",
|
||||||
|
treemapColorGroupingDepth: "Treemap color grouping depth (flat)",
|
||||||
|
treemapColor: "Treemap color (cascaded)",
|
||||||
|
treemapSize: "Treemap size",
|
||||||
|
theme: "Theme",
|
||||||
|
lightboxLoadOnlyCurrent: "Do not preload full-size images for adjacent slides in image viewer.",
|
||||||
|
slideDuration: "Slide duration",
|
||||||
|
resultSize: "Number of results per page",
|
||||||
|
tagOrOperator: "Use OR operator when specifying multiple tags.",
|
||||||
|
hideDuplicates: "Hide duplicate results based on checksum"
|
||||||
|
},
|
||||||
|
queryMode: {
|
||||||
|
simple: "Simple",
|
||||||
|
advanced: "Advanced",
|
||||||
|
},
|
||||||
|
lang: {
|
||||||
|
en: "English",
|
||||||
|
fr: "Français"
|
||||||
|
},
|
||||||
|
displayMode: {
|
||||||
|
grid: "Grid",
|
||||||
|
list: "List",
|
||||||
|
},
|
||||||
|
columns: {
|
||||||
|
auto: "Auto"
|
||||||
|
},
|
||||||
|
treemapType: {
|
||||||
|
cascaded: "Cascaded",
|
||||||
|
flat: "Flat (compact)"
|
||||||
|
},
|
||||||
|
treemapSize: {
|
||||||
|
small: "Small",
|
||||||
|
medium: "Medium",
|
||||||
|
large: "Large",
|
||||||
|
xLarge: "xLarge",
|
||||||
|
xxLarge: "xxLarge",
|
||||||
|
custom: "Custom",
|
||||||
|
},
|
||||||
|
treemapTiling: {
|
||||||
|
binary: "Binary",
|
||||||
|
squarify: "Squarify",
|
||||||
|
slice: "Slice",
|
||||||
|
dice: "Dice",
|
||||||
|
sliceDice: "Slice & Dice",
|
||||||
|
},
|
||||||
|
theme: {
|
||||||
|
light: "Light",
|
||||||
|
black: "Black"
|
||||||
|
},
|
||||||
|
hit: "hit",
|
||||||
|
hits: "hits",
|
||||||
|
details: "Details",
|
||||||
|
stats: "Stats",
|
||||||
|
queryTime: "Query time",
|
||||||
|
totalSize: "Total size",
|
||||||
|
pathBar: {
|
||||||
|
placeholder: "Filter path",
|
||||||
|
modalTitle: "Select path"
|
||||||
|
},
|
||||||
|
debug: "Debug information",
|
||||||
|
debugDescription: "Information useful for debugging. If you encounter bugs or have suggestions for" +
|
||||||
|
" new features, please submit a new issue <a href='https://github.com/simon987/sist2/issues/new/choose'>here</a>.",
|
||||||
|
tagline: "Tagline",
|
||||||
|
toast: {
|
||||||
|
esConnErrTitle: "Elasticsearch connection error",
|
||||||
|
esConnErr: "sist2 web module encountered an error while connecting to Elasticsearch." +
|
||||||
|
" See server logs for more information.",
|
||||||
|
esQueryErrTitle: "Query error",
|
||||||
|
esQueryErr: "Could not parse or execute query, please check the Advanced search documentation. " +
|
||||||
|
"See server logs for more information.",
|
||||||
|
dupeTagTitle: "Duplicate tag",
|
||||||
|
dupeTag: "This tag already exists for this document."
|
||||||
|
},
|
||||||
|
saveTagModalTitle: "Add tag",
|
||||||
|
saveTagPlaceholder: "Tag name",
|
||||||
|
confirm: "Confirm",
|
||||||
|
indexPickerPlaceholder: "Select indices",
|
||||||
|
sort: {
|
||||||
|
relevance: "Relevance",
|
||||||
|
dateAsc: "Date (Older first)",
|
||||||
|
dateDesc: "Date (Newer first)",
|
||||||
|
sizeAsc: "Size (Smaller first)",
|
||||||
|
sizeDesc: "Size (Larger first)",
|
||||||
|
random: "Random",
|
||||||
|
},
|
||||||
|
d3: {
|
||||||
|
mimeCount: "File count distribution by media type",
|
||||||
|
mimeSize: "Size distribution by media type",
|
||||||
|
dateHistogram: "File modification time distribution",
|
||||||
|
sizeHistogram: "File size distribution",
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fr: {
|
||||||
|
searchBar: {
|
||||||
|
simple: "Recherche",
|
||||||
|
advanced: "Recherche avancée",
|
||||||
|
fuzzy: "Approximatif"
|
||||||
|
},
|
||||||
|
download: "Télécharger",
|
||||||
|
and: "et",
|
||||||
|
page: "page",
|
||||||
|
pages: "pages",
|
||||||
|
mimeTypes: "Types de médias",
|
||||||
|
tags: "Tags",
|
||||||
|
help: {
|
||||||
|
simpleSearch: "Recherche simple",
|
||||||
|
advancedSearch: "Recherche avancée",
|
||||||
|
help: "Aide",
|
||||||
|
term: "<TERME>",
|
||||||
|
and: "opérator ET",
|
||||||
|
or: "opérator OU",
|
||||||
|
not: "exclut un terme",
|
||||||
|
quotes: "recherche la séquence de termes dans cet ordre spécifique.",
|
||||||
|
prefix: "lorsqu'utilisé à la fin d'un mot, recherche tous les termes avec le préfixe donné.",
|
||||||
|
parens: "utilisé pour regrouper des expressions",
|
||||||
|
tildeTerm: "recherche un terme avec une distance d'édition donnée",
|
||||||
|
tildePhrase: "recherche une phrase avec un nombre donné de mots intermédiaires tolérés",
|
||||||
|
example1:
|
||||||
|
"Par exemple: <code>\"fried eggs\" +(eggplant | potato) -frittata</code> va rechercher la " +
|
||||||
|
"phrase <i>fried eggs</i> et soit <i>eggplant</i> ou <i>potato</i>, mais vas exlure les résultats " +
|
||||||
|
"qui contiennent <i>frittata</i>.",
|
||||||
|
defaultOperator:
|
||||||
|
"Lorsqu'aucun des opérateurs <code>+</code> ou <code>|</code> sont spécifiés, l'opérateur par défaut " +
|
||||||
|
"est <code>+</code> (ET).",
|
||||||
|
fuzzy:
|
||||||
|
"Lorsque l'option <b>Approximatif</b> est activée, les résultats partiels basés sur les trigrammes sont" +
|
||||||
|
" également inclus.",
|
||||||
|
moreInfoSimple: "Pour plus d'information, voir <a target=\"_blank\" " +
|
||||||
|
"rel=\"noreferrer\" href=\"//www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl-simple-query-string-query.html\">documentation Elasticsearch</a>",
|
||||||
|
moreInfoAdvanced: "Pour plus d'information sur la recherche avancée, voir <a target=\"_blank\" rel=\"noreferrer\" href=\"//www.elastic.co/guide/en/elasticsearch/reference/current/query-dsl-query-string-query.html#query-string-syntax\">documentation Elasticsearch</a>"
|
||||||
|
},
|
||||||
|
config: "Configuration",
|
||||||
|
configDescription: "La configuration est enregistrée en temps réel pour ce navigateur.",
|
||||||
|
configReset: "Réinitialiser la configuration",
|
||||||
|
searchOptions: "Options de recherche",
|
||||||
|
treemapOptions: "Options du Treemap",
|
||||||
|
displayOptions: "Options d'affichage",
|
||||||
|
opt: {
|
||||||
|
lang: "Langue",
|
||||||
|
highlight: "Activer le surlignage",
|
||||||
|
fuzzy: "Activer la recherche approximative par défaut",
|
||||||
|
searchInPath: "Activer la recherche dans le chemin des documents",
|
||||||
|
suggestPath: "Activer l'autocomplétion dans la barre de filtre de chemin",
|
||||||
|
fragmentSize: "Longueur du contexte de surlignage, en nombre de caractères",
|
||||||
|
queryMode: "Mode de recherche",
|
||||||
|
displayMode: "Affichage",
|
||||||
|
columns: "Nombre de colonnes",
|
||||||
|
treemapType: "Type de Treemap",
|
||||||
|
treemapTiling: "Treemap tiling",
|
||||||
|
treemapColorGroupingDepth: "Groupage de couleur du Treemap (plat)",
|
||||||
|
treemapColor: "Couleur du Treemap (en cascade)",
|
||||||
|
treemapSize: "Taille du Treemap",
|
||||||
|
theme: "Thème",
|
||||||
|
lightboxLoadOnlyCurrent: "Désactiver le chargement des diapositives adjacentes pour le visualiseur d'images",
|
||||||
|
slideDuration: "Durée des diapositives",
|
||||||
|
resultSize: "Nombre de résultats par page",
|
||||||
|
tagOrOperator: "Utiliser l'opérateur OU lors de la spécification de plusieurs tags",
|
||||||
|
hideDuplicates: "Masquer les résultats en double"
|
||||||
|
},
|
||||||
|
queryMode: {
|
||||||
|
simple: "Simple",
|
||||||
|
advanced: "Avancé",
|
||||||
|
},
|
||||||
|
lang: {
|
||||||
|
en: "English",
|
||||||
|
fr: "Français"
|
||||||
|
},
|
||||||
|
displayMode: {
|
||||||
|
grid: "Grille",
|
||||||
|
list: "Liste",
|
||||||
|
},
|
||||||
|
columns: {
|
||||||
|
auto: "Auto"
|
||||||
|
},
|
||||||
|
treemapType: {
|
||||||
|
cascaded: "En cascade",
|
||||||
|
flat: "Plat (compact)"
|
||||||
|
},
|
||||||
|
treemapSize: {
|
||||||
|
small: "Petit",
|
||||||
|
medium: "Moyen",
|
||||||
|
large: "Grand",
|
||||||
|
xLarge: "xGrand",
|
||||||
|
xxLarge: "xxGrand",
|
||||||
|
custom: "Personnalisé",
|
||||||
|
},
|
||||||
|
treemapTiling: {
|
||||||
|
binary: "Binary",
|
||||||
|
squarify: "Squarify",
|
||||||
|
slice: "Slice",
|
||||||
|
dice: "Dice",
|
||||||
|
sliceDice: "Slice & Dice",
|
||||||
|
},
|
||||||
|
theme: {
|
||||||
|
light: "Clair",
|
||||||
|
black: "Noir"
|
||||||
|
},
|
||||||
|
hit: "résultat",
|
||||||
|
hits: "résultats",
|
||||||
|
details: "Détails",
|
||||||
|
stats: "Stats",
|
||||||
|
queryTime: "Durée de la requête",
|
||||||
|
totalSize: "Taille totale",
|
||||||
|
pathBar: {
|
||||||
|
placeholder: "Filtrer le chemin",
|
||||||
|
modalTitle: "Sélectionner le chemin"
|
||||||
|
},
|
||||||
|
debug: "Information de débogage",
|
||||||
|
debugDescription: "Informations utiles pour le débogage\n" +
|
||||||
|
"Si vous rencontrez des bogues ou si vous avez des suggestions pour de nouvelles fonctionnalités," +
|
||||||
|
" veuillez soumettre un nouvel Issue <a href='https://github.com/simon987/sist2/issues/new/choose'>ici</a>.",
|
||||||
|
tagline: "Tagline",
|
||||||
|
toast: {
|
||||||
|
esConnErrTitle: "Erreur de connexion Elasticsearch",
|
||||||
|
esConnErr: "Le module web a rencontré une erreur lors de la connexion à Elasticsearch." +
|
||||||
|
" Consultez les journaux du serveur pour plus d'informations..",
|
||||||
|
esQueryErrTitle: "Erreur de requête",
|
||||||
|
esQueryErr: "Impossible d'analyser ou d'exécuter la requête, veuillez consulter la documentation sur la " +
|
||||||
|
"recherche avancée. Voir les journaux du serveur pour plus d'informations.",
|
||||||
|
dupeTagTitle: "Tag en double",
|
||||||
|
dupeTag: "Ce tag existe déjà pour ce document."
|
||||||
|
},
|
||||||
|
saveTagModalTitle: "Ajouter un tag",
|
||||||
|
saveTagPlaceholder: "Nom du tag",
|
||||||
|
confirm: "Confirmer",
|
||||||
|
indexPickerPlaceholder: "Sélectionner un index",
|
||||||
|
sort: {
|
||||||
|
relevance: "Pertinence",
|
||||||
|
dateAsc: "Date (Plus ancient)",
|
||||||
|
dateDesc: "Date (Plus récent)",
|
||||||
|
sizeAsc: "Taille (Plus petit)",
|
||||||
|
sizeDesc: "Taille (Plus grand)",
|
||||||
|
random: "Aléatoire",
|
||||||
|
},
|
||||||
|
d3: {
|
||||||
|
mimeCount: "Distribution du nombre de fichiers par type de média",
|
||||||
|
mimeSize: "Distribution des tailles de fichiers par type de média",
|
||||||
|
dateHistogram: "Distribution des dates de modification",
|
||||||
|
sizeHistogram: "Distribution des tailles de fichier",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
29
sist2-vue/src/main.js
Normal file
29
sist2-vue/src/main.js
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import '@babel/polyfill'
|
||||||
|
import 'mutationobserver-shim'
|
||||||
|
import Vue from 'vue'
|
||||||
|
import './plugins/bootstrap-vue'
|
||||||
|
import App from './App.vue'
|
||||||
|
import router from './router'
|
||||||
|
import store from './store'
|
||||||
|
import VueI18n from "vue-i18n";
|
||||||
|
import messages from "@/i18n/messages";
|
||||||
|
|
||||||
|
|
||||||
|
import VueRouter from "vue-router";
|
||||||
|
|
||||||
|
Vue.config.productionTip = false;
|
||||||
|
|
||||||
|
Vue.use(VueI18n);
|
||||||
|
Vue.use(VueRouter);
|
||||||
|
|
||||||
|
const i18n = new VueI18n({
|
||||||
|
locale: "en",
|
||||||
|
messages: messages
|
||||||
|
});
|
||||||
|
|
||||||
|
new Vue({
|
||||||
|
router,
|
||||||
|
store,
|
||||||
|
i18n,
|
||||||
|
render: h => h(App)
|
||||||
|
}).$mount("#app");
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user