├── .dockerignore
├── .env.example
├── .github
└── workflows
│ └── main.yml
├── .gitignore
├── .node-version
├── .npmrc
├── .prettierrc.json
├── Dockerfile
├── LICENSE.txt
├── README.md
├── app.test.ts
├── app.ts
├── docker-compose.yml
├── docs
├── README.md
└── getting_started.md
├── extension
├── .gitignore
├── Dockerfile
├── README.md
├── addon
│ ├── background.html
│ ├── icons
│ │ ├── LICENSE
│ │ ├── logo.png
│ │ └── logo_dark.png
│ ├── manifest.json
│ ├── popup
│ │ ├── index.html
│ │ └── style.css
│ └── single-file
│ │ ├── dist
│ │ ├── extension-core.js
│ │ ├── extension-frames.js
│ │ ├── single-file-bootstrap.js
│ │ ├── single-file-frames.js
│ │ ├── single-file.js
│ │ └── web
│ │ │ └── hooks
│ │ │ ├── hooks-frames-web.js
│ │ │ └── hooks-web.js
│ │ └── lib
│ │ └── single-file
│ │ ├── LICENSE.md
│ │ ├── fetch.js
│ │ ├── frame-tree.js
│ │ └── lazy-timeout.js
├── content_scripts
│ └── archive.js
├── package-lock.json
├── package.json
├── popup
│ └── popup.js
└── webpack.config.js
├── index.ts
├── jest.config.ts
├── package-lock.json
├── package.json
├── src
├── helpers.ts
├── ledger
│ └── index.ts
├── qldb
│ └── index.ts
├── s3
│ ├── index.ts
│ └── s3.test.ts
├── store
│ ├── index.ts
│ └── store.test.ts
├── types
│ ├── Annotations.ts
│ ├── Bundle.ts
│ ├── Data.ts
│ ├── File.test.ts
│ ├── File.ts
│ └── Record.ts
└── verify
│ ├── index.ts
│ └── verify.test.ts
├── tsconfig.json
└── ui
├── .dockerignore
├── .gitignore
├── .node-version
├── .npmrc
├── .prettierrc.json
├── Dockerfile
├── default.conf
├── package-lock.json
├── package.json
├── src
├── app.css
├── app.html
├── global.d.ts
├── hooks.ts
├── lib
│ ├── Ledger
│ │ ├── EditingPanel.svelte
│ │ ├── EntryMetadata.svelte
│ │ ├── EntryThumbnail.svelte
│ │ ├── History.svelte
│ │ ├── LedgerEntry.svelte
│ │ └── index.ts
│ ├── header
│ │ ├── Header.svelte
│ │ └── logo.svg
│ ├── helpers.ts
│ ├── stores.ts
│ └── types.d.ts
└── routes
│ ├── __layout.svelte
│ ├── index.svelte
│ ├── library.svelte
│ └── verify.svelte
├── static
├── favicon.png
└── robots.txt
├── svelte.config.js
└── tsconfig.json
/.dockerignore:
--------------------------------------------------------------------------------
1 | ui/
2 | node_modules/
3 | out/
--------------------------------------------------------------------------------
/.env.example:
--------------------------------------------------------------------------------
1 | AWS_ACCESS_KEY_ID=""
2 | AWS_SECRET_ACCESS_KEY=""
3 | AWS_REGION=""
4 | LEDGER_NAME=""
5 | BUCKET_NAME=""
6 | DOC_TABLE_NAME=""
7 |
8 | SOURCE_FILES_DIRECTORY=""
9 | SOURCE_FILES_BUCKET=""
--------------------------------------------------------------------------------
/.github/workflows/main.yml:
--------------------------------------------------------------------------------
1 | name: CI
2 | on: [push]
3 |
4 | permissions:
5 | id-token: write
6 | contents: read
7 |
8 | jobs:
9 | run-tests:
10 | runs-on: ubuntu-latest
11 |
12 | steps:
13 | - uses: actions/checkout@v2
14 | - uses: actions/setup-node@v2
15 | with:
16 | node-version: '16'
17 | - name: Install modules
18 | run: npm install
19 | - name: Configure AWS Credentials
20 | uses: aws-actions/configure-aws-credentials@v4
21 | with:
22 | aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
23 | aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
24 | aws-region: eu-central-1
25 | - name: Run tests
26 | run: npm run test
27 | env:
28 | BUCKET_NAME: ${{ secrets.BUCKET_NAME }}
29 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | function.zip
3 | node_modules
4 | .env
5 | out/
--------------------------------------------------------------------------------
/.node-version:
--------------------------------------------------------------------------------
1 | 16.13.0
2 |
--------------------------------------------------------------------------------
/.npmrc:
--------------------------------------------------------------------------------
1 | engine-strict=true
2 |
--------------------------------------------------------------------------------
/.prettierrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "singleQuote": true,
3 | "arrowParens": "avoid",
4 | "jsxSingleQuote": true,
5 | "svelteSortOrder": "options-scripts-styles-markup"
6 | }
7 |
--------------------------------------------------------------------------------
/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:16.13.0-alpine3.14
2 |
3 | LABEL version="0.1"
4 | LABEL description="API handling interactiong with Amazon QLDB"
5 |
6 | WORKDIR /app
7 | COPY ["package.json", "package-lock.json", "./"]
8 | RUN npm install
9 | COPY . .
10 |
11 | EXPOSE 3000
12 |
13 | CMD ["npm", "run", "start"]
--------------------------------------------------------------------------------
/LICENSE.txt:
--------------------------------------------------------------------------------
1 |
2 | The Digital Evidence Preservation Toolkit
3 | Copyright (C) 2021 • Basile Simon
4 |
5 | This program is free software: you can redistribute it and/or modify
6 | it under the terms of the GNU General Public License as published by
7 | the Free Software Foundation, either version 3 of the License, or
8 | (at your option) any later version.
9 |
10 | This program is distributed in the hope that it will be useful,
11 | but WITHOUT ANY WARRANTY; without even the implied warranty of
12 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 | GNU General Public License for more details.
14 |
15 | GNU GENERAL PUBLIC LICENSE
16 | Version 3, 29 June 2007
17 |
18 | Copyright (C) 2007 Free Software Foundation, Inc.
19 | Everyone is permitted to copy and distribute verbatim copies
20 | of this license document, but changing it is not allowed.
21 |
22 | Preamble
23 |
24 | The GNU General Public License is a free, copyleft license for
25 | software and other kinds of works.
26 |
27 | The licenses for most software and other practical works are designed
28 | to take away your freedom to share and change the works. By contrast,
29 | the GNU General Public License is intended to guarantee your freedom to
30 | share and change all versions of a program--to make sure it remains free
31 | software for all its users. We, the Free Software Foundation, use the
32 | GNU General Public License for most of our software; it applies also to
33 | any other work released this way by its authors. You can apply it to
34 | your programs, too.
35 |
36 | When we speak of free software, we are referring to freedom, not
37 | price. Our General Public Licenses are designed to make sure that you
38 | have the freedom to distribute copies of free software (and charge for
39 | them if you wish), that you receive source code or can get it if you
40 | want it, that you can change the software or use pieces of it in new
41 | free programs, and that you know you can do these things.
42 |
43 | To protect your rights, we need to prevent others from denying you
44 | these rights or asking you to surrender the rights. Therefore, you have
45 | certain responsibilities if you distribute copies of the software, or if
46 | you modify it: responsibilities to respect the freedom of others.
47 |
48 | For example, if you distribute copies of such a program, whether
49 | gratis or for a fee, you must pass on to the recipients the same
50 | freedoms that you received. You must make sure that they, too, receive
51 | or can get the source code. And you must show them these terms so they
52 | know their rights.
53 |
54 | Developers that use the GNU GPL protect your rights with two steps:
55 | (1) assert copyright on the software, and (2) offer you this License
56 | giving you legal permission to copy, distribute and/or modify it.
57 |
58 | For the developers' and authors' protection, the GPL clearly explains
59 | that there is no warranty for this free software. For both users' and
60 | authors' sake, the GPL requires that modified versions be marked as
61 | changed, so that their problems will not be attributed erroneously to
62 | authors of previous versions.
63 |
64 | Some devices are designed to deny users access to install or run
65 | modified versions of the software inside them, although the manufacturer
66 | can do so. This is fundamentally incompatible with the aim of
67 | protecting users' freedom to change the software. The systematic
68 | pattern of such abuse occurs in the area of products for individuals to
69 | use, which is precisely where it is most unacceptable. Therefore, we
70 | have designed this version of the GPL to prohibit the practice for those
71 | products. If such problems arise substantially in other domains, we
72 | stand ready to extend this provision to those domains in future versions
73 | of the GPL, as needed to protect the freedom of users.
74 |
75 | Finally, every program is threatened constantly by software patents.
76 | States should not allow patents to restrict development and use of
77 | software on general-purpose computers, but in those that do, we wish to
78 | avoid the special danger that patents applied to a free program could
79 | make it effectively proprietary. To prevent this, the GPL assures that
80 | patents cannot be used to render the program non-free.
81 |
82 | The precise terms and conditions for copying, distribution and
83 | modification follow.
84 |
85 | TERMS AND CONDITIONS
86 |
87 | 0. Definitions.
88 |
89 | "This License" refers to version 3 of the GNU General Public License.
90 |
91 | "Copyright" also means copyright-like laws that apply to other kinds of
92 | works, such as semiconductor masks.
93 |
94 | "The Program" refers to any copyrightable work licensed under this
95 | License. Each licensee is addressed as "you". "Licensees" and
96 | "recipients" may be individuals or organizations.
97 |
98 | To "modify" a work means to copy from or adapt all or part of the work
99 | in a fashion requiring copyright permission, other than the making of an
100 | exact copy. The resulting work is called a "modified version" of the
101 | earlier work or a work "based on" the earlier work.
102 |
103 | A "covered work" means either the unmodified Program or a work based
104 | on the Program.
105 |
106 | To "propagate" a work means to do anything with it that, without
107 | permission, would make you directly or secondarily liable for
108 | infringement under applicable copyright law, except executing it on a
109 | computer or modifying a private copy. Propagation includes copying,
110 | distribution (with or without modification), making available to the
111 | public, and in some countries other activities as well.
112 |
113 | To "convey" a work means any kind of propagation that enables other
114 | parties to make or receive copies. Mere interaction with a user through
115 | a computer network, with no transfer of a copy, is not conveying.
116 |
117 | An interactive user interface displays "Appropriate Legal Notices"
118 | to the extent that it includes a convenient and prominently visible
119 | feature that (1) displays an appropriate copyright notice, and (2)
120 | tells the user that there is no warranty for the work (except to the
121 | extent that warranties are provided), that licensees may convey the
122 | work under this License, and how to view a copy of this License. If
123 | the interface presents a list of user commands or options, such as a
124 | menu, a prominent item in the list meets this criterion.
125 |
126 | 1. Source Code.
127 |
128 | The "source code" for a work means the preferred form of the work
129 | for making modifications to it. "Object code" means any non-source
130 | form of a work.
131 |
132 | A "Standard Interface" means an interface that either is an official
133 | standard defined by a recognized standards body, or, in the case of
134 | interfaces specified for a particular programming language, one that
135 | is widely used among developers working in that language.
136 |
137 | The "System Libraries" of an executable work include anything, other
138 | than the work as a whole, that (a) is included in the normal form of
139 | packaging a Major Component, but which is not part of that Major
140 | Component, and (b) serves only to enable use of the work with that
141 | Major Component, or to implement a Standard Interface for which an
142 | implementation is available to the public in source code form. A
143 | "Major Component", in this context, means a major essential component
144 | (kernel, window system, and so on) of the specific operating system
145 | (if any) on which the executable work runs, or a compiler used to
146 | produce the work, or an object code interpreter used to run it.
147 |
148 | The "Corresponding Source" for a work in object code form means all
149 | the source code needed to generate, install, and (for an executable
150 | work) run the object code and to modify the work, including scripts to
151 | control those activities. However, it does not include the work's
152 | System Libraries, or general-purpose tools or generally available free
153 | programs which are used unmodified in performing those activities but
154 | which are not part of the work. For example, Corresponding Source
155 | includes interface definition files associated with source files for
156 | the work, and the source code for shared libraries and dynamically
157 | linked subprograms that the work is specifically designed to require,
158 | such as by intimate data communication or control flow between those
159 | subprograms and other parts of the work.
160 |
161 | The Corresponding Source need not include anything that users
162 | can regenerate automatically from other parts of the Corresponding
163 | Source.
164 |
165 | The Corresponding Source for a work in source code form is that
166 | same work.
167 |
168 | 2. Basic Permissions.
169 |
170 | All rights granted under this License are granted for the term of
171 | copyright on the Program, and are irrevocable provided the stated
172 | conditions are met. This License explicitly affirms your unlimited
173 | permission to run the unmodified Program. The output from running a
174 | covered work is covered by this License only if the output, given its
175 | content, constitutes a covered work. This License acknowledges your
176 | rights of fair use or other equivalent, as provided by copyright law.
177 |
178 | You may make, run and propagate covered works that you do not
179 | convey, without conditions so long as your license otherwise remains
180 | in force. You may convey covered works to others for the sole purpose
181 | of having them make modifications exclusively for you, or provide you
182 | with facilities for running those works, provided that you comply with
183 | the terms of this License in conveying all material for which you do
184 | not control copyright. Those thus making or running the covered works
185 | for you must do so exclusively on your behalf, under your direction
186 | and control, on terms that prohibit them from making any copies of
187 | your copyrighted material outside their relationship with you.
188 |
189 | Conveying under any other circumstances is permitted solely under
190 | the conditions stated below. Sublicensing is not allowed; section 10
191 | makes it unnecessary.
192 |
193 | 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
194 |
195 | No covered work shall be deemed part of an effective technological
196 | measure under any applicable law fulfilling obligations under article
197 | 11 of the WIPO copyright treaty adopted on 20 December 1996, or
198 | similar laws prohibiting or restricting circumvention of such
199 | measures.
200 |
201 | When you convey a covered work, you waive any legal power to forbid
202 | circumvention of technological measures to the extent such circumvention
203 | is effected by exercising rights under this License with respect to
204 | the covered work, and you disclaim any intention to limit operation or
205 | modification of the work as a means of enforcing, against the work's
206 | users, your or third parties' legal rights to forbid circumvention of
207 | technological measures.
208 |
209 | 4. Conveying Verbatim Copies.
210 |
211 | You may convey verbatim copies of the Program's source code as you
212 | receive it, in any medium, provided that you conspicuously and
213 | appropriately publish on each copy an appropriate copyright notice;
214 | keep intact all notices stating that this License and any
215 | non-permissive terms added in accord with section 7 apply to the code;
216 | keep intact all notices of the absence of any warranty; and give all
217 | recipients a copy of this License along with the Program.
218 |
219 | You may charge any price or no price for each copy that you convey,
220 | and you may offer support or warranty protection for a fee.
221 |
222 | 5. Conveying Modified Source Versions.
223 |
224 | You may convey a work based on the Program, or the modifications to
225 | produce it from the Program, in the form of source code under the
226 | terms of section 4, provided that you also meet all of these conditions:
227 |
228 | a) The work must carry prominent notices stating that you modified
229 | it, and giving a relevant date.
230 |
231 | b) The work must carry prominent notices stating that it is
232 | released under this License and any conditions added under section
233 | 7. This requirement modifies the requirement in section 4 to
234 | "keep intact all notices".
235 |
236 | c) You must license the entire work, as a whole, under this
237 | License to anyone who comes into possession of a copy. This
238 | License will therefore apply, along with any applicable section 7
239 | additional terms, to the whole of the work, and all its parts,
240 | regardless of how they are packaged. This License gives no
241 | permission to license the work in any other way, but it does not
242 | invalidate such permission if you have separately received it.
243 |
244 | d) If the work has interactive user interfaces, each must display
245 | Appropriate Legal Notices; however, if the Program has interactive
246 | interfaces that do not display Appropriate Legal Notices, your
247 | work need not make them do so.
248 |
249 | A compilation of a covered work with other separate and independent
250 | works, which are not by their nature extensions of the covered work,
251 | and which are not combined with it such as to form a larger program,
252 | in or on a volume of a storage or distribution medium, is called an
253 | "aggregate" if the compilation and its resulting copyright are not
254 | used to limit the access or legal rights of the compilation's users
255 | beyond what the individual works permit. Inclusion of a covered work
256 | in an aggregate does not cause this License to apply to the other
257 | parts of the aggregate.
258 |
259 | 6. Conveying Non-Source Forms.
260 |
261 | You may convey a covered work in object code form under the terms
262 | of sections 4 and 5, provided that you also convey the
263 | machine-readable Corresponding Source under the terms of this License,
264 | in one of these ways:
265 |
266 | a) Convey the object code in, or embodied in, a physical product
267 | (including a physical distribution medium), accompanied by the
268 | Corresponding Source fixed on a durable physical medium
269 | customarily used for software interchange.
270 |
271 | b) Convey the object code in, or embodied in, a physical product
272 | (including a physical distribution medium), accompanied by a
273 | written offer, valid for at least three years and valid for as
274 | long as you offer spare parts or customer support for that product
275 | model, to give anyone who possesses the object code either (1) a
276 | copy of the Corresponding Source for all the software in the
277 | product that is covered by this License, on a durable physical
278 | medium customarily used for software interchange, for a price no
279 | more than your reasonable cost of physically performing this
280 | conveying of source, or (2) access to copy the
281 | Corresponding Source from a network server at no charge.
282 |
283 | c) Convey individual copies of the object code with a copy of the
284 | written offer to provide the Corresponding Source. This
285 | alternative is allowed only occasionally and noncommercially, and
286 | only if you received the object code with such an offer, in accord
287 | with subsection 6b.
288 |
289 | d) Convey the object code by offering access from a designated
290 | place (gratis or for a charge), and offer equivalent access to the
291 | Corresponding Source in the same way through the same place at no
292 | further charge. You need not require recipients to copy the
293 | Corresponding Source along with the object code. If the place to
294 | copy the object code is a network server, the Corresponding Source
295 | may be on a different server (operated by you or a third party)
296 | that supports equivalent copying facilities, provided you maintain
297 | clear directions next to the object code saying where to find the
298 | Corresponding Source. Regardless of what server hosts the
299 | Corresponding Source, you remain obligated to ensure that it is
300 | available for as long as needed to satisfy these requirements.
301 |
302 | e) Convey the object code using peer-to-peer transmission, provided
303 | you inform other peers where the object code and Corresponding
304 | Source of the work are being offered to the general public at no
305 | charge under subsection 6d.
306 |
307 | A separable portion of the object code, whose source code is excluded
308 | from the Corresponding Source as a System Library, need not be
309 | included in conveying the object code work.
310 |
311 | A "User Product" is either (1) a "consumer product", which means any
312 | tangible personal property which is normally used for personal, family,
313 | or household purposes, or (2) anything designed or sold for incorporation
314 | into a dwelling. In determining whether a product is a consumer product,
315 | doubtful cases shall be resolved in favor of coverage. For a particular
316 | product received by a particular user, "normally used" refers to a
317 | typical or common use of that class of product, regardless of the status
318 | of the particular user or of the way in which the particular user
319 | actually uses, or expects or is expected to use, the product. A product
320 | is a consumer product regardless of whether the product has substantial
321 | commercial, industrial or non-consumer uses, unless such uses represent
322 | the only significant mode of use of the product.
323 |
324 | "Installation Information" for a User Product means any methods,
325 | procedures, authorization keys, or other information required to install
326 | and execute modified versions of a covered work in that User Product from
327 | a modified version of its Corresponding Source. The information must
328 | suffice to ensure that the continued functioning of the modified object
329 | code is in no case prevented or interfered with solely because
330 | modification has been made.
331 |
332 | If you convey an object code work under this section in, or with, or
333 | specifically for use in, a User Product, and the conveying occurs as
334 | part of a transaction in which the right of possession and use of the
335 | User Product is transferred to the recipient in perpetuity or for a
336 | fixed term (regardless of how the transaction is characterized), the
337 | Corresponding Source conveyed under this section must be accompanied
338 | by the Installation Information. But this requirement does not apply
339 | if neither you nor any third party retains the ability to install
340 | modified object code on the User Product (for example, the work has
341 | been installed in ROM).
342 |
343 | The requirement to provide Installation Information does not include a
344 | requirement to continue to provide support service, warranty, or updates
345 | for a work that has been modified or installed by the recipient, or for
346 | the User Product in which it has been modified or installed. Access to a
347 | network may be denied when the modification itself materially and
348 | adversely affects the operation of the network or violates the rules and
349 | protocols for communication across the network.
350 |
351 | Corresponding Source conveyed, and Installation Information provided,
352 | in accord with this section must be in a format that is publicly
353 | documented (and with an implementation available to the public in
354 | source code form), and must require no special password or key for
355 | unpacking, reading or copying.
356 |
357 | 7. Additional Terms.
358 |
359 | "Additional permissions" are terms that supplement the terms of this
360 | License by making exceptions from one or more of its conditions.
361 | Additional permissions that are applicable to the entire Program shall
362 | be treated as though they were included in this License, to the extent
363 | that they are valid under applicable law. If additional permissions
364 | apply only to part of the Program, that part may be used separately
365 | under those permissions, but the entire Program remains governed by
366 | this License without regard to the additional permissions.
367 |
368 | When you convey a copy of a covered work, you may at your option
369 | remove any additional permissions from that copy, or from any part of
370 | it. (Additional permissions may be written to require their own
371 | removal in certain cases when you modify the work.) You may place
372 | additional permissions on material, added by you to a covered work,
373 | for which you have or can give appropriate copyright permission.
374 |
375 | Notwithstanding any other provision of this License, for material you
376 | add to a covered work, you may (if authorized by the copyright holders of
377 | that material) supplement the terms of this License with terms:
378 |
379 | a) Disclaiming warranty or limiting liability differently from the
380 | terms of sections 15 and 16 of this License; or
381 |
382 | b) Requiring preservation of specified reasonable legal notices or
383 | author attributions in that material or in the Appropriate Legal
384 | Notices displayed by works containing it; or
385 |
386 | c) Prohibiting misrepresentation of the origin of that material, or
387 | requiring that modified versions of such material be marked in
388 | reasonable ways as different from the original version; or
389 |
390 | d) Limiting the use for publicity purposes of names of licensors or
391 | authors of the material; or
392 |
393 | e) Declining to grant rights under trademark law for use of some
394 | trade names, trademarks, or service marks; or
395 |
396 | f) Requiring indemnification of licensors and authors of that
397 | material by anyone who conveys the material (or modified versions of
398 | it) with contractual assumptions of liability to the recipient, for
399 | any liability that these contractual assumptions directly impose on
400 | those licensors and authors.
401 |
402 | All other non-permissive additional terms are considered "further
403 | restrictions" within the meaning of section 10. If the Program as you
404 | received it, or any part of it, contains a notice stating that it is
405 | governed by this License along with a term that is a further
406 | restriction, you may remove that term. If a license document contains
407 | a further restriction but permits relicensing or conveying under this
408 | License, you may add to a covered work material governed by the terms
409 | of that license document, provided that the further restriction does
410 | not survive such relicensing or conveying.
411 |
412 | If you add terms to a covered work in accord with this section, you
413 | must place, in the relevant source files, a statement of the
414 | additional terms that apply to those files, or a notice indicating
415 | where to find the applicable terms.
416 |
417 | Additional terms, permissive or non-permissive, may be stated in the
418 | form of a separately written license, or stated as exceptions;
419 | the above requirements apply either way.
420 |
421 | 8. Termination.
422 |
423 | You may not propagate or modify a covered work except as expressly
424 | provided under this License. Any attempt otherwise to propagate or
425 | modify it is void, and will automatically terminate your rights under
426 | this License (including any patent licenses granted under the third
427 | paragraph of section 11).
428 |
429 | However, if you cease all violation of this License, then your
430 | license from a particular copyright holder is reinstated (a)
431 | provisionally, unless and until the copyright holder explicitly and
432 | finally terminates your license, and (b) permanently, if the copyright
433 | holder fails to notify you of the violation by some reasonable means
434 | prior to 60 days after the cessation.
435 |
436 | Moreover, your license from a particular copyright holder is
437 | reinstated permanently if the copyright holder notifies you of the
438 | violation by some reasonable means, this is the first time you have
439 | received notice of violation of this License (for any work) from that
440 | copyright holder, and you cure the violation prior to 30 days after
441 | your receipt of the notice.
442 |
443 | Termination of your rights under this section does not terminate the
444 | licenses of parties who have received copies or rights from you under
445 | this License. If your rights have been terminated and not permanently
446 | reinstated, you do not qualify to receive new licenses for the same
447 | material under section 10.
448 |
449 | 9. Acceptance Not Required for Having Copies.
450 |
451 | You are not required to accept this License in order to receive or
452 | run a copy of the Program. Ancillary propagation of a covered work
453 | occurring solely as a consequence of using peer-to-peer transmission
454 | to receive a copy likewise does not require acceptance. However,
455 | nothing other than this License grants you permission to propagate or
456 | modify any covered work. These actions infringe copyright if you do
457 | not accept this License. Therefore, by modifying or propagating a
458 | covered work, you indicate your acceptance of this License to do so.
459 |
460 | 10. Automatic Licensing of Downstream Recipients.
461 |
462 | Each time you convey a covered work, the recipient automatically
463 | receives a license from the original licensors, to run, modify and
464 | propagate that work, subject to this License. You are not responsible
465 | for enforcing compliance by third parties with this License.
466 |
467 | An "entity transaction" is a transaction transferring control of an
468 | organization, or substantially all assets of one, or subdividing an
469 | organization, or merging organizations. If propagation of a covered
470 | work results from an entity transaction, each party to that
471 | transaction who receives a copy of the work also receives whatever
472 | licenses to the work the party's predecessor in interest had or could
473 | give under the previous paragraph, plus a right to possession of the
474 | Corresponding Source of the work from the predecessor in interest, if
475 | the predecessor has it or can get it with reasonable efforts.
476 |
477 | You may not impose any further restrictions on the exercise of the
478 | rights granted or affirmed under this License. For example, you may
479 | not impose a license fee, royalty, or other charge for exercise of
480 | rights granted under this License, and you may not initiate litigation
481 | (including a cross-claim or counterclaim in a lawsuit) alleging that
482 | any patent claim is infringed by making, using, selling, offering for
483 | sale, or importing the Program or any portion of it.
484 |
485 | 11. Patents.
486 |
487 | A "contributor" is a copyright holder who authorizes use under this
488 | License of the Program or a work on which the Program is based. The
489 | work thus licensed is called the contributor's "contributor version".
490 |
491 | A contributor's "essential patent claims" are all patent claims
492 | owned or controlled by the contributor, whether already acquired or
493 | hereafter acquired, that would be infringed by some manner, permitted
494 | by this License, of making, using, or selling its contributor version,
495 | but do not include claims that would be infringed only as a
496 | consequence of further modification of the contributor version. For
497 | purposes of this definition, "control" includes the right to grant
498 | patent sublicenses in a manner consistent with the requirements of
499 | this License.
500 |
501 | Each contributor grants you a non-exclusive, worldwide, royalty-free
502 | patent license under the contributor's essential patent claims, to
503 | make, use, sell, offer for sale, import and otherwise run, modify and
504 | propagate the contents of its contributor version.
505 |
506 | In the following three paragraphs, a "patent license" is any express
507 | agreement or commitment, however denominated, not to enforce a patent
508 | (such as an express permission to practice a patent or covenant not to
509 | sue for patent infringement). To "grant" such a patent license to a
510 | party means to make such an agreement or commitment not to enforce a
511 | patent against the party.
512 |
513 | If you convey a covered work, knowingly relying on a patent license,
514 | and the Corresponding Source of the work is not available for anyone
515 | to copy, free of charge and under the terms of this License, through a
516 | publicly available network server or other readily accessible means,
517 | then you must either (1) cause the Corresponding Source to be so
518 | available, or (2) arrange to deprive yourself of the benefit of the
519 | patent license for this particular work, or (3) arrange, in a manner
520 | consistent with the requirements of this License, to extend the patent
521 | license to downstream recipients. "Knowingly relying" means you have
522 | actual knowledge that, but for the patent license, your conveying the
523 | covered work in a country, or your recipient's use of the covered work
524 | in a country, would infringe one or more identifiable patents in that
525 | country that you have reason to believe are valid.
526 |
527 | If, pursuant to or in connection with a single transaction or
528 | arrangement, you convey, or propagate by procuring conveyance of, a
529 | covered work, and grant a patent license to some of the parties
530 | receiving the covered work authorizing them to use, propagate, modify
531 | or convey a specific copy of the covered work, then the patent license
532 | you grant is automatically extended to all recipients of the covered
533 | work and works based on it.
534 |
535 | A patent license is "discriminatory" if it does not include within
536 | the scope of its coverage, prohibits the exercise of, or is
537 | conditioned on the non-exercise of one or more of the rights that are
538 | specifically granted under this License. You may not convey a covered
539 | work if you are a party to an arrangement with a third party that is
540 | in the business of distributing software, under which you make payment
541 | to the third party based on the extent of your activity of conveying
542 | the work, and under which the third party grants, to any of the
543 | parties who would receive the covered work from you, a discriminatory
544 | patent license (a) in connection with copies of the covered work
545 | conveyed by you (or copies made from those copies), or (b) primarily
546 | for and in connection with specific products or compilations that
547 | contain the covered work, unless you entered into that arrangement,
548 | or that patent license was granted, prior to 28 March 2007.
549 |
550 | Nothing in this License shall be construed as excluding or limiting
551 | any implied license or other defenses to infringement that may
552 | otherwise be available to you under applicable patent law.
553 |
554 | 12. No Surrender of Others' Freedom.
555 |
556 | If conditions are imposed on you (whether by court order, agreement or
557 | otherwise) that contradict the conditions of this License, they do not
558 | excuse you from the conditions of this License. If you cannot convey a
559 | covered work so as to satisfy simultaneously your obligations under this
560 | License and any other pertinent obligations, then as a consequence you may
561 | not convey it at all. For example, if you agree to terms that obligate you
562 | to collect a royalty for further conveying from those to whom you convey
563 | the Program, the only way you could satisfy both those terms and this
564 | License would be to refrain entirely from conveying the Program.
565 |
566 | 13. Use with the GNU Affero General Public License.
567 |
568 | Notwithstanding any other provision of this License, you have
569 | permission to link or combine any covered work with a work licensed
570 | under version 3 of the GNU Affero General Public License into a single
571 | combined work, and to convey the resulting work. The terms of this
572 | License will continue to apply to the part which is the covered work,
573 | but the special requirements of the GNU Affero General Public License,
574 | section 13, concerning interaction through a network will apply to the
575 | combination as such.
576 |
577 | 14. Revised Versions of this License.
578 |
579 | The Free Software Foundation may publish revised and/or new versions of
580 | the GNU General Public License from time to time. Such new versions will
581 | be similar in spirit to the present version, but may differ in detail to
582 | address new problems or concerns.
583 |
584 | Each version is given a distinguishing version number. If the
585 | Program specifies that a certain numbered version of the GNU General
586 | Public License "or any later version" applies to it, you have the
587 | option of following the terms and conditions either of that numbered
588 | version or of any later version published by the Free Software
589 | Foundation. If the Program does not specify a version number of the
590 | GNU General Public License, you may choose any version ever published
591 | by the Free Software Foundation.
592 |
593 | If the Program specifies that a proxy can decide which future
594 | versions of the GNU General Public License can be used, that proxy's
595 | public statement of acceptance of a version permanently authorizes you
596 | to choose that version for the Program.
597 |
598 | Later license versions may give you additional or different
599 | permissions. However, no additional obligations are imposed on any
600 | author or copyright holder as a result of your choosing to follow a
601 | later version.
602 |
603 | 15. Disclaimer of Warranty.
604 |
605 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
606 | APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
607 | HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
608 | OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
609 | THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
610 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
611 | IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
612 | ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
613 |
614 | 16. Limitation of Liability.
615 |
616 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
617 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
618 | THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
619 | GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
620 | USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
621 | DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
622 | PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
623 | EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
624 | SUCH DAMAGES.
625 |
626 | 17. Interpretation of Sections 15 and 16.
627 |
628 | If the disclaimer of warranty and limitation of liability provided
629 | above cannot be given local legal effect according to their terms,
630 | reviewing courts shall apply local law that most closely approximates
631 | an absolute waiver of all civil liability in connection with the
632 | Program, unless a warranty or assumption of liability accompanies a
633 | copy of the Program in return for a fee.
634 |
635 | END OF TERMS AND CONDITIONS
636 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # The Digital Evidence Preservation Toolkit
2 |
3 | _(Or **DEPToolkit** for short. Bit of a mouthful otherwise)_
4 |
5 |
6 |
7 |
8 |
9 | A proof-of-concept software for researchers and small teams sifting through online material. With only one click of the mouse, the material will be **archived in a framework demonstrating chain of custody and stored durably**. Once included in the growing database, users will be able to go back to search through and **annotate the material**, and to **export working copies** of said material for publication and dissemination.
10 |
11 | A database built thusly can be handed to a prosecutor ten years down the line, and they will be able to say with mathematical certainty: “the material in this archive is identical and contemporary to the one saved at the time, ten years ago.”
12 |
13 | Built with the support of **Prototype Fund**, the **German Federal Ministry for Education and Research**, the **Open Knowledge Foundation**, and **Amazon Web Services**.
14 |
15 | ### Architecture
16 |
17 | The Docker Compose orchestration is composed of several services:
18 |
19 | - An Express/TypeScript API,
20 | - A plain JS browser extension
21 | - And a frontend
22 |
23 | To start the whole app:
24 |
25 | ```sh
26 | $ docker-compose up
27 | ```
28 |
29 | ### Help & Contact
30 |
31 | - [Getting started](docs/getting_started.md)
32 | - [Technical documentation](docs/)
33 | - Email: basile at digitalevidencetoolkit dot org
34 |
--------------------------------------------------------------------------------
/app.test.ts:
--------------------------------------------------------------------------------
1 | import supertest from 'supertest';
2 | import { ImportMock } from 'ts-mock-imports';
3 |
4 | import { pprint } from './src/helpers';
5 | import * as Ledger from './src/ledger';
6 |
7 | import app from './app';
8 |
9 | // TODO
10 | // - isolate test env (pass outDir as a param to app, create a specific outDir
11 | // for test)
12 | // - test that an existing file is returned
13 | // - test that an existing dotFile is not returned
14 | describe('The file endpoint', () => {
15 | it('should 404 if the file doesnt exist.', async () => {
16 | // return supertest(app)
17 | // .get('/file/foo')
18 | // .then(response => {
19 | // expect(response.status).toBe(500);
20 | // });
21 | });
22 |
23 | it('should 200 with the file if it exists', () => {
24 | // create a file in the test dir
25 | // ask for it
26 | // file is returned
27 | });
28 |
29 | it('should not return an existing dot file', () => {
30 | // create a dot file in the test dir
31 | // ask for it
32 | // file is not returned
33 | });
34 | });
35 |
36 | describe('The history endpoint', () => {
37 | afterEach(() => {
38 | ImportMock.restore();
39 | });
40 |
41 | it("should use the ledger's history API", () => {
42 | const result = { foo: 'bar' };
43 |
44 | const getListHistoryMock = ImportMock.mockFunction(
45 | Ledger,
46 | 'listDocHistory',
47 | Promise.resolve(result)
48 | );
49 |
50 | const name = 'foo';
51 |
52 | return supertest(app)
53 | .get(`/history/${name}`)
54 | .then(response => {
55 | expect(getListHistoryMock.calledOnce).toBe(true);
56 | expect(getListHistoryMock.calledWith(name)).toBe(true);
57 |
58 | expect(response.status).toBe(200);
59 | expect(response.text).toBe(pprint(result));
60 | });
61 | });
62 | });
63 |
64 | describe('The export-copy endpoint', () => {});
65 |
--------------------------------------------------------------------------------
/app.ts:
--------------------------------------------------------------------------------
1 | import express, { Application, Request, Response } from 'express';
2 | import { config } from 'dotenv';
3 | import { join } from 'path';
4 | import cors from 'cors';
5 | import sharp from 'sharp';
6 | import formidable, { Fields, Files } from 'formidable';
7 | import chalk from 'chalk';
8 | import fs from 'fs/promises';
9 | import { parse } from 'path';
10 |
11 | import * as Ledger from './src/ledger';
12 | import * as Store from './src/store';
13 | import * as Bundle from './src/types/Bundle';
14 | import * as Record from './src/types/Record';
15 | import * as Verify from './src/verify';
16 |
17 | import { pprint, cleanupBase64 } from './src/helpers';
18 |
19 | // set up .env variables as environment variables
20 | config();
21 |
22 | const outDir = join(__dirname, './out');
23 |
24 | const app: Application = express();
25 |
26 | app.use(cors());
27 |
28 | // Body parsing Middleware
29 | app.use(express.json());
30 | app.use(express.urlencoded({ extended: true }));
31 |
32 | app.get('/file/:id', async (req: Request, res: Response) => {
33 | const { id } = req.params;
34 | return Store.getFile(id, res);
35 | });
36 |
37 | app.get(
38 | '/history/:sku',
39 | async (req: Request, res: Response): Promise => {
40 | const { sku } = req.params;
41 | const result = await Ledger.listDocHistory(sku);
42 | console.log(chalk.bold(`GET /history/${sku}`));
43 | return res.status(200).send(pprint(result));
44 | }
45 | );
46 |
47 | app.get(
48 | '/export-copy/:sku',
49 | async (req: Request, res: Response): Promise => {
50 | const { sku } = req.params;
51 | const rootDir = outDir;
52 | const options = {
53 | root: rootDir,
54 | dotfiles: 'deny',
55 | };
56 | // `sku` comes with .zip at the end, which we must remove
57 | const cleanSku = parse(sku).name;
58 | const result = await Ledger.getDoc(cleanSku, 'id');
59 | if (result === null) {
60 | res.status(404).send(`Could not find the resource you asked for: ${sku}`);
61 | } else {
62 | await Store.makeZip(result, rootDir, rootDir);
63 | res
64 | .set(`Content-Type`, `application/octet-stream`)
65 | .set(`Content-Disposition`, `attachment; filename=${sku}`)
66 | .sendFile(`${sku}`, options);
67 | }
68 | }
69 | );
70 |
71 | app.get(
72 | '/list-docs',
73 | async (req: Request, res: Response): Promise => {
74 | const result = await Ledger.listDocs();
75 | console.log(chalk.bold(`GET /list-docs`));
76 | return res.status(200).send(pprint(result));
77 | }
78 | );
79 |
80 | app.post('/form', async (req: Request, res: Response): Promise => {
81 | console.log(chalk.bold(`POST /form`));
82 | return new Promise((resolve, reject) => {
83 | const form = new formidable.IncomingForm();
84 | form.parse(req, async (err: Error, fields: Fields): Promise => {
85 | if (err) {
86 | resolve(res.status(400).send(`${err.name}: ${err.message}`)); // FIXME: don't expose js errors to public
87 | }
88 | const { url, title, scr } = fields;
89 | const base64Data = cleanupBase64(scr as string); // FIXME: don't use 'as string', instead make sure we have string instead of string[]
90 | const screenshotData = new Buffer(base64Data, 'base64');
91 | const thumbnailData = await sharp(screenshotData)
92 | .resize(320, 240, { fit: 'inside' })
93 | .toBuffer();
94 | let onefileData: string;
95 | onefileData = fields.onefile as string;
96 |
97 | const screenshot = { kind: 'screenshot' as const, data: screenshotData };
98 | const thumbnail = {
99 | kind: 'screenshot_thumbnail' as const,
100 | data: thumbnailData,
101 | };
102 | const onefile = { kind: 'one_file' as const, data: onefileData };
103 |
104 | await Store.newBundle([screenshot, thumbnail, onefile], {
105 | type: 'local',
106 | directory: outDir,
107 | })
108 | .then((bundle: Bundle.Bundle) => {
109 | const record = {
110 | bundle,
111 | annotations: { description: '' },
112 | data: { url: url as string, title: title as string },
113 | };
114 | return record;
115 | })
116 | .then((r: Record.Record) => {
117 | return Ledger.insertDoc(r);
118 | })
119 | .then(_ => {
120 | console.log(`ledger inserted correctly`);
121 | resolve(res.status(200).send('Received POST on /form'));
122 | })
123 | .catch(e => {
124 | resolve(
125 | res.status(422).send(`${e.name} (type ${e.type}): ${e.message}`)
126 | );
127 | });
128 | });
129 | });
130 | });
131 |
132 | app.post(
133 | '/edit-description/:sku',
134 | async (req: Request, res: Response): Promise => {
135 | const { sku } = req.params;
136 | console.log(chalk.bold(`POST /edit-description/${sku}`));
137 | return new Promise((resolve, reject) => {
138 | const form = new formidable.IncomingForm();
139 | form.parse(req, async (err: Error, fields: Fields): Promise => {
140 | if (err) {
141 | resolve(res.status(400).send(`${err.name}: ${err.message}`)); // FIXME: don't expose js errors to public
142 | }
143 | const { description } = fields;
144 | const update = { description: description as string };
145 | Ledger.updateDoc(sku, update);
146 |
147 | resolve(res.status(200).send(`Wrote description to ${sku}`));
148 | });
149 | });
150 | }
151 | );
152 |
153 | app.post('/verify', async (req: Request, res: Response): Promise => {
154 | console.log(chalk.bold(`POST /verify`));
155 | return new Promise((resolve, reject) => {
156 | const form = new formidable.IncomingForm();
157 | form.parse(req, async (err: Error, fields: Fields, files: Files) => {
158 | // verify each file by opening it and sending it to check
159 | // (which involves hashing and comparing to QLDB), then building
160 | // the Response one file at a time and sending it once
161 | Promise.all(
162 | Object.keys(files).map(async i => {
163 | await fs
164 | // @ts-expect-error
165 | .readFile(files[i].path)
166 | .then(f => Verify.verifyFile(f))
167 | .then(result =>
168 | res.write(
169 | result
170 | ? JSON.stringify(result)
171 | : JSON.stringify({ not_found: 'item not found' })
172 | )
173 | );
174 | })
175 | ).then(() => resolve(res.send()));
176 | });
177 | });
178 | });
179 |
180 | export default app;
181 |
--------------------------------------------------------------------------------
/docker-compose.yml:
--------------------------------------------------------------------------------
1 | version: '3'
2 |
3 | services:
4 | endpoint:
5 | restart: always
6 | build:
7 | context: .
8 | dockerfile: ./Dockerfile
9 | ports:
10 | - '3000:3000'
11 | networks:
12 | - app
13 | volumes:
14 | - './out:/app/out'
15 |
16 | ui:
17 | build:
18 | context: ui/
19 | dockerfile: Dockerfile
20 | env_file: .env
21 | logging:
22 | driver: none
23 | ports:
24 | - '8000:80'
25 | networks:
26 | - app
27 | depends_on:
28 | - endpoint
29 |
30 | networks:
31 | app:
32 |
--------------------------------------------------------------------------------
/docs/README.md:
--------------------------------------------------------------------------------
1 | Welcome to the documentation of the Digital Evidence Preservation Toolkit, a one-click tool to archive and annotate webpages while demonstrating chain of custody throughout. The Toolkit is a proof-of-concept software for researchers and small teams sifting through online material.
2 |
3 | With only one click of the mouse, the material will be **archived in a framework demonstrating chain of custody** and **stored durably**. Once included in the growing database, users will be able to **go back to search through** and **annotate the material**, and to **export working copies** of said material for publication and dissemination.
4 |
5 | A database built thusly can be handed to a prosecutor ten years down the line, and they will be able to say with mathematical certainty: **“the material in this archive is identical and contemporary to the one saved at the time, ten years ago.”**
6 |
7 | ---
8 |
9 | ### **The flow from 30,000ft:**
10 |
11 | A **browser extension** is tasked with passing data from the user to the system.
12 | The system receives this data through HTTP requests and **records it into the ledger.
13 | A GUI of the library** is served by the system, and this can also add data to the ledger. **Annotations** can be added to the archive through the UI. **Working copies,** true to the originals, can be exported through the UI.
14 |
15 | ---
16 |
17 | ### 🤔 What is where?
18 |
19 | To install and hit the ground running, see [Getting Started](./getting_started.md).
20 |
21 | **The browser extension** is currently written in **plain JS** (as well as some HTML/CSS). The JS assets are bundled and moved in place by Webpack, which also provides auto-reloading of the extension in-browser.
22 |
23 | **The app and API** are currently written in (mostly) **Node & TypeScript**. It presently exposes REST endpoints (such as `/list-docs`, `/form`, etc.) and handles the interfacing with QLDB.
24 |
25 | An **example UI** is included and built in **Svelte**, an amazing frontend framework. It demonstrates how some of the above API endpoints can be implemented and some of the capabilities of the tool.
26 |
27 | All the above runs with `docker-compose`, as well as standalone `npm` scripts.
28 |
29 | ---
30 |
31 | ### 🥱 So, where are we ${today}?
32 |
33 | #### The API
34 |
35 | Both the browser extension and the app/API are in a functioning state, though features need to be developed in sync to be considered complete.
36 |
37 | Among other things, the browser extension is able to POST an object of the following shape to the API endpoint `/form` (ed: this name is terrible):
38 |
39 | ```tsx
40 | { url: string,
41 | title: string,
42 | scr: Base64DataURI,
43 | one_file: HTMLCodeString }
44 | ```
45 |
46 | We're including:
47 |
48 | - **a base64-encoded screenshot PNG** which, disappointingly, is only the visible part of the screen (see [`browser.tabs.captureVisibleTab`](https://developer.mozilla.org/en-US/docs/Mozilla/Add-ons/WebExtensions/API/Tabs/captureVisibleTab)).
49 | Moving to a full-page screenshot will involve some fiddling with simulating a scroll while capturing with Screen Capture API, I'm told.
50 | These screenshots can be quite large (from a few hundred kbs to a couple of mbs) so on the app/API side we account for a chunked, streamed payload. All data from the browser is grouped in a `FormData()`.
51 | - **a long string of HTML code** which contains all HTML, inlined styles and JavaScript, as well as encoded images where possible.
52 | This is most definitely not quite `.mhtml`, which apparently is not supported on Firefox anymore since Quantum. Go figure!
53 |
54 | ---
55 |
56 | A main `Record` type is defined as the central data structure flowing through the application.
57 |
58 | ```tsx
59 | type Record {
60 |
61 | // list of files preserved and hashed
62 | // type Bundle
63 | bundle: [
64 | { typ: 'screenshot' | 'one_file' | 'screenshot_thumbnail',
65 | hash: 'aaaaaaa'
66 | }, {...}
67 | ],
68 |
69 | // user-created data about the record
70 | // most probably after the original archive
71 | // type Annotations
72 | annotations: [
73 | description: 'description',
74 | other_key: 'other val'
75 | ],
76 |
77 | // data points about the page saved
78 | // type Data
79 | data: {
80 | title: 'page title',
81 | url: 'https://foo.bar.com'
82 | }
83 |
84 | }
85 | ```
86 |
87 | **Examples of this data flowing:**
88 |
89 | - Upon receiving `POST /form` , the API wrangles the payload data into this shape, which can then be passed to `Ledger.insertDoc` to be added to the ledger.
90 | (this includes side effects such as the writing to disk of screenshots and of the one-file archive)
91 | - The frontend consumes the result of `GET /list-docs`, which still fetches data from QLDB after passing it through two successive formatting functions:
92 | - `Record.fromLedger`, which takes QLDB-shaped data and builds a nice `Record` as defined above,
93 | - then `Record.toFrontend` , which takes a `Record` and builds a simplified shape for the frontend.
94 |
95 | **Central to this type definition is the _Bundle:_**
96 |
97 | ```tsx
98 | type Bundle = File.File[];
99 |
100 | type File = {
101 | kind: "screenshot" | "one_file" | "screenshot_thumbnail";
102 | hash: "xxx";
103 | };
104 | ```
105 |
106 | A Bundle is a list of files, which can only be of some kinds. At the back of our mind, these are the three kinds of files we're interested in for now:
107 |
108 | - a page screenshot,
109 | - and its thumbnail for rendering in the UI,
110 | - as well as a one-file download of all the HTML/CSS/JS assets
111 |
112 | The QLDB logic can be found under the `QLDB.*` namespace.
113 |
114 | ### The UI
115 |
116 | The webapp uses [SvelteKit](https://kit.svelte.dev), a JS framework. It implements two notables routes – the two main use stories:
117 |
118 | - The Library: `src/routes/library.svelte`
119 | - The Verification: `src/routes/verify.svelte`
120 |
121 | **Library** renders a list of ledger entries, with their accompanying metadata. It supports the querying of a record's history, as well as the addition of metadata (i.e. a "description" field).
122 |
123 | - details about how each of these features is replicated through the API
124 |
125 | **Verification** implements the lookup process and surfacing of information made possible by the tool.
126 |
127 | ---
128 |
129 | ### Miscellaneous
130 |
131 | #### On uniqueness
132 |
133 | Each record in our database contains a list of files that make it up (as of Aug 10th: a screenshot, its thumbnail, and a one-file HTML archive). Each is represented by its _kind_ and its hash (sha256).
134 |
135 | The ID of the record is the hash of the concatenated hashes of its files:
136 |
137 | `Record.id = hash(Record.files.sort().map(File.id).join(''))`
138 |
139 | With self-identifiable data, it is possible to associate files to their ledger entries, since the ID can be computed from the files.
140 |
141 | #### On ledgers
142 |
143 | "A non ledger database is table-centric. A ledger database is log-centric. **The log is the database.**" ([Ivan Moto](https://ivan.mw/2019-11-24/what-is-a-ledger-database))
144 |
145 | "Standard databases track a sequence of transactions that add, delete, or change entries. Ledger databases add a layer of digital signatures for each transaction so anyone can audit the list and see that it was constructed correctly. More importantly, no one has gone back to adjust a previous transaction — to change history, so to speak." ([VentureBeat](https://venturebeat.com/2021/01/18/database-trends-why-you-need-a-ledger-database/))
146 |
147 |
--------------------------------------------------------------------------------
/docs/getting_started.md:
--------------------------------------------------------------------------------
1 | ### Getting started
2 |
3 | If the instructions in this guide feel a bit much, it's likely because the Toolkit is still an alpha-version software which assumes a certain technical knowledge. There are technical solutions to simplifying this setup, but these were not prioritised.
4 |
5 |
9 |
10 | ---
11 |
12 | #### Setting up the ledger
13 |
14 | The Toolkit requires a working connection with Amazon Web Services, and thus that you have some kind of well-permissioned account or IAM role.
15 |
16 | In short, you will need:
17 |
18 | 1. the AWS CLI and an authorised profile in `~/.aws/credentials` (see [“Installing the AWS CLI v2”](https://docs.aws.amazon.com/cli/latest/userguide/install-cliv2.html) - _docs.aws.amazon.com)_
19 | 2. an existing QLDB ledger, with a blank table in it (see [“Creating a QLDB ledger from the AWS Console”](https://qldbguide.com/docs/guide/getting-started/#using-aws-console) - _qldbguide.com_)
20 |
21 | Not required but recommended is an S3 bucket in which to store Toolkit data.
22 |
23 | **Remember the names** of the ledger and of its table. You'll need them shortly (see "Environment" below).
24 |
25 | ---
26 |
27 | #### Environment
28 |
29 | After cloning the repository, create an `.env` file at the root or copy `.env.example`. The job of this file is to contain variables you really don't want to share publicly, so keep this out of version control software.
30 |
31 | This file **must contain:**
32 |
33 | - AWS access credentials and preferred region
34 | - Details about the ledger and S3 bucket
35 |
36 | ```bash
37 | AWS_ACCESS_KEY_ID="your aws access key"
38 | AWS_SECRET_ACCESS_KEY="your aws secret key"
39 | AWS_REGION="eu-central-1 (or another region)"
40 | BUCKET_NAME="anS3BucketName"
41 | LEDGER_NAME="yourLedgerName"
42 | DOC_TABLE_NAME="yourTableName"
43 | ```
44 |
45 | ---
46 |
47 | #### Recommended way of running the Toolkit
48 |
49 |
52 |
53 | The Docker Compose orchestration is composed of several services:
54 |
55 | 1. An Express/TypeScript API
56 | 2. A plain JS browser extension
57 | 3. And a frontend
58 |
59 | To start the whole app:
60 |
61 | ```bash
62 | $ docker-compose up
63 | ```
64 |
65 | ---
66 |
67 | #### Running without Docker
68 |
69 | Ensure you're running `node > 10.0` — the recommended version is the LTS, i.e. `node v16`. If you are using `nvm`:
70 |
71 | ```bash
72 | $ nvm use --lts
73 | > Now using node v16.13.0
74 | ```
75 |
76 | Manually install dependencies for each service:
77 |
78 | ```bash
79 | $ cd ui/ & npm install
80 | $ cd extension/ & npm install
81 | $ npm install
82 | ```
83 |
84 | Then use the npm script including all services:
85 |
86 | ```bash
87 | $ npm run all
88 | ```
89 |
90 | ---
91 |
92 | #### Storage options
93 |
94 | By including a bucket in the `.env` config, you’re choosing to replicate your archival on S3.
95 |
96 | Namely, the Store (`src/store/index.ts`) will:
97 |
98 | - upon receiving an archive request, store the Bundle files both locally and on S3,
99 | - and upon receiving a file request (e.g. the UI fetching thumbnails), serve it from S3.
100 |
101 | ---
102 |
103 | #### Is there anybody out there?
104 |
105 | ##### API and frontend
106 |
107 | The API should be available at `http://localhost:3000` — assert this by running:
108 |
109 | ```bash
110 | $ curl http://localhost:3000/list-docs
111 | > [ {...}, {...} ]
112 | ```
113 |
114 | The UI should be available at `http://localhost:8000` in your web browser of choice. API requests are proxied through the UI. Thus, the following queries are equivalent:
115 |
116 | ```bash
117 | $ curl http://localhost:3000/list-docs // as before
118 | $ curl http://localhost:8000/api/list-docs
119 | ```
120 |
121 | ##### Browser extension
122 |
123 | The extension should be being bundled on your filesystem. Pop open your browser's extension runtime by pasting this in the URL bar:
124 |
125 | `about:debugging#/runtime/this-firefox`
126 |
127 | Click _“Load temporary Add-on...”_ and navigate to `extension/addon` to select `manifest.json`.
128 |
129 | The extension should have been loaded in your extension bar, as shown below:
130 |
131 | 
132 |
--------------------------------------------------------------------------------
/extension/.gitignore:
--------------------------------------------------------------------------------
1 | addon/content_scripts/
2 | addon/popup/index.js
3 |
4 |
--------------------------------------------------------------------------------
/extension/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:14.16.1-alpine3.12
2 | LABEL version="0.1"
3 | LABEL description="Firefox extension for the DEPToolkit"
4 |
5 | WORKDIR /extension
6 | COPY ["package.json", "package-lock.json", "./"]
7 |
8 | RUN apk --no-cache add git
9 | RUN apk --no-cache --virtual build-dependencies add \
10 | g++ gcc libgcc \
11 | libstdc++ linux-headers \
12 | make python3 \
13 | && npm install \
14 | && apk del build-dependencies
15 |
16 | COPY . .
17 |
18 | EXPOSE 9000
19 |
20 | CMD ["npm", "run", "watch"]
--------------------------------------------------------------------------------
/extension/README.md:
--------------------------------------------------------------------------------
1 | # Digital Evidence Toolkit Firefox extension
2 |
3 | Browser extension used to trigger the archiving of a webpage, according to the (coming) Digital Evidence Toolkit's standard.
4 |
5 | ## Install
6 |
7 | ```sh
8 | $ npm install
9 | $ npm run watch
10 | ```
11 |
12 | Then open `http://about:debugging` in Firefox and pick `manifest.json`.
13 |
14 | The Webpack task bundles dependencies and moves `popup/popup.js` and `content_scripts/archive.js` into their proper place in `addon/`.
15 |
16 | Your changes will automatically reload.
17 |
18 | ## Usage
19 |
20 | Use while the Node API is running in parallel.
21 |
--------------------------------------------------------------------------------
/extension/addon/background.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/extension/addon/icons/LICENSE:
--------------------------------------------------------------------------------
1 |
2 | The icon "beasts-32.png" is taken from the IconBeast Lite iconset, and used under the terms of its license (http://www.iconbeast.com/faq/), with a link back to the website: http://www.iconbeast.com/free/.
3 |
4 | The icon "beasts-48.png" is taken from Aha-Soft’s Free Retina iconset, and used under the terms of its license (http://www.aha-soft.com/free-icons/free-retina-icon-set/), with a link back to the website: http://www.aha-soft.com/.
5 |
--------------------------------------------------------------------------------
/extension/addon/icons/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/digitalevidencetoolkit/deptoolkit/348a9ed840a2771e3d08b83d122acdbc3cb9667c/extension/addon/icons/logo.png
--------------------------------------------------------------------------------
/extension/addon/icons/logo_dark.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/digitalevidencetoolkit/deptoolkit/348a9ed840a2771e3d08b83d122acdbc3cb9667c/extension/addon/icons/logo_dark.png
--------------------------------------------------------------------------------
/extension/addon/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "manifest_version": 2,
3 | "name": "deptoolkit-extension",
4 | "description": "A Firefox extension to archive digital media content",
5 | "version": "1.0",
6 | "homepage_url": "https://digitalevidencetoolkit.org",
7 | "icons": {
8 | "48": "icons/logo.png"
9 | },
10 |
11 | "permissions": ["activeTab", "*://localhost/*", ""],
12 |
13 | "browser_action": {
14 | "default_icon": "icons/logo.png",
15 | "theme_icons": [
16 | {
17 | "light": "icons/logo_dark.png",
18 | "dark": "icons/logo.png",
19 | "size": 32
20 | }
21 | ],
22 | "default_title": "Archiver",
23 | "default_popup": "popup/index.html"
24 | },
25 |
26 | "content_scripts": [
27 | {
28 | "matches": [""],
29 | "run_at": "document_start",
30 | "js": [
31 | "single-file/dist/single-file-frames.js",
32 | "single-file/dist/extension-frames.js"
33 | ],
34 | "all_frames": true,
35 | "match_about_blank": true
36 | },
37 | {
38 | "matches": [""],
39 | "run_at": "document_start",
40 | "js": [
41 | "single-file/dist/single-file-bootstrap.js",
42 | "single-file/dist/extension-core.js",
43 | "single-file/dist/single-file.js"
44 | ],
45 | "all_frames": false
46 | }
47 | ],
48 | "background": {
49 | "page": "background.html",
50 | "persistent": false
51 | },
52 | "web_accessible_resources": [
53 | "single-file/dist/web/hooks/hooks-web.js",
54 | "single-file/dist/web/hooks/hooks-frames-web.js"
55 | ]
56 | }
57 |
--------------------------------------------------------------------------------
/extension/addon/popup/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
⛑ Archive current tab
12 |
📚 List documents
13 |
14 |
15 |
Can't archive this web page.
16 |
Try a different page.
17 |
18 |
19 |
20 |
21 |
--------------------------------------------------------------------------------
/extension/addon/popup/style.css:
--------------------------------------------------------------------------------
1 | :root {
2 | --text-color: rgb(94, 94, 94);
3 | --text-color-emphasis: #111;
4 | --text-color-muted: #ddd;
5 | --white: #fff;
6 | }
7 | html,
8 | body {
9 | width: 200px;
10 | font-family: sans-serif;
11 | color: var(--text-color);
12 | }
13 |
14 | .hidden {
15 | display: none;
16 | }
17 |
18 | .button {
19 | margin: 3% auto;
20 | padding: 4px;
21 | text-align: center;
22 | font-size: 1rem;
23 | cursor: pointer;
24 | border: 1px solid #ddd;
25 | background-color: #ececec;
26 |
27 | transition: background-color 0.2s, border 0.2s, color 0.3s, opacity 0.2s;
28 | }
29 |
30 | .button:hover {
31 | color: var(--text-color-emphasis);
32 | background-color: var(--white);
33 | border: 1px solid #333;
34 | text-decoration: underline;
35 | }
36 | .button.selected {
37 | color: var(--text-color-muted);
38 | background-color: var(--white);
39 | border: 1px solid var(--white);
40 | }
41 | .button.archived {
42 | background-color: var(--white);
43 | border: none;
44 | cursor: default;
45 | }
46 | .button.archived:hover {
47 | color: var(--text-color);
48 | background-color: var(--white);
49 | text-decoration: none;
50 | }
51 |
--------------------------------------------------------------------------------
/extension/addon/single-file/dist/extension-core.js:
--------------------------------------------------------------------------------
1 | !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).extension={})}(this,(function(e){"use strict";let t,r;const a=["dist/chrome-browser-polyfill.js","dist/single-file.js"],n=["dist/chrome-browser-polyfill.js","dist/single-file-frames.js"];async function s(e,s){let o;if(await async function(e){const s=e.extensionScriptFiles||[];t||r||([t,r]=await Promise.all([i(a.concat(s)),i(n)]))}(s),!s.removeFrames)try{await browser.tabs.executeScript(e,{code:r,allFrames:!0,matchAboutBlank:!0,runAt:"document_start"})}catch(e){}try{await browser.tabs.executeScript(e,{code:t,allFrames:!1,runAt:"document_idle"}),o=!0}catch(e){}return o&&s.frameId&&await browser.tabs.executeScript(e,{code:"document.documentElement.dataset.requestedFrameId = true",frameId:s.frameId,matchAboutBlank:!0,runAt:"document_start"}),o}async function i(e){const t=e.map((async e=>{if("function"==typeof e)return"("+e.toString()+")();";{const t=await fetch(browser.runtime.getURL("../../../"+e));return(new TextDecoder).decode(await t.arrayBuffer())}}));let r="";for(const e of t)r+=await e;return r}const o="single-file-response-fetch",c=window.fetch;async function f(e,t={}){try{let t=await c(e,{cache:"force-cache"});return 401!=t.status&&403!=t.status&&404!=t.status||(t=await l(e)),t}catch(r){const a=await d({method:"singlefile.fetch",url:e,referrer:t.referrer});return{status:a.status,headers:{get:e=>a.headers&&a.headers[e]},arrayBuffer:async()=>new Uint8Array(a.array).buffer}}}async function u(e,t){const r=await d({method:"singlefile.fetchFrame",url:e,frameId:t.frameId,referrer:t.referrer});return{status:r.status,headers:new Map(r.headers),arrayBuffer:async()=>new Uint8Array(r.array).buffer}}async function d(e){const t=await browser.runtime.sendMessage(e);if(!t||t.error)throw new Error(t&&t.error&&t.error.toString());return t}function l(e){return new Promise(((t,r)=>{var a,n,s,i;a=new CustomEvent("single-file-request-fetch",{detail:e}),window.dispatchEvent(a),n=o,s=function a(n){var s,i,c;n.detail?n.detail.url==e&&(s=o,i=a,c=!1,window.removeEventListener(s,i,c),n.detail.response?t({status:n.detail.status,headers:new Map(n.detail.headers),arrayBuffer:async()=>n.detail.response}):r(n.detail.error)):r()},i=!1,window.addEventListener(n,s,i)}))}browser.runtime.onMessage.addListener((e=>{if("singlefile.fetchFrame"==e.method&&window.frameId&&window.frameId==e.frameId)return async function(e){try{let t=await c(e.url,{cache:"force-cache"});return 401!=t.status&&403!=t.status&&404!=t.status||(t=await Promise.race([l(e.url),new Promise(((e,t)=>setTimeout((()=>t()),5e3)))])),{status:t.status,headers:[...t.headers],array:Array.from(new Uint8Array(await t.arrayBuffer()))}}catch(e){return{error:e&&e.toString()}}}(e)})),e.getPageData=function(e,t,r,a={fetch:f,frameFetch:u}){return globalThis.singlefile.getPageData(e,a,t,r)},e.injectScript=function(e,t){return s(e,t)},Object.defineProperty(e,"__esModule",{value:!0})}));
2 |
--------------------------------------------------------------------------------
/extension/addon/single-file/dist/extension-frames.js:
--------------------------------------------------------------------------------
1 | !function(){"use strict";const e="single-file-response-fetch",t=window.fetch;function r(t){return new Promise(((r,a)=>{var s,n,i,o;s=new CustomEvent("single-file-request-fetch",{detail:t}),window.dispatchEvent(s),n=e,i=function s(n){var i,o,d;n.detail?n.detail.url==t&&(i=e,o=s,d=!1,window.removeEventListener(i,o,d),n.detail.response?r({status:n.detail.status,headers:new Map(n.detail.headers),arrayBuffer:async()=>n.detail.response}):a(n.detail.error)):a()},o=!1,window.addEventListener(n,i,o)}))}browser.runtime.onMessage.addListener((e=>{if("singlefile.fetchFrame"==e.method&&window.frameId&&window.frameId==e.frameId)return async function(e){try{let a=await t(e.url,{cache:"force-cache"});return 401!=a.status&&403!=a.status&&404!=a.status||(a=await Promise.race([r(e.url),new Promise(((e,t)=>setTimeout((()=>t()),5e3)))])),{status:a.status,headers:[...a.headers],array:Array.from(new Uint8Array(await a.arrayBuffer()))}}catch(e){return{error:e&&e.toString()}}}(e)}))}();
2 |
--------------------------------------------------------------------------------
/extension/addon/single-file/dist/single-file-bootstrap.js:
--------------------------------------------------------------------------------
1 | !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).singlefileBootstrap={})}(this,(function(e){"use strict";const t="single-file-load-image",s="single-file-image-loaded",o=globalThis.browser,n=e=>globalThis.dispatchEvent(e),a=globalThis.CustomEvent,i=globalThis.document,r=globalThis.HTMLDocument;let l;if(l=window._singleFile_fontFaces?window._singleFile_fontFaces:window._singleFile_fontFaces=[],i instanceof r&&o&&o.runtime&&o.runtime.getURL){d="single-file-new-font-face",c=e=>{const t=e.detail;l.find((e=>JSON.stringify(e)==JSON.stringify(t)))||l.push(e.detail)},globalThis.addEventListener(d,c,m);let e=i.createElement("script");e.textContent="("+function(){"undefined"==typeof globalThis&&(window.globalThis=window);const e=globalThis.console,t=e=>globalThis.dispatchEvent(e),s=globalThis.CustomEvent,o=globalThis.FileReader,n=globalThis.Blob,a=e&&e.warn&&((...t)=>e.warn(...t))||(()=>{}),i="single-file-new-font-face",r={family:"font-family",style:"font-style",weight:"font-weight",stretch:"font-stretch",unicodeRange:"unicode-range",variant:"font-variant",featureSettings:"font-feature-settings"};if(globalThis.FontFace){const e=globalThis.FontFace;let l;globalThis.FontFace=function(){l||(a("SingleFile is hooking the FontFace constructor to get font URLs."),l=!0);const d={};d["font-family"]=arguments[0],d.src=arguments[1];const c=arguments[2];if(c&&Object.keys(c).forEach((e=>{r[e]&&(d[r[e]]=c[e])})),d.src instanceof ArrayBuffer){const e=new o;e.readAsDataURL(new n([d.src])),e.addEventListener("load",(()=>{d.src="url("+e.result+")",t(new s(i,{detail:d}))}))}else t(new s(i,{detail:d}));return new e(...arguments)},globalThis.FontFace.toString=function(){return"function FontFace() { [native code] }"}}}.toString()+")()",(i.documentElement||i).appendChild(e),e.remove(),e=i.createElement("script"),e.src=o.runtime.getURL("/single-file/dist/web/hooks/hooks-frames-web.js"),e.async=!1,(i.documentElement||i).appendChild(e),e.remove()}var d,c,m;const u=new RegExp("\\\\([\\da-f]{1,6}[\\x20\\t\\r\\n\\f]?|([\\x20\\t\\r\\n\\f])|.)","ig");const g="single-file-on-before-capture",p="single-file-on-after-capture",f="data-single-file-removed-content",h="data-single-file-hidden-content",E="data-single-file-kept-content",T="data-single-file-hidden-frame",b="data-single-file-preserved-space-element",y="data-single-file-shadow-root-element",w="data-single-file-image",I="data-single-file-poster",A="data-single-file-canvas",N="data-single-file-import",v="data-single-file-input-value",S="data-single-file-lazy-loaded-src",R="data-single-file-stylesheet",F="data-single-file-disabled-noscript",C="data-single-file-async-script",_="*:not(base):not(link):not(meta):not(noscript):not(script):not(style):not(template):not(title)",x=["NOSCRIPT","DISABLED-NOSCRIPT","META","LINK","STYLE","TITLE","TEMPLATE","SOURCE","OBJECT","SCRIPT","HEAD"],P=/^'(.*?)'$/,M=/^"(.*?)"$/,O={regular:"400",normal:"400",bold:"700",bolder:"700",lighter:"100"},L="single-file-ui-element",D=(e,t,s)=>globalThis.addEventListener(e,t,s);function k(e,t,s){let o;return e.querySelectorAll("noscript:not([data-single-file-disabled-noscript])").forEach((e=>{e.setAttribute(F,e.textContent),e.textContent=""})),function(e){e.querySelectorAll("meta[http-equiv=refresh]").forEach((e=>{e.removeAttribute("http-equiv"),e.setAttribute("disabled-http-equiv","refresh")}))}(e),e.head&&e.head.querySelectorAll(_).forEach((e=>e.hidden=!0)),e.querySelectorAll("svg foreignObject").forEach((e=>{const t=e.querySelectorAll("html > head > "+_+", html > body > "+_);t.length&&(Array.from(e.childNodes).forEach((e=>e.remove())),t.forEach((t=>e.appendChild(t))))})),o=t&&e.documentElement?q(t,e,e.documentElement,s):{canvases:[],images:[],posters:[],usedFonts:[],shadowRoots:[],imports:[],markedElements:[]},{canvases:o.canvases,fonts:l,stylesheets:z(e),images:o.images,posters:o.posters,usedFonts:Array.from(o.usedFonts.values()),shadowRoots:o.shadowRoots,imports:o.imports,referrer:e.referrer,markedElements:o.markedElements}}function q(e,t,s,o,n={usedFonts:new Map,canvases:[],images:[],posters:[],shadowRoots:[],imports:[],markedElements:[]},a){return Array.from(s.childNodes).filter((t=>t instanceof e.HTMLElement||t instanceof e.SVGElement)).forEach((s=>{let i,r,l;if(!o.autoSaveExternalSave&&(o.removeHiddenElements||o.removeUnusedFonts||o.compressHTML)&&(l=e.getComputedStyle(s),s instanceof e.HTMLElement&&o.removeHiddenElements&&(r=(a||s.closest("html > head"))&&x.includes(s.tagName)||s.closest("details"),r||(i=a||function(e,t){let s=!1;if(t){const o=t.getPropertyValue("display"),n=t.getPropertyValue("opacity"),a=t.getPropertyValue("visibility");if(s="none"==o,!s&&("0"==n||"hidden"==a)&&e.getBoundingClientRect){const t=e.getBoundingClientRect();s=!t.width&&!t.height}}return Boolean(s)}(s,l),i&&(s.setAttribute(h,""),n.markedElements.push(s)))),!i)){if(o.compressHTML&&l){const e=l.getPropertyValue("white-space");e&&e.startsWith("pre")&&(s.setAttribute(b,""),n.markedElements.push(s))}o.removeUnusedFonts&&(U(l,o,n.usedFonts),U(e.getComputedStyle(s,":first-letter"),o,n.usedFonts),U(e.getComputedStyle(s,":before"),o,n.usedFonts),U(e.getComputedStyle(s,":after"),o,n.usedFonts))}!function(e,t,s,o,n,a,i){if("CANVAS"==s.tagName)try{const t=W(e,s,i);n.canvases.push({dataURI:s.toDataURL("image/png",""),width:t.width,height:t.height}),s.setAttribute(A,n.canvases.length-1),n.markedElements.push(s)}catch(e){}if("IMG"==s.tagName){const t={currentSrc:a?"":o.loadDeferredImages&&s.getAttribute(S)||s.currentSrc};if(n.images.push(t),s.setAttribute(w,n.images.length-1),n.markedElements.push(s),s.removeAttribute(S),i=i||e.getComputedStyle(s)){t.size=W(e,s,i);const o=i.getPropertyValue("box-shadow"),n=i.getPropertyValue("background-image");o&&"none"!=o||n&&"none"!=n||!(t.size.pxWidth>1||t.size.pxHeight>1)||(t.replaceable=!0,t.backgroundColor=i.getPropertyValue("background-color"),t.objectFit=i.getPropertyValue("object-fit"),t.boxSizing=i.getPropertyValue("box-sizing"),t.objectPosition=i.getPropertyValue("object-position"))}}if("VIDEO"==s.tagName&&!s.poster){const e=t.createElement("canvas"),o=e.getContext("2d");e.width=s.clientWidth,e.height=s.clientHeight;try{o.drawImage(s,0,0,e.width,e.height),n.posters.push(e.toDataURL("image/png","")),s.setAttribute(I,n.posters.length-1),n.markedElements.push(s)}catch(e){}}"IFRAME"==s.tagName&&a&&o.removeHiddenElements&&(s.setAttribute(T,""),n.markedElements.push(s));"LINK"==s.tagName&&s.import&&s.import.documentElement&&(n.imports.push({content:J(s.import)}),s.setAttribute(N,n.imports.length-1),n.markedElements.push(s));"INPUT"==s.tagName&&("password"!=s.type&&(s.setAttribute(v,s.value),n.markedElements.push(s)),"radio"!=s.type&&"checkbox"!=s.type||(s.setAttribute(v,s.checked),n.markedElements.push(s)));"TEXTAREA"==s.tagName&&(s.setAttribute(v,s.value),n.markedElements.push(s));"SELECT"==s.tagName&&s.querySelectorAll("option").forEach((e=>{e.selected&&(e.setAttribute(v,""),n.markedElements.push(e))}));"SCRIPT"==s.tagName&&(s.async&&""!=s.getAttribute("async")&&"async"!=s.getAttribute("async")&&(s.setAttribute(C,""),n.markedElements.push(s)),s.textContent=s.textContent.replace(/<\/script>/gi,"<\\/script>"))}(e,t,s,o,n,i,l);const d=!(s instanceof e.SVGElement)&&H(s);if(d&&!s.classList.contains(L)){const a={};s.setAttribute(y,n.shadowRoots.length),n.markedElements.push(s),n.shadowRoots.push(a),q(e,t,d,o,n,i),a.content=d.innerHTML,a.delegatesFocus=d.delegatesFocus,a.mode=d.mode,d.adoptedStyleSheets&&d.adoptedStyleSheets.length&&(a.adoptedStyleSheets=Array.from(d.adoptedStyleSheets).map((e=>Array.from(e.cssRules).map((e=>e.cssText)).join("\n"))))}q(e,t,s,o,n,i),!o.autoSaveExternalSave&&o.removeHiddenElements&&a&&(r||""==s.getAttribute(E)?s.parentElement&&(s.parentElement.setAttribute(E,""),n.markedElements.push(s.parentElement)):i&&(s.setAttribute(f,""),n.markedElements.push(s)))})),n}function U(e,t,s){if(e){const o=e.getPropertyValue("font-style")||"normal";e.getPropertyValue("font-family").split(",").forEach((n=>{if(n=V(n),!t.loadedFonts||t.loadedFonts.find((e=>V(e.family)==n&&e.style==o))){const t=(a=e.getPropertyValue("font-weight"),O[a.toLowerCase().trim()]||a),i=e.getPropertyValue("font-variant")||"normal",r=[n,t,o,i];s.set(JSON.stringify(r),[n,t,o,i])}var a}))}}function H(e){const t=globalThis.chrome;if(e.openOrClosedShadowRoot)return e.openOrClosedShadowRoot;if(!(t&&t.dom&&t.dom.openOrClosedShadowRoot))return e.shadowRoot;try{return t.dom.openOrClosedShadowRoot(e)}catch(t){return e.shadowRoot}}function V(e=""){return function(e){e=e.match(P)?e.replace(P,"$1"):e.replace(M,"$1");return e.trim()}((t=e.trim(),t.replace(u,((e,t,s)=>{const o="0x"+t-65536;return o!=o||s?t:o<0?String.fromCharCode(o+65536):String.fromCharCode(o>>10|55296,1023&o|56320)})))).toLowerCase();var t}function B(e,t){if(e.querySelectorAll("[data-single-file-disabled-noscript]").forEach((e=>{e.textContent=e.getAttribute(F),e.removeAttribute(F)})),e.querySelectorAll("meta[disabled-http-equiv]").forEach((e=>{e.setAttribute("http-equiv",e.getAttribute("disabled-http-equiv")),e.removeAttribute("disabled-http-equiv")})),e.head&&e.head.querySelectorAll("*:not(base):not(link):not(meta):not(noscript):not(script):not(style):not(template):not(title)").forEach((e=>e.removeAttribute("hidden"))),!t){const s=[f,T,h,b,w,I,A,v,y,N,R,C];t=e.querySelectorAll(s.map((e=>"["+e+"]")).join(","))}t.forEach((e=>{e.removeAttribute(f),e.removeAttribute(h),e.removeAttribute(E),e.removeAttribute(T),e.removeAttribute(b),e.removeAttribute(w),e.removeAttribute(I),e.removeAttribute(A),e.removeAttribute(v),e.removeAttribute(y),e.removeAttribute(N),e.removeAttribute(R),e.removeAttribute(C)}))}function z(e){if(e){const t=[];return e.querySelectorAll("style").forEach(((s,o)=>{try{const n=e.createElement("style");n.textContent=s.textContent,e.body.appendChild(n);const a=n.sheet;n.remove(),a&&a.cssRules.length==s.sheet.cssRules.length||(s.setAttribute(R,o),t[o]=Array.from(s.sheet.cssRules).map((e=>e.cssText)).join("\n"))}catch(e){}})),t}}function W(e,t,s){let o=t.naturalWidth,n=t.naturalHeight;if(!o&&!n){let a,i,r,l,d,c,m,u,g=!1;if("content-box"==(s=s||e.getComputedStyle(t)).getPropertyValue("box-sizing")){const e=t.style.getPropertyValue("box-sizing"),s=t.style.getPropertyPriority("box-sizing"),o=t.clientWidth;t.style.setProperty("box-sizing","border-box","important"),g=t.clientWidth!=o,e?t.style.setProperty("box-sizing",e,s):t.style.removeProperty("box-sizing")}a=j("padding-left",s),i=j("padding-right",s),r=j("padding-top",s),l=j("padding-bottom",s),g?(d=j("border-left-width",s),c=j("border-right-width",s),m=j("border-top-width",s),u=j("border-bottom-width",s)):d=c=m=u=0,o=Math.max(0,t.clientWidth-a-i-d-c),n=Math.max(0,t.clientHeight-r-l-m-u)}return{pxWidth:o,pxHeight:n}}function j(e,t){if(t.getPropertyValue(e).endsWith("px"))return parseFloat(t.getPropertyValue(e))}function J(e){const t=e.doctype;let s="";return t&&(s=" "),s+e.documentElement.outerHTML}const G=S,Y=L,K="attributes",$=globalThis.browser,Q=globalThis.document,X=globalThis.MutationObserver,Z=(e,t,s)=>globalThis.addEventListener(e,t,s),ee=(e,t,s)=>globalThis.removeEventListener(e,t,s),te=new Map;async function se(e){if(Q.documentElement){te.clear();const o=Math.max(Q.documentElement.scrollHeight-1.5*Q.documentElement.clientHeight,0),i=Math.max(Q.documentElement.scrollWidth-1.5*Q.documentElement.clientWidth,0);if(globalThis.scrollY<=o&&globalThis.scrollX<=i)return function(e){return new Promise((async o=>{let i;const r=new Set,l=new X((async t=>{if((t=t.filter((e=>e.type==K))).length){t.filter((e=>{if("src"==e.attributeName&&(e.target.setAttribute(G,e.target.src),e.target.addEventListener("load",c)),"src"==e.attributeName||"srcset"==e.attributeName||"SOURCE"==e.target.tagName)return!e.target.classList||!e.target.classList.contains(Y)})).length&&(i=!0,await ne(l,e,g),r.size||await oe(l,e,g))}}));async function d(t){await ie("idleTimeout",(async()=>{i?(le("idleTimeout"),await d(Math.max(500,t/2))):(le("loadTimeout"),le("maxTimeout"),ae(l,e,g))}),t)}function c(e){const t=e.target;t.removeAttribute(G),t.removeEventListener("load",c)}async function m(t){i=!0,await ne(l,e,g),await oe(l,e,g),t.detail&&r.add(t.detail)}async function u(t){await ne(l,e,g),await oe(l,e,g),r.delete(t.detail),r.size||await oe(l,e,g)}function g(e){l.disconnect(),ee(t,m),ee(s,u),o(e)}await d(2*e.loadDeferredImagesMaxIdleTime),await ne(l,e,g),l.observe(Q,{subtree:!0,childList:!0,attributes:!0}),Z(t,m),Z(s,u),function(e){e.loadDeferredImagesBlockCookies&&n(new a("single-file-block-cookies-start")),e.loadDeferredImagesBlockStorage&&n(new a("single-file-block-storage-start")),e.loadDeferredImagesKeepZoomLevel?n(new a("single-file-load-deferred-images-keep-zoom-level-start")):n(new a("single-file-load-deferred-images-start"))}(e)}))}(e)}}async function oe(e,t,s){await ie("loadTimeout",(()=>ae(e,t,s)),t.loadDeferredImagesMaxIdleTime)}async function ne(e,t,s){await ie("maxTimeout",(async()=>{await le("loadTimeout"),await ae(e,t,s)}),10*t.loadDeferredImagesMaxIdleTime)}async function ae(e,t,s){await le("idleTimeout"),function(e){e.loadDeferredImagesBlockCookies&&n(new a("single-file-block-cookies-end")),e.loadDeferredImagesBlockStorage&&n(new a("single-file-block-storage-end")),e.loadDeferredImagesKeepZoomLevel?n(new a("single-file-load-deferred-images-keep-zoom-level-end")):n(new a("single-file-load-deferred-images-end"))}(t),await ie("endTimeout",(async()=>{await le("maxTimeout"),s()}),t.loadDeferredImagesMaxIdleTime/2),e.disconnect()}async function ie(e,t,s){if($&&$.runtime&&$.runtime.sendMessage){if(!te.get(e)||!te.get(e).pending){const o={callback:t,pending:!0};te.set(e,o);try{await $.runtime.sendMessage({method:"singlefile.lazyTimeout.setTimeout",type:e,delay:s})}catch(o){re(e,t,s)}o.pending=!1}}else re(e,t,s)}function re(e,t,s){const o=te.get(e);o&&globalThis.clearTimeout(o),te.set(e,t),globalThis.setTimeout(t,s)}async function le(e){if($&&$.runtime&&$.runtime.sendMessage)try{await $.runtime.sendMessage({method:"singlefile.lazyTimeout.clearTimeout",type:e})}catch(t){de(e)}else de(e)}function de(e){const t=te.get(e);te.delete(e),t&&globalThis.clearTimeout(t)}$&&$.runtime&&$.runtime.onMessage&&$.runtime.onMessage.addListener&&$.runtime.onMessage.addListener((e=>{if("singlefile.lazyTimeout.onTimeout"==e.method){const t=te.get(e.type);if(t){te.delete(e.type);try{t.callback()}catch(t){de(e.type)}}}}));const ce={ON_BEFORE_CAPTURE_EVENT_NAME:g,ON_AFTER_CAPTURE_EVENT_NAME:p,WIN_ID_ATTRIBUTE_NAME:"data-single-file-win-id",preProcessDoc:k,serialize:J,postProcessDoc:B,getShadowRoot:H},me="__frameTree__::",ue='iframe, frame, object[type="text/html"][data]',ge="singlefile.frameTree.initRequest",pe="singlefile.frameTree.ackInitRequest",fe="singlefile.frameTree.cleanupRequest",he="singlefile.frameTree.initResponse",Ee=".",Te=globalThis.window==globalThis.top,be=globalThis.browser,ye=globalThis.top,we=globalThis.MessageChannel,Ie=globalThis.document,Ae=new Map;let Ne;function ve(){return globalThis.crypto.getRandomValues(new Uint32Array(32)).join("")}async function Se(e){const t=e.sessionId,s=globalThis._singleFile_waitForUserScript;Te||(Ne=globalThis.frameId=e.windowId),Ce(Ie,e.options,Ne,t),Te||(e.options.userScriptEnabled&&s&&await s(ce.ON_BEFORE_CAPTURE_EVENT_NAME),Me({frames:[Le(Ie,globalThis,Ne,e.options)],sessionId:t,requestedFrameId:Ie.documentElement.dataset.requestedFrameId&&Ne}),e.options.userScriptEnabled&&s&&await s(ce.ON_AFTER_CAPTURE_EVENT_NAME),delete Ie.documentElement.dataset.requestedFrameId)}function Re(e){const t=e.sessionId;Pe(De(Ie),e.windowId,t)}function Fe(e){e.frames.forEach((t=>_e("responseTimeouts",e.sessionId,t.windowId)));const t=Ae.get(e.sessionId);if(t){e.requestedFrameId&&(t.requestedFrameId=e.requestedFrameId),e.frames.forEach((e=>{let s=t.frames.find((t=>e.windowId==t.windowId));s||(s={windowId:e.windowId},t.frames.push(s)),s.processed||(s.content=e.content,s.baseURI=e.baseURI,s.title=e.title,s.canvases=e.canvases,s.fonts=e.fonts,s.stylesheets=e.stylesheets,s.images=e.images,s.posters=e.posters,s.usedFonts=e.usedFonts,s.shadowRoots=e.shadowRoots,s.imports=e.imports,s.processed=e.processed)}));t.frames.filter((e=>!e.processed)).length||(t.frames=t.frames.sort(((e,t)=>t.windowId.split(Ee).length-e.windowId.split(Ee).length)),t.resolve&&(t.requestedFrameId&&t.frames.forEach((e=>{e.windowId==t.requestedFrameId&&(e.requestedFrame=!0)})),t.resolve(t.frames)))}}function Ce(e,t,s,o){const n=De(e);!function(e,t,s,o,n){const a=[];let i;Ae.get(n)?i=Ae.get(n).requestTimeouts:(i={},Ae.set(n,{requestTimeouts:i}));t.forEach(((e,t)=>{const s=o+Ee+t;e.setAttribute(ce.WIN_ID_ATTRIBUTE_NAME,s),a.push({windowId:s})})),Me({frames:a,sessionId:n,requestedFrameId:e.documentElement.dataset.requestedFrameId&&o}),t.forEach(((e,t)=>{const a=o+Ee+t;try{Oe(e.contentWindow,{method:ge,windowId:a,sessionId:n,options:s})}catch(e){}i[a]=globalThis.setTimeout((()=>Me({frames:[{windowId:a,processed:!0}],sessionId:n})),750)})),delete e.documentElement.dataset.requestedFrameId}(e,n,t,s,o),n.length&&function(e,t,s,o,n){const a=[];t.forEach(((e,t)=>{const i=o+Ee+t;let r;try{r=e.contentDocument}catch(e){}if(r)try{const t=e.contentWindow;t.stop(),_e("requestTimeouts",n,i),Ce(r,s,i,n),a.push(Le(r,t,i,s))}catch(e){a.push({windowId:i,processed:!0})}})),Me({frames:a,sessionId:n,requestedFrameId:e.documentElement.dataset.requestedFrameId&&o}),delete e.documentElement.dataset.requestedFrameId}(e,n,t,s,o)}function _e(e,t,s){const o=Ae.get(t);if(o&&o[e]){const t=o[e][s];t&&(globalThis.clearTimeout(t),delete o[e][s])}}function xe(e,t){const s=Ae.get(e);s&&s.responseTimeouts&&(s.responseTimeouts[t]=globalThis.setTimeout((()=>Me({frames:[{windowId:t,processed:!0}],sessionId:e})),1e4))}function Pe(e,t,s){e.forEach(((e,o)=>{const n=t+Ee+o;e.removeAttribute(ce.WIN_ID_ATTRIBUTE_NAME);try{Oe(e.contentWindow,{method:fe,windowId:n,sessionId:s})}catch(e){}})),e.forEach(((e,o)=>{const n=t+Ee+o;let a;try{a=e.contentDocument}catch(e){}if(a)try{Pe(De(a),n,s)}catch(e){}}))}function Me(e){e.method=he;try{ye.singlefile.processors.frameTree.initResponse(e)}catch(t){Oe(ye,e,!0)}}function Oe(e,t,s){if(e==ye&&be&&be.runtime&&be.runtime.sendMessage)be.runtime.sendMessage(t);else if(s){const s=new we;e.postMessage(me+JSON.stringify({method:t.method,sessionId:t.sessionId}),"*",[s.port2]),s.port1.postMessage(t)}else e.postMessage(me+JSON.stringify(t),"*")}function Le(e,t,s,o){const n=ce.preProcessDoc(e,t,o),a=ce.serialize(e);ce.postProcessDoc(e,n.markedElements);return{windowId:s,content:a,baseURI:e.baseURI.split("#")[0],title:e.title,canvases:n.canvases,fonts:n.fonts,stylesheets:n.stylesheets,images:n.images,posters:n.posters,usedFonts:n.usedFonts,shadowRoots:n.shadowRoots,imports:n.imports,processed:!0}}function De(e){let t=Array.from(e.querySelectorAll(ue));return e.querySelectorAll("*").forEach((e=>{const s=ce.getShadowRoot(e);s&&(t=t.concat(...s.querySelectorAll(ue)))})),t}Te&&(Ne="0",be&&be.runtime&&be.runtime.onMessage&&be.runtime.onMessage.addListener&&be.runtime.onMessage.addListener((e=>e.method==he?(Fe(e),Promise.resolve({})):e.method==pe?(_e("requestTimeouts",e.sessionId,e.windowId),xe(e.sessionId,e.windowId),Promise.resolve({})):void 0))),((e,t,s)=>{globalThis.addEventListener(e,t,s)})("message",(async e=>{if("string"==typeof e.data&&e.data.startsWith(me)){e.preventDefault(),e.stopPropagation();const t=JSON.parse(e.data.substring(me.length));if(t.method==ge)e.source&&Oe(e.source,{method:pe,windowId:t.windowId,sessionId:t.sessionId}),Te||(globalThis.stop(),t.options.loadDeferredImages&&se(t.options),await Se(t));else if(t.method==pe)_e("requestTimeouts",t.sessionId,t.windowId),xe(t.sessionId,t.windowId);else if(t.method==fe)Re(t);else if(t.method==he&&Ae.get(t.sessionId)){e.ports[0].onmessage=e=>Fe(e.data)}}}),!0);const ke=["area","base","br","col","command","embed","hr","img","input","keygen","link","meta","param","source","track","wbr"],qe=[{tagName:"head",accept:e=>!e.childNodes.length||1==e.childNodes[0].nodeType},{tagName:"body",accept:e=>!e.childNodes.length}],Ue=[{tagName:"html",accept:e=>!e||8!=e.nodeType},{tagName:"head",accept:e=>!e||8!=e.nodeType&&(3!=e.nodeType||!Be(e.textContent))},{tagName:"body",accept:e=>!e||8!=e.nodeType},{tagName:"li",accept:(e,t)=>!e&&t.parentElement&&("UL"==t.parentElement.tagName||"OL"==t.parentElement.tagName)||e&&["LI"].includes(e.tagName)},{tagName:"dt",accept:e=>!e||["DT","DD"].includes(e.tagName)},{tagName:"p",accept:e=>e&&["ADDRESS","ARTICLE","ASIDE","BLOCKQUOTE","DETAILS","DIV","DL","FIELDSET","FIGCAPTION","FIGURE","FOOTER","FORM","H1","H2","H3","H4","H5","H6","HEADER","HR","MAIN","NAV","OL","P","PRE","SECTION","TABLE","UL"].includes(e.tagName)},{tagName:"dd",accept:e=>!e||["DT","DD"].includes(e.tagName)},{tagName:"rt",accept:e=>!e||["RT","RP"].includes(e.tagName)},{tagName:"rp",accept:e=>!e||["RT","RP"].includes(e.tagName)},{tagName:"optgroup",accept:e=>!e||["OPTGROUP"].includes(e.tagName)},{tagName:"option",accept:e=>!e||["OPTION","OPTGROUP"].includes(e.tagName)},{tagName:"colgroup",accept:e=>!e||8!=e.nodeType&&(3!=e.nodeType||!Be(e.textContent))},{tagName:"caption",accept:e=>!e||8!=e.nodeType&&(3!=e.nodeType||!Be(e.textContent))},{tagName:"thead",accept:e=>!e||["TBODY","TFOOT"].includes(e.tagName)},{tagName:"tbody",accept:e=>!e||["TBODY","TFOOT"].includes(e.tagName)},{tagName:"tfoot",accept:e=>!e},{tagName:"tr",accept:e=>!e||["TR"].includes(e.tagName)},{tagName:"td",accept:e=>!e||["TD","TH"].includes(e.tagName)},{tagName:"th",accept:e=>!e||["TD","TH"].includes(e.tagName)}],He=["style","script","xmp","iframe","noembed","noframes","plaintext","noscript"];function Ve(e,t,s){return 3==e.nodeType?function(e){const t=e.parentNode;let s;t&&1==t.nodeType&&(s=t.tagName.toLowerCase());return!s||He.includes(s)?"script"==s?e.textContent.replace(/<\//gi,"<\\/").replace(/\/>/gi,"\\/>"):e.textContent:e.textContent.replace(/&/g,"&").replace(/\u00a0/g," ").replace(//g,">")}(e):8==e.nodeType?"\x3c!--"+e.textContent+"--\x3e":1==e.nodeType?function(e,t,s){const o=e.tagName.toLowerCase(),n=t&&qe.find((t=>o==t.tagName&&t.accept(e)));let a="";n&&!e.attributes.length||(a="<"+o,Array.from(e.attributes).forEach((s=>a+=function(e,t,s){const o=e.name;let n="";if(!o.match(/["'>/=]/)){let a,i=e.value;s&&"class"==o&&(i=Array.from(t.classList).map((e=>e.trim())).join(" ")),i=i.replace(/&/g,"&").replace(/\u00a0/g," "),i.includes('"')&&(i.includes("'")||!s?i=i.replace(/"/g,"""):a=!0);const r=!s||!i.match(/^[^ \t\n\f\r'"`=<>]+$/);n+=" ",e.namespace?"http://www.w3.org/XML/1998/namespace"==e.namespaceURI?n+="xml:"+o:"http://www.w3.org/2000/xmlns/"==e.namespaceURI?("xmlns"!==o&&(n+="xmlns:"),n+=o):"http://www.w3.org/1999/xlink"==e.namespaceURI?n+="xlink:"+o:n+=o:n+=o,""!=i&&(n+="=",r&&(n+=a?"'":'"'),n+=i,r&&(n+=a?"'":'"'))}return n}(s,e,t))),a+=">");"TEMPLATE"!=e.tagName||e.childNodes.length?Array.from(e.childNodes).forEach((e=>a+=Ve(e,t,s||"svg"==o))):a+=e.innerHTML;const i=t&&Ue.find((t=>o==t.tagName&&t.accept(e.nextSibling,e)));(s||!i&&!ke.includes(o))&&(a+=""+o+">");return a}(e,t,s):void 0}function Be(e){return Boolean(e.match(/^[ \t\n\f\r]/))}const ze={frameTree:Object.freeze({__proto__:null,getAsync:function(e){const t=ve();return e=JSON.parse(JSON.stringify(e)),new Promise((s=>{Ae.set(t,{frames:[],requestTimeouts:{},responseTimeouts:{},resolve:e=>{e.sessionId=t,s(e)}}),Se({windowId:Ne,sessionId:t,options:e})}))},getSync:function(e){const t=ve();e=JSON.parse(JSON.stringify(e)),Ae.set(t,{frames:[],requestTimeouts:{},responseTimeouts:{}}),function(e){const t=e.sessionId,s=globalThis._singleFile_waitForUserScript;Te||(Ne=globalThis.frameId=e.windowId);Ce(Ie,e.options,Ne,t),Te||(e.options.userScriptEnabled&&s&&s(ce.ON_BEFORE_CAPTURE_EVENT_NAME),Me({frames:[Le(Ie,globalThis,Ne,e.options)],sessionId:t,requestedFrameId:Ie.documentElement.dataset.requestedFrameId&&Ne}),e.options.userScriptEnabled&&s&&s(ce.ON_AFTER_CAPTURE_EVENT_NAME),delete Ie.documentElement.dataset.requestedFrameId)}({windowId:Ne,sessionId:t,options:e});const s=Ae.get(t).frames;return s.sessionId=t,s},cleanup:function(e){Ae.delete(e),Re({windowId:Ne,sessionId:e,options:{sessionId:e}})},initResponse:Fe,TIMEOUT_INIT_REQUEST_MESSAGE:750})},We={COMMENT_HEADER:"Page saved with SingleFile",COMMENT_HEADER_LEGACY:"Archive processed by SingleFile",ON_BEFORE_CAPTURE_EVENT_NAME:g,ON_AFTER_CAPTURE_EVENT_NAME:p,preProcessDoc:k,postProcessDoc:B,serialize:(e,t)=>function(e,t){const s=e.doctype;let o="";return s&&(o=" "),o+Ve(e.documentElement,t)}(e,t),getShadowRoot:H};D("single-file-user-script-init",(()=>globalThis._singleFile_waitForUserScript=async e=>{const t=new CustomEvent(e+"-request",{cancelable:!0}),s=new Promise((t=>D(e+"-response",t)));(e=>{globalThis.dispatchEvent(e)})(t),t.defaultPrevented&&await s})),e.helper=We,e.processors=ze,Object.defineProperty(e,"__esModule",{value:!0})}));
2 |
--------------------------------------------------------------------------------
/extension/addon/single-file/dist/single-file-frames.js:
--------------------------------------------------------------------------------
1 | !function(e){"function"==typeof define&&define.amd?define(e):e()}((function(){"use strict";const e="single-file-load-image",t="single-file-image-loaded",s=globalThis.browser,o=e=>globalThis.dispatchEvent(e),n=globalThis.CustomEvent,i=globalThis.document,a=globalThis.HTMLDocument;let r;if(r=window._singleFile_fontFaces?window._singleFile_fontFaces:window._singleFile_fontFaces=[],i instanceof a&&s&&s.runtime&&s.runtime.getURL){l="single-file-new-font-face",d=e=>{const t=e.detail;r.find((e=>JSON.stringify(e)==JSON.stringify(t)))||r.push(e.detail)},globalThis.addEventListener(l,d,c);let e=i.createElement("script");e.textContent="("+function(){"undefined"==typeof globalThis&&(window.globalThis=window);const e=globalThis.console,t=e=>globalThis.dispatchEvent(e),s=globalThis.CustomEvent,o=globalThis.FileReader,n=globalThis.Blob,i=e&&e.warn&&((...t)=>e.warn(...t))||(()=>{}),a="single-file-new-font-face",r={family:"font-family",style:"font-style",weight:"font-weight",stretch:"font-stretch",unicodeRange:"unicode-range",variant:"font-variant",featureSettings:"font-feature-settings"};if(globalThis.FontFace){const e=globalThis.FontFace;let l;globalThis.FontFace=function(){l||(i("SingleFile is hooking the FontFace constructor to get font URLs."),l=!0);const d={};d["font-family"]=arguments[0],d.src=arguments[1];const c=arguments[2];if(c&&Object.keys(c).forEach((e=>{r[e]&&(d[r[e]]=c[e])})),d.src instanceof ArrayBuffer){const e=new o;e.readAsDataURL(new n([d.src])),e.addEventListener("load",(()=>{d.src="url("+e.result+")",t(new s(a,{detail:d}))}))}else t(new s(a,{detail:d}));return new e(...arguments)},globalThis.FontFace.toString=function(){return"function FontFace() { [native code] }"}}}.toString()+")()",(i.documentElement||i).appendChild(e),e.remove(),e=i.createElement("script"),e.src=s.runtime.getURL("/single-file/dist/web/hooks/hooks-frames-web.js"),e.async=!1,(i.documentElement||i).appendChild(e),e.remove()}var l,d,c;const m=new RegExp("\\\\([\\da-f]{1,6}[\\x20\\t\\r\\n\\f]?|([\\x20\\t\\r\\n\\f])|.)","ig");const u="data-single-file-removed-content",g="data-single-file-hidden-content",f="data-single-file-kept-content",h="data-single-file-hidden-frame",p="data-single-file-preserved-space-element",b="data-single-file-shadow-root-element",w="data-single-file-image",y="data-single-file-poster",E="data-single-file-canvas",T="data-single-file-import",A="data-single-file-input-value",I="data-single-file-lazy-loaded-src",v="data-single-file-stylesheet",S="data-single-file-disabled-noscript",R="data-single-file-async-script",F="*:not(base):not(link):not(meta):not(noscript):not(script):not(style):not(template):not(title)",C=["NOSCRIPT","DISABLED-NOSCRIPT","META","LINK","STYLE","TITLE","TEMPLATE","SOURCE","OBJECT","SCRIPT","HEAD"],k=/^'(.*?)'$/,M=/^"(.*?)"$/,x={regular:"400",normal:"400",bold:"700",bolder:"700",lighter:"100"},N="single-file-ui-element";function q(e,t,s,o,n={usedFonts:new Map,canvases:[],images:[],posters:[],shadowRoots:[],imports:[],markedElements:[]},i){return Array.from(s.childNodes).filter((t=>t instanceof e.HTMLElement||t instanceof e.SVGElement)).forEach((s=>{let a,r,l;if(!o.autoSaveExternalSave&&(o.removeHiddenElements||o.removeUnusedFonts||o.compressHTML)&&(l=e.getComputedStyle(s),s instanceof e.HTMLElement&&o.removeHiddenElements&&(r=(i||s.closest("html > head"))&&C.includes(s.tagName)||s.closest("details"),r||(a=i||function(e,t){let s=!1;if(t){const o=t.getPropertyValue("display"),n=t.getPropertyValue("opacity"),i=t.getPropertyValue("visibility");if(s="none"==o,!s&&("0"==n||"hidden"==i)&&e.getBoundingClientRect){const t=e.getBoundingClientRect();s=!t.width&&!t.height}}return Boolean(s)}(s,l),a&&(s.setAttribute(g,""),n.markedElements.push(s)))),!a)){if(o.compressHTML&&l){const e=l.getPropertyValue("white-space");e&&e.startsWith("pre")&&(s.setAttribute(p,""),n.markedElements.push(s))}o.removeUnusedFonts&&(P(l,o,n.usedFonts),P(e.getComputedStyle(s,":first-letter"),o,n.usedFonts),P(e.getComputedStyle(s,":before"),o,n.usedFonts),P(e.getComputedStyle(s,":after"),o,n.usedFonts))}!function(e,t,s,o,n,i,a){if("CANVAS"==s.tagName)try{const t=O(e,s,a);n.canvases.push({dataURI:s.toDataURL("image/png",""),width:t.width,height:t.height}),s.setAttribute(E,n.canvases.length-1),n.markedElements.push(s)}catch(e){}if("IMG"==s.tagName){const t={currentSrc:i?"":o.loadDeferredImages&&s.getAttribute(I)||s.currentSrc};if(n.images.push(t),s.setAttribute(w,n.images.length-1),n.markedElements.push(s),s.removeAttribute(I),a=a||e.getComputedStyle(s)){t.size=O(e,s,a);const o=a.getPropertyValue("box-shadow"),n=a.getPropertyValue("background-image");o&&"none"!=o||n&&"none"!=n||!(t.size.pxWidth>1||t.size.pxHeight>1)||(t.replaceable=!0,t.backgroundColor=a.getPropertyValue("background-color"),t.objectFit=a.getPropertyValue("object-fit"),t.boxSizing=a.getPropertyValue("box-sizing"),t.objectPosition=a.getPropertyValue("object-position"))}}if("VIDEO"==s.tagName&&!s.poster){const e=t.createElement("canvas"),o=e.getContext("2d");e.width=s.clientWidth,e.height=s.clientHeight;try{o.drawImage(s,0,0,e.width,e.height),n.posters.push(e.toDataURL("image/png","")),s.setAttribute(y,n.posters.length-1),n.markedElements.push(s)}catch(e){}}"IFRAME"==s.tagName&&i&&o.removeHiddenElements&&(s.setAttribute(h,""),n.markedElements.push(s));"LINK"==s.tagName&&s.import&&s.import.documentElement&&(n.imports.push({content:U(s.import)}),s.setAttribute(T,n.imports.length-1),n.markedElements.push(s));"INPUT"==s.tagName&&("password"!=s.type&&(s.setAttribute(A,s.value),n.markedElements.push(s)),"radio"!=s.type&&"checkbox"!=s.type||(s.setAttribute(A,s.checked),n.markedElements.push(s)));"TEXTAREA"==s.tagName&&(s.setAttribute(A,s.value),n.markedElements.push(s));"SELECT"==s.tagName&&s.querySelectorAll("option").forEach((e=>{e.selected&&(e.setAttribute(A,""),n.markedElements.push(e))}));"SCRIPT"==s.tagName&&(s.async&&""!=s.getAttribute("async")&&"async"!=s.getAttribute("async")&&(s.setAttribute(R,""),n.markedElements.push(s)),s.textContent=s.textContent.replace(/<\/script>/gi,"<\\/script>"))}(e,t,s,o,n,a,l);const d=!(s instanceof e.SVGElement)&&L(s);if(d&&!s.classList.contains(N)){const i={};s.setAttribute(b,n.shadowRoots.length),n.markedElements.push(s),n.shadowRoots.push(i),q(e,t,d,o,n,a),i.content=d.innerHTML,i.delegatesFocus=d.delegatesFocus,i.mode=d.mode,d.adoptedStyleSheets&&d.adoptedStyleSheets.length&&(i.adoptedStyleSheets=Array.from(d.adoptedStyleSheets).map((e=>Array.from(e.cssRules).map((e=>e.cssText)).join("\n"))))}q(e,t,s,o,n,a),!o.autoSaveExternalSave&&o.removeHiddenElements&&i&&(r||""==s.getAttribute(f)?s.parentElement&&(s.parentElement.setAttribute(f,""),n.markedElements.push(s.parentElement)):a&&(s.setAttribute(u,""),n.markedElements.push(s)))})),n}function P(e,t,s){if(e){const o=e.getPropertyValue("font-style")||"normal";e.getPropertyValue("font-family").split(",").forEach((n=>{if(n=_(n),!t.loadedFonts||t.loadedFonts.find((e=>_(e.family)==n&&e.style==o))){const t=(i=e.getPropertyValue("font-weight"),x[i.toLowerCase().trim()]||i),a=e.getPropertyValue("font-variant")||"normal",r=[n,t,o,a];s.set(JSON.stringify(r),[n,t,o,a])}var i}))}}function L(e){const t=globalThis.chrome;if(e.openOrClosedShadowRoot)return e.openOrClosedShadowRoot;if(!(t&&t.dom&&t.dom.openOrClosedShadowRoot))return e.shadowRoot;try{return t.dom.openOrClosedShadowRoot(e)}catch(t){return e.shadowRoot}}function _(e=""){return function(e){e=e.match(k)?e.replace(k,"$1"):e.replace(M,"$1");return e.trim()}((t=e.trim(),t.replace(m,((e,t,s)=>{const o="0x"+t-65536;return o!=o||s?t:o<0?String.fromCharCode(o+65536):String.fromCharCode(o>>10|55296,1023&o|56320)})))).toLowerCase();var t}function D(e){if(e){const t=[];return e.querySelectorAll("style").forEach(((s,o)=>{try{const n=e.createElement("style");n.textContent=s.textContent,e.body.appendChild(n);const i=n.sheet;n.remove(),i&&i.cssRules.length==s.sheet.cssRules.length||(s.setAttribute(v,o),t[o]=Array.from(s.sheet.cssRules).map((e=>e.cssText)).join("\n"))}catch(e){}})),t}}function O(e,t,s){let o=t.naturalWidth,n=t.naturalHeight;if(!o&&!n){let i,a,r,l,d,c,m,u,g=!1;if("content-box"==(s=s||e.getComputedStyle(t)).getPropertyValue("box-sizing")){const e=t.style.getPropertyValue("box-sizing"),s=t.style.getPropertyPriority("box-sizing"),o=t.clientWidth;t.style.setProperty("box-sizing","border-box","important"),g=t.clientWidth!=o,e?t.style.setProperty("box-sizing",e,s):t.style.removeProperty("box-sizing")}i=V("padding-left",s),a=V("padding-right",s),r=V("padding-top",s),l=V("padding-bottom",s),g?(d=V("border-left-width",s),c=V("border-right-width",s),m=V("border-top-width",s),u=V("border-bottom-width",s)):d=c=m=u=0,o=Math.max(0,t.clientWidth-i-a-d-c),n=Math.max(0,t.clientHeight-r-l-m-u)}return{pxWidth:o,pxHeight:n}}function V(e,t){if(t.getPropertyValue(e).endsWith("px"))return parseFloat(t.getPropertyValue(e))}function U(e){const t=e.doctype;let s="";return t&&(s=" "),s+e.documentElement.outerHTML}const z=I,B=N,H="attributes",W=globalThis.browser,j=globalThis.document,J=globalThis.MutationObserver,K=(e,t,s)=>globalThis.addEventListener(e,t,s),G=(e,t,s)=>globalThis.removeEventListener(e,t,s),Y=new Map;async function $(s){if(j.documentElement){Y.clear();const i=Math.max(j.documentElement.scrollHeight-1.5*j.documentElement.clientHeight,0),a=Math.max(j.documentElement.scrollWidth-1.5*j.documentElement.clientWidth,0);if(globalThis.scrollY<=i&&globalThis.scrollX<=a)return function(s){return new Promise((async i=>{let a;const r=new Set,l=new J((async e=>{if((e=e.filter((e=>e.type==H))).length){e.filter((e=>{if("src"==e.attributeName&&(e.target.setAttribute(z,e.target.src),e.target.addEventListener("load",c)),"src"==e.attributeName||"srcset"==e.attributeName||"SOURCE"==e.target.tagName)return!e.target.classList||!e.target.classList.contains(B)})).length&&(a=!0,await Z(l,s,g),r.size||await X(l,s,g))}}));async function d(e){await ee("idleTimeout",(async()=>{a?(se("idleTimeout"),await d(Math.max(500,e/2))):(se("loadTimeout"),se("maxTimeout"),Q(l,s,g))}),e)}function c(e){const t=e.target;t.removeAttribute(z),t.removeEventListener("load",c)}async function m(e){a=!0,await Z(l,s,g),await X(l,s,g),e.detail&&r.add(e.detail)}async function u(e){await Z(l,s,g),await X(l,s,g),r.delete(e.detail),r.size||await X(l,s,g)}function g(s){l.disconnect(),G(e,m),G(t,u),i(s)}await d(2*s.loadDeferredImagesMaxIdleTime),await Z(l,s,g),l.observe(j,{subtree:!0,childList:!0,attributes:!0}),K(e,m),K(t,u),function(e){e.loadDeferredImagesBlockCookies&&o(new n("single-file-block-cookies-start")),e.loadDeferredImagesBlockStorage&&o(new n("single-file-block-storage-start")),e.loadDeferredImagesKeepZoomLevel?o(new n("single-file-load-deferred-images-keep-zoom-level-start")):o(new n("single-file-load-deferred-images-start"))}(s)}))}(s)}}async function X(e,t,s){await ee("loadTimeout",(()=>Q(e,t,s)),t.loadDeferredImagesMaxIdleTime)}async function Z(e,t,s){await ee("maxTimeout",(async()=>{await se("loadTimeout"),await Q(e,t,s)}),10*t.loadDeferredImagesMaxIdleTime)}async function Q(e,t,s){await se("idleTimeout"),function(e){e.loadDeferredImagesBlockCookies&&o(new n("single-file-block-cookies-end")),e.loadDeferredImagesBlockStorage&&o(new n("single-file-block-storage-end")),e.loadDeferredImagesKeepZoomLevel?o(new n("single-file-load-deferred-images-keep-zoom-level-end")):o(new n("single-file-load-deferred-images-end"))}(t),await ee("endTimeout",(async()=>{await se("maxTimeout"),s()}),t.loadDeferredImagesMaxIdleTime/2),e.disconnect()}async function ee(e,t,s){if(W&&W.runtime&&W.runtime.sendMessage){if(!Y.get(e)||!Y.get(e).pending){const o={callback:t,pending:!0};Y.set(e,o);try{await W.runtime.sendMessage({method:"singlefile.lazyTimeout.setTimeout",type:e,delay:s})}catch(o){te(e,t,s)}o.pending=!1}}else te(e,t,s)}function te(e,t,s){const o=Y.get(e);o&&globalThis.clearTimeout(o),Y.set(e,t),globalThis.setTimeout(t,s)}async function se(e){if(W&&W.runtime&&W.runtime.sendMessage)try{await W.runtime.sendMessage({method:"singlefile.lazyTimeout.clearTimeout",type:e})}catch(t){oe(e)}else oe(e)}function oe(e){const t=Y.get(e);Y.delete(e),t&&globalThis.clearTimeout(t)}W&&W.runtime&&W.runtime.onMessage&&W.runtime.onMessage.addListener&&W.runtime.onMessage.addListener((e=>{if("singlefile.lazyTimeout.onTimeout"==e.method){const t=Y.get(e.type);if(t){Y.delete(e.type);try{t.callback()}catch(t){oe(e.type)}}}}));const ne={ON_BEFORE_CAPTURE_EVENT_NAME:"single-file-on-before-capture",ON_AFTER_CAPTURE_EVENT_NAME:"single-file-on-after-capture",WIN_ID_ATTRIBUTE_NAME:"data-single-file-win-id",preProcessDoc:function(e,t,s){let o;return e.querySelectorAll("noscript:not([data-single-file-disabled-noscript])").forEach((e=>{e.setAttribute(S,e.textContent),e.textContent=""})),function(e){e.querySelectorAll("meta[http-equiv=refresh]").forEach((e=>{e.removeAttribute("http-equiv"),e.setAttribute("disabled-http-equiv","refresh")}))}(e),e.head&&e.head.querySelectorAll(F).forEach((e=>e.hidden=!0)),e.querySelectorAll("svg foreignObject").forEach((e=>{const t=e.querySelectorAll("html > head > "+F+", html > body > "+F);t.length&&(Array.from(e.childNodes).forEach((e=>e.remove())),t.forEach((t=>e.appendChild(t))))})),o=t&&e.documentElement?q(t,e,e.documentElement,s):{canvases:[],images:[],posters:[],usedFonts:[],shadowRoots:[],imports:[],markedElements:[]},{canvases:o.canvases,fonts:r,stylesheets:D(e),images:o.images,posters:o.posters,usedFonts:Array.from(o.usedFonts.values()),shadowRoots:o.shadowRoots,imports:o.imports,referrer:e.referrer,markedElements:o.markedElements}},serialize:U,postProcessDoc:function(e,t){if(e.querySelectorAll("[data-single-file-disabled-noscript]").forEach((e=>{e.textContent=e.getAttribute(S),e.removeAttribute(S)})),e.querySelectorAll("meta[disabled-http-equiv]").forEach((e=>{e.setAttribute("http-equiv",e.getAttribute("disabled-http-equiv")),e.removeAttribute("disabled-http-equiv")})),e.head&&e.head.querySelectorAll("*:not(base):not(link):not(meta):not(noscript):not(script):not(style):not(template):not(title)").forEach((e=>e.removeAttribute("hidden"))),!t){const s=[u,h,g,p,w,y,E,A,b,T,v,R];t=e.querySelectorAll(s.map((e=>"["+e+"]")).join(","))}t.forEach((e=>{e.removeAttribute(u),e.removeAttribute(g),e.removeAttribute(f),e.removeAttribute(h),e.removeAttribute(p),e.removeAttribute(w),e.removeAttribute(y),e.removeAttribute(E),e.removeAttribute(A),e.removeAttribute(b),e.removeAttribute(T),e.removeAttribute(v),e.removeAttribute(R)}))},getShadowRoot:L},ie="__frameTree__::",ae='iframe, frame, object[type="text/html"][data]',re="singlefile.frameTree.initRequest",le="singlefile.frameTree.ackInitRequest",de="singlefile.frameTree.cleanupRequest",ce="singlefile.frameTree.initResponse",me=".",ue=globalThis.window==globalThis.top,ge=globalThis.browser,fe=globalThis.top,he=globalThis.MessageChannel,pe=globalThis.document,be=new Map;let we;function ye(e){e.frames.forEach((t=>Te("responseTimeouts",e.sessionId,t.windowId)));const t=be.get(e.sessionId);if(t){e.requestedFrameId&&(t.requestedFrameId=e.requestedFrameId),e.frames.forEach((e=>{let s=t.frames.find((t=>e.windowId==t.windowId));s||(s={windowId:e.windowId},t.frames.push(s)),s.processed||(s.content=e.content,s.baseURI=e.baseURI,s.title=e.title,s.canvases=e.canvases,s.fonts=e.fonts,s.stylesheets=e.stylesheets,s.images=e.images,s.posters=e.posters,s.usedFonts=e.usedFonts,s.shadowRoots=e.shadowRoots,s.imports=e.imports,s.processed=e.processed)}));t.frames.filter((e=>!e.processed)).length||(t.frames=t.frames.sort(((e,t)=>t.windowId.split(me).length-e.windowId.split(me).length)),t.resolve&&(t.requestedFrameId&&t.frames.forEach((e=>{e.windowId==t.requestedFrameId&&(e.requestedFrame=!0)})),t.resolve(t.frames)))}}function Ee(e,t,s,o){const n=Fe(e);!function(e,t,s,o,n){const i=[];let a;be.get(n)?a=be.get(n).requestTimeouts:(a={},be.set(n,{requestTimeouts:a}));t.forEach(((e,t)=>{const s=o+me+t;e.setAttribute(ne.WIN_ID_ATTRIBUTE_NAME,s),i.push({windowId:s})})),ve({frames:i,sessionId:n,requestedFrameId:e.documentElement.dataset.requestedFrameId&&o}),t.forEach(((e,t)=>{const i=o+me+t;try{Se(e.contentWindow,{method:re,windowId:i,sessionId:n,options:s})}catch(e){}a[i]=globalThis.setTimeout((()=>ve({frames:[{windowId:i,processed:!0}],sessionId:n})),750)})),delete e.documentElement.dataset.requestedFrameId}(e,n,t,s,o),n.length&&function(e,t,s,o,n){const i=[];t.forEach(((e,t)=>{const a=o+me+t;let r;try{r=e.contentDocument}catch(e){}if(r)try{const t=e.contentWindow;t.stop(),Te("requestTimeouts",n,a),Ee(r,s,a,n),i.push(Re(r,t,a,s))}catch(e){i.push({windowId:a,processed:!0})}})),ve({frames:i,sessionId:n,requestedFrameId:e.documentElement.dataset.requestedFrameId&&o}),delete e.documentElement.dataset.requestedFrameId}(e,n,t,s,o)}function Te(e,t,s){const o=be.get(t);if(o&&o[e]){const t=o[e][s];t&&(globalThis.clearTimeout(t),delete o[e][s])}}function Ae(e,t){const s=be.get(e);s&&s.responseTimeouts&&(s.responseTimeouts[t]=globalThis.setTimeout((()=>ve({frames:[{windowId:t,processed:!0}],sessionId:e})),1e4))}function Ie(e,t,s){e.forEach(((e,o)=>{const n=t+me+o;e.removeAttribute(ne.WIN_ID_ATTRIBUTE_NAME);try{Se(e.contentWindow,{method:de,windowId:n,sessionId:s})}catch(e){}})),e.forEach(((e,o)=>{const n=t+me+o;let i;try{i=e.contentDocument}catch(e){}if(i)try{Ie(Fe(i),n,s)}catch(e){}}))}function ve(e){e.method=ce;try{fe.singlefile.processors.frameTree.initResponse(e)}catch(t){Se(fe,e,!0)}}function Se(e,t,s){if(e==fe&&ge&&ge.runtime&&ge.runtime.sendMessage)ge.runtime.sendMessage(t);else if(s){const s=new he;e.postMessage(ie+JSON.stringify({method:t.method,sessionId:t.sessionId}),"*",[s.port2]),s.port1.postMessage(t)}else e.postMessage(ie+JSON.stringify(t),"*")}function Re(e,t,s,o){const n=ne.preProcessDoc(e,t,o),i=ne.serialize(e);ne.postProcessDoc(e,n.markedElements);return{windowId:s,content:i,baseURI:e.baseURI.split("#")[0],title:e.title,canvases:n.canvases,fonts:n.fonts,stylesheets:n.stylesheets,images:n.images,posters:n.posters,usedFonts:n.usedFonts,shadowRoots:n.shadowRoots,imports:n.imports,processed:!0}}function Fe(e){let t=Array.from(e.querySelectorAll(ae));return e.querySelectorAll("*").forEach((e=>{const s=ne.getShadowRoot(e);s&&(t=t.concat(...s.querySelectorAll(ae)))})),t}ue&&(we="0",ge&&ge.runtime&&ge.runtime.onMessage&&ge.runtime.onMessage.addListener&&ge.runtime.onMessage.addListener((e=>e.method==ce?(ye(e),Promise.resolve({})):e.method==le?(Te("requestTimeouts",e.sessionId,e.windowId),Ae(e.sessionId,e.windowId),Promise.resolve({})):void 0))),((e,t,s)=>{globalThis.addEventListener(e,t,s)})("message",(async e=>{if("string"==typeof e.data&&e.data.startsWith(ie)){e.preventDefault(),e.stopPropagation();const t=JSON.parse(e.data.substring(ie.length));if(t.method==re)e.source&&Se(e.source,{method:le,windowId:t.windowId,sessionId:t.sessionId}),ue||(globalThis.stop(),t.options.loadDeferredImages&&$(t.options),await async function(e){const t=e.sessionId,s=globalThis._singleFile_waitForUserScript;ue||(we=globalThis.frameId=e.windowId);Ee(pe,e.options,we,t),ue||(e.options.userScriptEnabled&&s&&await s(ne.ON_BEFORE_CAPTURE_EVENT_NAME),ve({frames:[Re(pe,globalThis,we,e.options)],sessionId:t,requestedFrameId:pe.documentElement.dataset.requestedFrameId&&we}),e.options.userScriptEnabled&&s&&await s(ne.ON_AFTER_CAPTURE_EVENT_NAME),delete pe.documentElement.dataset.requestedFrameId)}(t));else if(t.method==le)Te("requestTimeouts",t.sessionId,t.windowId),Ae(t.sessionId,t.windowId);else if(t.method==de)!function(e){const t=e.sessionId;Ie(Fe(pe),e.windowId,t)}(t);else if(t.method==ce&&be.get(t.sessionId)){e.ports[0].onmessage=e=>ye(e.data)}}}),!0)}));
2 |
--------------------------------------------------------------------------------
/extension/addon/single-file/dist/web/hooks/hooks-frames-web.js:
--------------------------------------------------------------------------------
1 | !function(){"use strict";(e=>{const t="single-file-lazy-load",n="single-file-load-image",i="single-file-image-loaded",o="single-file-new-font-face",r={family:"font-family",style:"font-style",weight:"font-weight",stretch:"font-stretch",unicodeRange:"unicode-range",variant:"font-variant",featureSettings:"font-feature-settings"},l=(t,n,i)=>e.addEventListener(t,n,i),s=t=>e.dispatchEvent(t),c=e.CustomEvent,d=e.document,g=e.screen,a=e.Element,m=e.UIEvent,_=e.FileReader,f=e.Blob,u=e.console,y=u&&u.warn&&((...e)=>u.warn(...e))||(()=>{}),h=new Map,p=new Map;function F(o){const r=d.scrollingElement||d.documentElement,l=r.clientHeight,_=r.clientWidth,f=Math.max(r.scrollHeight-l,l),u=Math.max(r.scrollWidth-_,_);if(d.querySelectorAll("[loading=lazy]").forEach((e=>{e.loading="eager",e.setAttribute(t,"")})),r.__defineGetter__("clientHeight",(()=>f)),r.__defineGetter__("clientWidth",(()=>u)),g.__defineGetter__("height",(()=>f)),g.__defineGetter__("width",(()=>u)),e._singleFile_innerHeight=e.innerHeight,e._singleFile_innerWidth=e.innerWidth,e.__defineGetter__("innerHeight",(()=>f)),e.__defineGetter__("innerWidth",(()=>u)),o||e._singleFile_getBoundingClientRect||(e._singleFile_getBoundingClientRect=a.prototype.getBoundingClientRect,a.prototype.getBoundingClientRect=function(){const t=e._singleFile_getBoundingClientRect.call(this);return this==r&&(t.__defineGetter__("height",(()=>f)),t.__defineGetter__("bottom",(()=>f+t.top)),t.__defineGetter__("width",(()=>u)),t.__defineGetter__("right",(()=>u+t.left))),t}),!e._singleFileImage){const t=e.Image;e._singleFileImage=e.Image,e.__defineGetter__("Image",(function(){return function(){const e=new t(...arguments),o=new t(...arguments);return o.__defineSetter__("src",(function(t){e.src=t,s(new c(n,{detail:e.src}))})),o.__defineGetter__("src",(function(){return e.src})),o.__defineSetter__("srcset",(function(t){s(new c(n)),e.srcset=t})),o.__defineGetter__("srcset",(function(){return e.srcset})),e.onload=e.onloadend=e.onerror=t=>{s(new c(i,{detail:e.src})),o.dispatchEvent(new m(t.type,t))},e.decode&&(o.decode=()=>e.decode()),o}}))}let y,F;o?(y=l/f,F=_/u):(y=(l+e.scrollY)/f,F=(_+e.scrollX)/u);const E=Math.min(y,F);if(E<1){const e=d.documentElement.style.getPropertyValue("transform"),t=d.documentElement.style.getPropertyPriority("transform"),n=d.documentElement.style.getPropertyValue("transform-origin"),i=d.documentElement.style.getPropertyPriority("transform-origin"),r=d.documentElement.style.getPropertyValue("min-height"),l=d.documentElement.style.getPropertyPriority("min-height");d.documentElement.style.setProperty("transform-origin",(y<1?"50%":"0")+" "+(F<1?"50%":"0")+" 0","important"),d.documentElement.style.setProperty("transform","scale3d("+E+", "+E+", 1)","important"),d.documentElement.style.setProperty("min-height",100/E+"vh","important"),P(),o?(d.documentElement.style.setProperty("-sf-transform",e,t),d.documentElement.style.setProperty("-sf-transform-origin",n,i),d.documentElement.style.setProperty("-sf-min-height",r,l)):(d.documentElement.style.setProperty("transform",e,t),d.documentElement.style.setProperty("transform-origin",n,i),d.documentElement.style.setProperty("min-height",r,l))}if(!o){P();const e=r.getBoundingClientRect();[...h].forEach((([t,n])=>{const i=n.options&&n.options.root&&n.options.root.getBoundingClientRect,o=i&&n.options.root.getBoundingClientRect(),r=p.get(t);r&&n.callback(r.map((t=>{const n=t.getBoundingClientRect();return{target:t,intersectionRatio:1,boundingClientRect:n,intersectionRect:n,isIntersecting:!0,rootBounds:i?o:e,time:0}})),t)}))}}function E(n){d.querySelectorAll("[single-file-lazy-load]").forEach((e=>{e.loading="lazy",e.removeAttribute(t)})),n||e._singleFile_getBoundingClientRect&&(a.prototype.getBoundingClientRect=e._singleFile_getBoundingClientRect,delete e._singleFile_getBoundingClientRect),e._singleFileImage&&(delete e.Image,e.Image=e._singleFileImage,delete e._singleFileImage),n||P()}function b(){const t=d.scrollingElement||d.documentElement;null!=e._singleFile_innerHeight&&(e.innerHeight=e._singleFile_innerHeight,delete e._singleFile_innerHeight),null!=e._singleFile_innerWidth&&(e.innerWidth=e._singleFile_innerWidth,delete e._singleFile_innerWidth),delete t.clientHeight,delete t.clientWidth,delete g.height,delete g.width}if(l("single-file-load-deferred-images-start",(()=>F())),l("single-file-load-deferred-images-keep-zoom-level-start",(()=>F(!0))),l("single-file-load-deferred-images-end",(()=>E())),l("single-file-load-deferred-images-keep-zoom-level-end",(()=>E(!0))),l("single-file-load-deferred-images-reset",b),l("single-file-load-deferred-images-keep-zoom-level-reset",(()=>{const e=d.documentElement.style.getPropertyValue("-sf-transform"),t=d.documentElement.style.getPropertyPriority("-sf-transform"),n=d.documentElement.style.getPropertyValue("-sf-transform-origin"),i=d.documentElement.style.getPropertyPriority("-sf-transform-origin"),o=d.documentElement.style.getPropertyValue("-sf-min-height"),r=d.documentElement.style.getPropertyPriority("-sf-min-height");d.documentElement.style.setProperty("transform",e,t),d.documentElement.style.setProperty("transform-origin",n,i),d.documentElement.style.setProperty("min-height",o,r),d.documentElement.style.removeProperty("-sf-transform"),d.documentElement.style.removeProperty("-sf-transform-origin"),d.documentElement.style.removeProperty("-sf-min-height"),b()})),l("single-file-block-cookies-start",(()=>{try{d.__defineGetter__("cookie",(()=>{throw new Error("document.cookie temporary blocked by SingleFile")}))}catch(e){}})),l("single-file-block-cookies-end",(()=>{delete d.cookie})),l("single-file-block-storage-start",(()=>{e._singleFile_localStorage||(e._singleFile_localStorage=e.localStorage,e.__defineGetter__("localStorage",(()=>{throw new Error("localStorage temporary blocked by SingleFile")}))),e._singleFile_indexedDB||(e._singleFile_indexedDB=e.indexedDB,e.__defineGetter__("indexedDB",(()=>{throw new Error("indexedDB temporary blocked by SingleFile")})))})),l("single-file-block-storage-end",(()=>{e._singleFile_localStorage&&(delete e.localStorage,e.localStorage=e._singleFile_localStorage,delete e._singleFile_localStorage),e._singleFile_indexedDB||(delete e.indexedDB,e.indexedDB=e._singleFile_indexedDB,delete e._singleFile_indexedDB)})),e.FontFace){const t=e.FontFace;let n;e.FontFace=function(){n||(y("SingleFile is hooking the FontFace constructor to get font URLs."),n=!0);const e={};e["font-family"]=arguments[0],e.src=arguments[1];const i=arguments[2];if(i&&Object.keys(i).forEach((t=>{r[t]&&(e[r[t]]=i[t])})),e.src instanceof ArrayBuffer){const t=new _;t.readAsDataURL(new f([e.src])),t.addEventListener("load",(()=>{e.src="url("+t.result+")",s(new c(o,{detail:e}))}))}else s(new c(o,{detail:e}));return new t(...arguments)},e.FontFace.toString=function(){return"function FontFace() { [native code] }"}}if(e.IntersectionObserver){const t=e.IntersectionObserver;let n;e.IntersectionObserver=function(){n||(y("SingleFile is hooking the IntersectionObserver API to detect and load deferred images."),n=!0);const e=new t(...arguments),i=t.prototype.observe||e.observe,o=t.prototype.unobserve||e.unobserve,r=arguments[0],l=arguments[1];return i&&(e.observe=function(t){let n=p.get(e);return n||(n=[],p.set(e,n)),n.push(t),i.call(e,t)}),o&&(e.unobserve=function(t){let n=p.get(e);return n&&(n=n.filter((e=>e!=t)),n.length?p.set(e,n):(p.delete(e),h.delete(e))),o.call(e,t)}),h.set(e,{callback:r,options:l}),e},e.IntersectionObserver.prototype=t.prototype,e.IntersectionObserver.toString=function(){return"function IntersectionObserver() { [native code] }"}}function P(){try{s(new m("resize"))}catch(e){}}})("object"==typeof globalThis?globalThis:window)}();
2 |
--------------------------------------------------------------------------------
/extension/addon/single-file/dist/web/hooks/hooks-web.js:
--------------------------------------------------------------------------------
1 | !function(){"use strict";(e=>{const t=e.CustomEvent,s=e.fetch;var a,r,c;a="single-file-request-fetch",r=async a=>{const r=a.detail;let c;try{const e=await s(r,{cache:"force-cache"});c={url:r,response:await e.arrayBuffer(),headers:[...e.headers],status:e.status}}catch(e){c={url:r,error:e&&e.toString()}}(t=>{e.dispatchEvent(t)})(new t("single-file-response-fetch",{detail:c}))},e.addEventListener(a,r,c)})("object"==typeof globalThis?globalThis:window)}();
2 |
--------------------------------------------------------------------------------
/extension/addon/single-file/lib/single-file/LICENSE.md:
--------------------------------------------------------------------------------
1 | Copyright 2010-2020 Gildas Lormeau
2 | contact : gildas.lormeau gmail.com
3 |
4 | The files in this directory are part of SingleFile.
5 |
6 | The code in this dir is free software: you can redistribute it and/or
7 | modify it under the terms of the GNU Affero General Public License
8 | (GNU AGPL) as published by the Free Software Foundation, either version 3
9 | of the License, or (at your option) any later version.
10 |
11 | The code in this dir is distributed in the hope that it will be useful,
12 | but WITHOUT ANY WARRANTY; without even the implied warranty of
13 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
14 | General Public License for more details.
15 |
16 | As additional permission under GNU AGPL version 3 section 7, you may
17 | distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
18 | AGPL normally required by section 4, provided you include this license
19 | notice and a URL through which recipients can access the Corresponding
20 | Source.
21 |
--------------------------------------------------------------------------------
/extension/addon/single-file/lib/single-file/fetch.js:
--------------------------------------------------------------------------------
1 | /* global browser, XMLHttpRequest */
2 |
3 | const referrers = new Map();
4 | const REQUEST_ID_HEADER_NAME = 'x-single-file-request-id';
5 | // export {
6 | // REQUEST_ID_HEADER_NAME,
7 | // referrers
8 | // };
9 |
10 | browser.runtime.onMessage.addListener((message, sender) => {
11 | if (message.method && message.method.startsWith('singlefile.fetch')) {
12 | return new Promise(resolve => {
13 | onRequest(message, sender)
14 | .then(resolve)
15 | .catch(error => resolve({ error: error && error.toString() }));
16 | });
17 | }
18 | });
19 |
20 | function onRequest(message, sender) {
21 | if (message.method == 'singlefile.fetch') {
22 | return fetchResource(message.url, { referrer: message.referrer });
23 | } else if (message.method == 'singlefile.fetchFrame') {
24 | return browser.tabs.sendMessage(sender.tab.id, message);
25 | }
26 | }
27 |
28 | function fetchResource(url, options, includeRequestId) {
29 | return new Promise((resolve, reject) => {
30 | const xhrRequest = new XMLHttpRequest();
31 | xhrRequest.withCredentials = true;
32 | xhrRequest.responseType = 'arraybuffer';
33 | xhrRequest.onerror = event => reject(new Error(event.detail));
34 | xhrRequest.onreadystatechange = () => {
35 | if (xhrRequest.readyState == XMLHttpRequest.DONE) {
36 | if (xhrRequest.status || xhrRequest.response.byteLength) {
37 | if (
38 | (xhrRequest.status == 401 ||
39 | xhrRequest.status == 403 ||
40 | xhrRequest.status == 404) &&
41 | !includeRequestId
42 | ) {
43 | fetchResource(url, options, true).then(resolve).catch(reject);
44 | } else {
45 | resolve({
46 | array: Array.from(new Uint8Array(xhrRequest.response)),
47 | headers: {
48 | 'content-type': xhrRequest.getResponseHeader('Content-Type'),
49 | },
50 | status: xhrRequest.status,
51 | });
52 | }
53 | } else {
54 | reject();
55 | }
56 | }
57 | };
58 | xhrRequest.open('GET', url, true);
59 | if (includeRequestId) {
60 | const randomId = String(Math.random()).substring(2);
61 | setReferrer(randomId, options.referrer);
62 | xhrRequest.setRequestHeader(REQUEST_ID_HEADER_NAME, randomId);
63 | }
64 | xhrRequest.send();
65 | });
66 | }
67 |
68 | function setReferrer(requestId, referrer) {
69 | referrers.set(requestId, referrer);
70 | }
71 |
--------------------------------------------------------------------------------
/extension/addon/single-file/lib/single-file/frame-tree.js:
--------------------------------------------------------------------------------
1 | /* global browser */
2 |
3 | browser.runtime.onMessage.addListener((message, sender) => {
4 | if (
5 | message.method == 'singlefile.frameTree.initResponse' ||
6 | message.method == 'singlefile.frameTree.ackInitRequest'
7 | ) {
8 | browser.tabs.sendMessage(sender.tab.id, message, { frameId: 0 });
9 | return Promise.resolve({});
10 | }
11 | });
12 |
--------------------------------------------------------------------------------
/extension/addon/single-file/lib/single-file/lazy-timeout.js:
--------------------------------------------------------------------------------
1 | /* global browser, setTimeout, clearTimeout */
2 |
3 | const timeouts = new Map();
4 |
5 | browser.runtime.onMessage.addListener((message, sender) => {
6 | if (message.method == 'singlefile.lazyTimeout.setTimeout') {
7 | let tabTimeouts = timeouts.get(sender.tab.id);
8 | let frameTimeouts;
9 | if (tabTimeouts) {
10 | frameTimeouts = tabTimeouts.get(sender.frameId);
11 | if (frameTimeouts) {
12 | const previousTimeoutId = frameTimeouts.get(message.type);
13 | if (previousTimeoutId) {
14 | clearTimeout(previousTimeoutId);
15 | }
16 | } else {
17 | frameTimeouts = new Map();
18 | }
19 | }
20 | const timeoutId = setTimeout(async () => {
21 | try {
22 | const tabTimeouts = timeouts.get(sender.tab.id);
23 | const frameTimeouts = tabTimeouts.get(sender.frameId);
24 | if (tabTimeouts && frameTimeouts) {
25 | deleteTimeout(frameTimeouts, message.type);
26 | }
27 | await browser.tabs.sendMessage(sender.tab.id, {
28 | method: 'singlefile.lazyTimeout.onTimeout',
29 | type: message.type,
30 | });
31 | } catch (error) {
32 | // ignored
33 | }
34 | }, message.delay);
35 | if (!tabTimeouts) {
36 | tabTimeouts = new Map();
37 | frameTimeouts = new Map();
38 | tabTimeouts.set(sender.frameId, frameTimeouts);
39 | timeouts.set(sender.tab.id, tabTimeouts);
40 | }
41 | frameTimeouts.set(message.type, timeoutId);
42 | return Promise.resolve({});
43 | }
44 | if (message.method == 'singlefile.lazyTimeout.clearTimeout') {
45 | let tabTimeouts = timeouts.get(sender.tab.id);
46 | if (tabTimeouts) {
47 | const frameTimeouts = tabTimeouts.get(sender.frameId);
48 | if (frameTimeouts) {
49 | const timeoutId = frameTimeouts.get(message.type);
50 | if (timeoutId) {
51 | clearTimeout(timeoutId);
52 | }
53 | deleteTimeout(frameTimeouts, message.type);
54 | }
55 | }
56 | return Promise.resolve({});
57 | }
58 | });
59 |
60 | browser.tabs.onRemoved.addListener(tabId => timeouts.delete(tabId));
61 |
62 | function deleteTimeout(framesTimeouts, type) {
63 | framesTimeouts.delete(type);
64 | }
65 |
--------------------------------------------------------------------------------
/extension/content_scripts/archive.js:
--------------------------------------------------------------------------------
1 | function listDocuments() {
2 | fetch('http://localhost:3000/list-docs', {
3 | method: 'GET',
4 | headers: {
5 | 'content-type': 'text/json',
6 | },
7 | })
8 | .then(response => response.json())
9 | .then(data => console.log(data))
10 | .catch(err => {
11 | console.log(err);
12 | });
13 | }
14 |
15 | async function archiveCurrentTab(dataURI) {
16 | let form = new FormData();
17 | form.append('url', window.location.href);
18 | form.append('title', document.title);
19 | form.append('scr', dataURI);
20 | const { content, title, filename } = await extension
21 | .getPageData({
22 | removeHiddenElements: true,
23 | removeUnusedStyles: true,
24 | removeUnusedFonts: true,
25 | removeImports: true,
26 | removeScripts: true,
27 | compressHTML: true,
28 | removeAudioSrc: true,
29 | removeVideoSrc: true,
30 | removeAlternativeFonts: true,
31 | removeAlternativeMedias: true,
32 | removeAlternativeImages: true,
33 | groupDuplicateImages: true,
34 | })
35 | .then(d => {
36 | form.append('onefile', d.content);
37 | fetch('http://localhost:3000/form', {
38 | method: 'POST',
39 | body: form,
40 | }).then(res =>
41 | browser.runtime.sendMessage({
42 | command: 'response-received',
43 | data: res.status,
44 | })
45 | );
46 | });
47 | console.log('FORM', [...form]);
48 | }
49 |
50 | browser.runtime.onMessage.addListener(message => {
51 | if (message.command === 'list-docs') {
52 | listDocuments();
53 | } else if (message.command === 'archive-current-tab') {
54 | archiveCurrentTab(message.data);
55 | }
56 | });
57 |
--------------------------------------------------------------------------------
/extension/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "deptoolkit-extension",
3 | "version": "1.0.0",
4 | "scripts": {
5 | "test": "echo \"Error: no test specified\" && exit 1",
6 | "build": "webpack",
7 | "watch": "webpack -w"
8 | },
9 | "devDependencies": {
10 | "webpack": "5.94.0",
11 | "webpack-webext-plugin": "github:rpl/webpack-webext-plugin"
12 | },
13 | "dependencies": {
14 | "form-data": "^4.0.0"
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/extension/popup/popup.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Listen for clicks on the buttons, and send the appropriate message to
3 | * the content script in the page.
4 | */
5 | const listenForClicks = () => {
6 | function reportError(error) {
7 | console.error(`Could not archive: ${error}`);
8 | }
9 |
10 | const listDocs = tabs => {
11 | browser.tabs.sendMessage(tabs[0].id, {
12 | command: 'list-docs',
13 | });
14 | };
15 |
16 | const archiveCurrentTab = tabs => {
17 | document.getElementById('archive-current-tab').classList.add('selected');
18 | browser.tabs.captureVisibleTab().then(res => {
19 | let screenshotData = res;
20 | browser.tabs.sendMessage(tabs[0].id, {
21 | command: 'archive-current-tab',
22 | data: screenshotData,
23 | });
24 | });
25 | };
26 |
27 | document.getElementById('list-docs').addEventListener('click', e => {
28 | browser.tabs
29 | .query({ active: true, currentWindow: true })
30 | .then(listDocs)
31 | .catch(reportError);
32 | });
33 | document
34 | .getElementById('archive-current-tab')
35 | .addEventListener('click', e => {
36 | browser.tabs
37 | .query({ active: true, currentWindow: true })
38 | .then(archiveCurrentTab)
39 | .catch(reportError);
40 | });
41 | };
42 |
43 | /**
44 | * There was an error executing the script.
45 | * Display the popup's error message, and hide the normal UI.
46 | */
47 | function reportExecuteScriptError(error) {
48 | document.querySelector('#popup-content').classList.add('hidden');
49 | document.querySelector('#error-content').classList.remove('hidden');
50 | console.error(`Failed to execute archiver content script: ${error.message}`);
51 | }
52 |
53 | /**
54 | * HTML element transitions from .selected to .archived
55 | */
56 | function changeArchiveButtonState(btn) {
57 | btn.classList.remove('selected');
58 | btn.classList.add('archived');
59 | btn.innerHTML = '✅ Archived';
60 | }
61 |
62 | /**
63 | * When the popup loads, inject a content script into the active tab,
64 | * and add a click handler.
65 | * If we couldn't inject the script, handle the error.
66 | */
67 | browser.tabs
68 | .executeScript({ file: '/content_scripts/index.js' })
69 | .then(listenForClicks)
70 | .catch(reportExecuteScriptError);
71 |
72 | browser.runtime.onMessage.addListener(message => {
73 | /**
74 | * On receiving an OK response from the API, change the CSS
75 | * and content of the button to prevent duplicate archiving.
76 | * Note: simulate a 400ms job... so it looks like the tool is
77 | * "doing something"
78 | */
79 | if (message.command === 'response-received') {
80 | if (message.data === 200) {
81 | setTimeout(() => {
82 | changeArchiveButtonState(
83 | document.getElementById('archive-current-tab')
84 | );
85 | }, 400);
86 | }
87 | }
88 | });
89 |
--------------------------------------------------------------------------------
/extension/webpack.config.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 | const WebpackWebExt = require('webpack-webext-plugin');
3 |
4 | module.exports = {
5 | entry: {
6 | content_scripts: './content_scripts/archive.js',
7 | popup: './popup/popup.js',
8 | },
9 | output: {
10 | path: path.resolve(__dirname, 'addon'),
11 | filename: '[name]/index.js',
12 | },
13 | plugins: [
14 | new WebpackWebExt({
15 | runOnce: false,
16 | argv: ['lint', '-s', 'addon/'],
17 | }),
18 | ],
19 | };
20 |
--------------------------------------------------------------------------------
/index.ts:
--------------------------------------------------------------------------------
1 | import chalk from 'chalk';
2 |
3 | import app from './app';
4 |
5 | const port = 3000;
6 |
7 | try {
8 | app.listen(port, (): void => {
9 | console.log(chalk.green(`Connected successfully on port ${port} 🚀`));
10 | });
11 | } catch (error) {
12 | console.error(chalk.red(`❌ Error occured: ${(error as Error).message}`));
13 | }
14 |
--------------------------------------------------------------------------------
/jest.config.ts:
--------------------------------------------------------------------------------
1 | import type { Config } from '@jest/types';
2 |
3 | const cfg: Config.InitialOptions = {
4 | transform: {
5 | '^.+\\.(ts|tsx)$': 'ts-jest'
6 | },
7 | testEnvironment: 'node',
8 | }
9 |
10 | export default cfg;
11 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "deptoolkit-api",
3 | "author": "Basile Simon ",
4 | "contributors": [
5 | {
6 | "name": "Niko Para",
7 | "email": "niko@niko.io"
8 | }
9 | ],
10 | "version": "0.11.0",
11 | "repository": {
12 | "type": "git",
13 | "url": "https://github.com/digitalevidencetoolkit/deptoolkit"
14 | },
15 | "engines": {
16 | "node": "^16"
17 | },
18 | "license": "gpl-3.0",
19 | "dependencies": {
20 | "amazon-qldb-driver-nodejs": "^2.2.0",
21 | "aws-sdk": "^2.1354.0",
22 | "body-parser": "^1.20.3",
23 | "chalk": "^4.1.2",
24 | "cors": "^2.8.5",
25 | "dotenv": "^8.2.0",
26 | "esm": "^3.2.25",
27 | "express": "^4.21.0",
28 | "formidable": "^1.2.2",
29 | "ion-js": "^4.2.1",
30 | "jsbi": "^3.1.6",
31 | "node-fetch": "^3.1.1",
32 | "sharp": "^0.32.6"
33 | },
34 | "scripts": {
35 | "start": "ts-node-dev ./index.ts",
36 | "server": "ts-node-dev ./index.ts",
37 | "ui": "cd ui && npm run dev",
38 | "all": "npm-run-all --parallel server ui",
39 | "test": "jest",
40 | "testw": "jest --watch"
41 | },
42 | "devDependencies": {
43 | "@jest/types": "^27.0.6",
44 | "@types/archiver": "^5.1.1",
45 | "@types/cors": "^2.8.12",
46 | "@types/express": "^4.17.13",
47 | "@types/formidable": "^1.2.3",
48 | "@types/jest": "^27.0.1",
49 | "@types/node": "^15.14.7",
50 | "@types/sharp": "^0.28.5",
51 | "@types/supertest": "^2.0.11",
52 | "archiver": "^5.3.0",
53 | "autoprefixer": "^10.3.1",
54 | "jest": "^27.0.6",
55 | "node-stream-zip": "^1.14.0",
56 | "npm-run-all": "^4.1.5",
57 | "sinon": "^11.1.2",
58 | "supertest": "^6.1.6",
59 | "ts-jest": "^27.0.5",
60 | "ts-mock-imports": "^1.3.7",
61 | "ts-node-dev": "^1.1.8",
62 | "typescript": "^4.3.5",
63 | "yup": "^0.32.9"
64 | }
65 | }
66 |
--------------------------------------------------------------------------------
/src/helpers.ts:
--------------------------------------------------------------------------------
1 | import { createHash } from 'crypto';
2 |
3 | /**
4 | * Pretty prints a JSON var
5 | * @returns A nicely-formatted string
6 | */
7 | export const pprint = (json: {}): string => JSON.stringify(json, null, 4);
8 |
9 | export const makeHash = (str: string | Buffer): string => {
10 | let hash = createHash('sha256');
11 | hash.update(str);
12 | return hash.digest('hex');
13 | };
14 |
15 | /**
16 | * change the format of a base64 string to a type we can work with
17 | * @param s the string, as given by the formidable library, or something else
18 | * @returns a string, without the data:image things, and also without the + character
19 | */
20 | export const cleanupBase64 = (s: string): string => {
21 | let base64Data: string;
22 | base64Data = s.replace(/^data:image\/png;base64,/, '');
23 | base64Data += base64Data.replace('+', ' ');
24 | return base64Data;
25 | };
26 |
--------------------------------------------------------------------------------
/src/ledger/index.ts:
--------------------------------------------------------------------------------
1 | import * as Record from '../types/Record';
2 | import * as Annotations from '../types/Annotations';
3 | import type { Annotation } from '../types/Annotations';
4 | import * as QLDB from '../qldb';
5 | import { dom } from 'ion-js';
6 |
7 | import { Result } from 'amazon-qldb-driver-nodejs';
8 |
9 | export const insertDoc = (r: Record.Record): Promise =>
10 | Record.validate(r)
11 | .then(Record.toLedger)
12 | .then(dbItem =>
13 | QLDB.insertDocuments(process.env.DOC_TABLE_NAME as string, dbItem)
14 | )
15 | .catch(err => console.log(`${err.name}: ${err.errors}`));
16 |
17 | export const listDocs = async (): Promise => {
18 | const list: Result | undefined = await QLDB.listDocuments(
19 | process.env.DOC_TABLE_NAME as string
20 | );
21 | const result = list?.getResultList() || [];
22 | return result
23 | .map(e => Record.fromLedger(e))
24 | .map((e: Record.Record): Record.FrontEndRecord => Record.toFrontend(e));
25 | };
26 |
27 | export const getDoc = async (
28 | id: string,
29 | col: string
30 | ): Promise => {
31 | const list: Result | undefined = await QLDB.getOneDocument(
32 | id,
33 | col,
34 | process.env.DOC_TABLE_NAME as string
35 | );
36 | const result = list?.getResultList() || [];
37 | if (result.length > 0) {
38 | return Record.fromLedger(result[0]);
39 | } else {
40 | return null;
41 | }
42 | };
43 |
44 | export const listDocHistory = async (sku: string): Promise => {
45 | const list = await QLDB.queryHistoryOfDocument(sku);
46 | const result = list?.getResultList() || [];
47 | return result;
48 | };
49 |
50 | export const updateDoc = async (sku: string, data: Annotation) => {
51 | Annotations.validate(data)
52 | .then(annotation => annotation.description)
53 | .then(description =>
54 | QLDB.updateDocument(
55 | process.env.DOC_TABLE_NAME as string,
56 | description,
57 | sku
58 | )
59 | )
60 | .catch(err => console.log(`${err.name}: ${err.errors}`));
61 | };
62 |
--------------------------------------------------------------------------------
/src/qldb/index.ts:
--------------------------------------------------------------------------------
1 | import {
2 | QldbDriver,
3 | Result,
4 | TransactionExecutor,
5 | RetryConfig,
6 | } from 'amazon-qldb-driver-nodejs';
7 | import { config } from 'dotenv';
8 | import { ClientConfiguration } from 'aws-sdk/clients/qldbsession';
9 |
10 | config();
11 | const qldbDriver: QldbDriver = createQldbDriver(
12 | process.env.LEDGER_NAME as string
13 | );
14 |
15 | /**
16 | * Create a driver for creating sessions.
17 | * @param ledgerName The name of the ledger to create the driver on.
18 | * @param serviceConfigurationOptions The configurations for the AWS SDK client that the driver uses.
19 | * @returns The driver for creating sessions.
20 | */
21 | function createQldbDriver(
22 | ledgerName: string,
23 | serviceConfigurationOptions: ClientConfiguration = {
24 | region: process.env.AWS_REGION,
25 | }
26 | ): QldbDriver {
27 | const retryLimit = 4;
28 | const maxConcurrentTransactions = 10;
29 | //Use driver's default backoff function (and hence, no second parameter provided to RetryConfig)
30 | const retryConfig: RetryConfig = new RetryConfig(retryLimit);
31 | const qldbDriver: QldbDriver = new QldbDriver(
32 | ledgerName,
33 | serviceConfigurationOptions,
34 | 10,
35 | retryConfig
36 | );
37 | return qldbDriver;
38 | }
39 |
40 | function getQldbDriver(): QldbDriver {
41 | return qldbDriver;
42 | }
43 |
44 | /**
45 | * Connect to a session for a given ledger using default settings.
46 | * @returns Promise which fulfills with void.
47 | */
48 | const ConnectToLedger = async function (): Promise {
49 | try {
50 | console.log('Listing table names...');
51 | const tableNames: string[] = await qldbDriver.getTableNames();
52 | tableNames.forEach((tableName: string): void => {
53 | console.log(tableName);
54 | });
55 | } catch (e) {
56 | console.log(`Unable to create session: ${e}`);
57 | }
58 | };
59 |
60 | /**
61 | * List all documents in a given table.
62 | * @param tableName Name of the table to look up documents from.
63 | * @returns Promise which fulfills with a {@linkcode Result} object.
64 | */
65 | export const listDocuments = async function (
66 | tableName: string
67 | ): Promise {
68 | try {
69 | const qldbDriver: QldbDriver = getQldbDriver();
70 | const statement: string = `SELECT * FROM ${tableName}`;
71 | let r = qldbDriver.executeLambda(async (txn: TransactionExecutor) => {
72 | let results = await txn.execute(statement);
73 | return results;
74 | });
75 | return r;
76 | } catch (e) {
77 | console.log(`Unable to list documents in ${tableName}: ${e}`);
78 | }
79 | };
80 |
81 | export const getOneDocument = async function (
82 | id: string,
83 | column: string,
84 | tableName: string
85 | ): Promise {
86 | try {
87 | const qldbDriver: QldbDriver = getQldbDriver();
88 | const statement: string = `SELECT * FROM ${tableName} WHERE ${column}='${id}'`;
89 | let r = qldbDriver.executeLambda(async (txn: TransactionExecutor) => {
90 | let results = await txn.execute(statement);
91 | return results;
92 | });
93 | return r;
94 | } catch (e) {
95 | console.log(`Unable to list documents in ${tableName}: ${e}`);
96 | }
97 | };
98 |
99 | /**
100 | * Insert documents into a table in a QLDB ledger.
101 | * @param tableName Name of the table to insert documents into.
102 | * @param documents List of documents to insert.
103 | * @returns Promise which fulfills with with a {@linkcode Result} object.
104 | */
105 | export const insertDocuments = async function (
106 | tableName: string,
107 | documents: object
108 | ): Promise {
109 | try {
110 | const qldbDriver: QldbDriver = getQldbDriver();
111 | const statement: string = `INSERT INTO ${tableName} ?`;
112 | let r = qldbDriver.executeLambda(async (txn: TransactionExecutor) => {
113 | let results = await txn.execute(statement, documents);
114 | return results;
115 | });
116 | return r;
117 | } catch (e) {
118 | console.log(`Unable to insert documents: ${e}`);
119 | }
120 | };
121 |
122 | /**
123 | * Get the document IDs from the given table.
124 | * @param txn The {@linkcode TransactionExecutor} for lambda execute.
125 | * @param tableName The table name to query.
126 | * @param field A field to query.
127 | * @param value The key of the given field.
128 | * @returns Promise which fulfills with the document ID as a string.
129 | */
130 | async function getDocumentIdByField(
131 | txn: TransactionExecutor,
132 | tableName: string,
133 | field: string,
134 | value: string
135 | ): Promise {
136 | const query: string = `SELECT id FROM ${tableName} AS t BY id WHERE t.${field} = ?`;
137 | let documentId: string | undefined;
138 | await txn
139 | .execute(query, value)
140 | .then((result: Result) => {
141 | const resultList = result.getResultList();
142 | if (resultList.length === 0) {
143 | throw new Error(
144 | `Unable to retrieve document ID using ${field}: ${value}.`
145 | );
146 | }
147 | documentId = resultList[0].get('id')?.stringValue() || undefined;
148 | })
149 | .catch((err: any) => {
150 | console.log(`Error getting documentId: ${err}`);
151 | });
152 | return documentId;
153 | }
154 |
155 | /**
156 | * List the history of a particular document.
157 | * @param sku Unique identifier of the document.
158 | * @returns Promise which fulfills with a list of results
159 | */
160 | export const queryHistoryOfDocument = async function (
161 | sku: string
162 | ): Promise {
163 | const tableName = process.env.DOC_TABLE_NAME as string;
164 | try {
165 | const qldbDriver: QldbDriver = getQldbDriver();
166 | const statement: string = `SELECT * from history (${tableName}) AS h WHERE h.metadata.id = ?`;
167 |
168 | let r = qldbDriver.executeLambda(async (txn: TransactionExecutor) => {
169 | const documentId: string | undefined = await getDocumentIdByField(
170 | txn,
171 | tableName,
172 | 'id',
173 | sku
174 | );
175 |
176 | // @TODO handle results = undefined because sku can't be found
177 | // current solution doesn't feel like the correct way to handle errors
178 | if (documentId) {
179 | let results = await txn.execute(statement, documentId);
180 | return results;
181 | }
182 | });
183 | return r;
184 | } catch (e) {
185 | console.log(`Unable to query history of document in ${tableName}`);
186 | }
187 | };
188 |
189 | /**
190 | * Update a particular document's description field
191 | * @param tableName Name of the table to insert documents into.
192 | * @param description String to add to the ledger
193 | * @param id Unique identifier of the document
194 | * @returns Promise which fulfills with a list of result(s)
195 | */
196 | export const updateDocument = async function (
197 | tableName: string,
198 | description: string,
199 | id: string
200 | ): Promise {
201 | try {
202 | const qldbDriver: QldbDriver = getQldbDriver();
203 | const statement: string = `UPDATE ${tableName} AS r
204 | SET r.description = ?
205 | WHERE r.id = ?`;
206 | let r = qldbDriver.executeLambda(async (txn: TransactionExecutor) => {
207 | let results = await txn.execute(statement, description, id);
208 | return results;
209 | });
210 | return r;
211 | } catch (e) {
212 | console.log(`Unable to update document: ${e}`);
213 | }
214 | };
215 |
--------------------------------------------------------------------------------
/src/s3/index.ts:
--------------------------------------------------------------------------------
1 | import { S3 } from 'aws-sdk';
2 | import { Readable } from 'stream';
3 | import { config } from 'dotenv';
4 |
5 | config();
6 |
7 | export const getFileInBucket = (id: string, bucket: string): Readable => {
8 | const s3 = new S3({ region: process.env.AWS_REGION });
9 | const params = { Bucket: bucket as string, Key: id };
10 | return s3.getObject(params).createReadStream();
11 | };
12 |
13 | export const writeFileInBucket = (
14 | id: string,
15 | body: Buffer | string,
16 | bucket: string
17 | ) => {
18 | const s3 = new S3({ region: process.env.AWS_REGION });
19 | const params = { Bucket: bucket, Body: body, Key: id };
20 | return s3.upload(params, (err: Error, data: S3.ManagedUpload.SendData) => {
21 | if (err) {
22 | throw err;
23 | }
24 | console.log(`☁️ S3 upload OK`);
25 | });
26 | };
27 |
--------------------------------------------------------------------------------
/src/s3/s3.test.ts:
--------------------------------------------------------------------------------
1 | // good test suite from AWS:
2 | // https://github.com/aws/aws-sdk-js-v3/blob/main/clients/client-s3/test/e2e/S3.ispec.ts
3 |
4 | import * as S3 from './index';
5 | import { PassThrough } from 'stream';
6 |
7 | describe('uploading files to S3', () => {
8 | it('should be able to upload an object', async () => {
9 | //
10 | });
11 | });
12 |
13 | describe('finding files on S3', () => {
14 | it('should find a known object', async () => {
15 | const key =
16 | 'a8eab0456eb979653a3c1ca77c37239100ef7300c9026db868ad9dc57f5aa580.png';
17 | expect(S3.getFileInBucket(key, 'deptoolkit-public')).toBeInstanceOf(
18 | PassThrough
19 | );
20 | });
21 | it('should error on an unknown object', async () => {
22 | const key = 'foo.png';
23 | const r = S3.getFileInBucket(key, 'deptoolkit-public');
24 | r.on('error', e => expect(e).toBeInstanceOf(Error));
25 | });
26 | });
27 |
--------------------------------------------------------------------------------
/src/store/index.ts:
--------------------------------------------------------------------------------
1 | import archiver from 'archiver';
2 | import { Response } from 'express';
3 | import { config } from 'dotenv';
4 | import * as fs from 'fs';
5 | import * as fsp from 'fs/promises';
6 | import * as path from 'path';
7 | import { makeHash, pprint } from '../helpers';
8 | import * as File from '../types/File';
9 | import * as Bundle from '../types/Bundle';
10 | import * as Record from '../types/Record';
11 | import * as S3 from '../s3';
12 | // import * as s3storage from s3storage
13 |
14 | // set up .env variables as environment variables
15 | config();
16 |
17 | type WriteConfiguration = {
18 | type: 'local' | 'S3' | undefined;
19 | directory: string;
20 | bucket?: string;
21 | };
22 |
23 | /**
24 | * Write the specified `data` in the specified `directory`, under the specified
25 | * `fileName`.
26 | *
27 | * If `directory` doesn't exist, it is created. If there is already a file
28 | * named `fileName` in that location, it is silently overwritten.
29 | *
30 | * @returns a Promise resolving once the write is done.
31 | */
32 | const writeToDisk = async (
33 | data: Buffer | string,
34 | directory: string,
35 | fileName: string
36 | ): Promise => {
37 | // create the directory if it does not yet exist
38 | await fsp.mkdir(directory, { recursive: true });
39 | await fsp.writeFile(path.join(directory, fileName), data);
40 | };
41 |
42 | /**
43 | * Write the specified `newFile` to disk or the cloud, according to the specified
44 | * `configuration`.
45 | *
46 | * If the same file is written twice, the old version is replaced silently.
47 | *
48 | * @param newFile object holding and describing the data to be written
49 | * @param configuration object describing how the new file should be written
50 | * @returns a promise for the resulting file, resolving once the write is
51 | * complete.
52 | **/
53 | const writeOne = async (
54 | newFile: File.NewFile,
55 | configuration: WriteConfiguration
56 | ): Promise => {
57 | const destination = sourceToFavour();
58 | const { directory } = configuration;
59 | const { kind, data } = newFile;
60 |
61 | const hash = makeHash(data);
62 | const result: File.File = { kind, hash };
63 |
64 | if (destination === 'directory') {
65 | await writeToDisk(data, directory, File.fileName(result));
66 | } else if (destination === 'bucket') {
67 | await writeToDisk(data, directory, File.fileName(result));
68 | await S3.writeFileInBucket(
69 | File.fileName(result),
70 | data,
71 | process.env.SOURCE_FILES_BUCKET as string
72 | );
73 | }
74 |
75 | return result;
76 | };
77 |
78 | /**
79 | * Writes to disk the data contained in the specified `newBundle`, following the
80 | * specified `configuration`.
81 | *
82 | * If the same new bundle is written twice (or part of a new bundle is
83 | * re-written as part of another new bundle), the relevant files are replaced
84 | * silently.
85 | *
86 | * @param newBundle a NewBundle, made of base64 string data
87 | * @param configuration object describing how to write the new bundle
88 | * @returns a promise for the resulting bundle, resolving once the write is
89 | * complete.
90 | **/
91 | export const newBundle = (
92 | newBundle: Bundle.NewBundle,
93 | configuration: WriteConfiguration
94 | ): Promise => {
95 | return Promise.all(
96 | newBundle.map(newFile => writeOne(newFile, configuration))
97 | );
98 | };
99 |
100 | /**
101 | * Generates a string describing the specified Record `r`.
102 | */
103 | export const generateAboutString = (r: Record.Record): string => {
104 | const screenshotFile = r.bundle.find(e => e.kind === 'screenshot');
105 | const oneFileFile = r.bundle.find(e => e.kind === 'one_file');
106 | if (screenshotFile === undefined || oneFileFile === undefined) {
107 | throw new Error(
108 | `This record's bundle is malformed (missing either screenshot or oneFile): ${pprint(
109 | r
110 | )}`
111 | );
112 | }
113 |
114 | const screenshot = File.fileName(screenshotFile);
115 | const one_file = File.fileName(oneFileFile);
116 |
117 | return `THE DIGITAL EVIDENCE PRESERVATION TOOLKIT
118 | ============
119 | Working copy export generated on ${Date.now()}
120 |
121 | ${r.data.title}
122 | ${r.data.url}
123 |
124 | Files included:
125 | ${screenshot}
126 | ${one_file}`;
127 | };
128 |
129 | /**
130 | * Builds a ZIP file from the specified Record `r`, in the specified
131 | * `outDirectory`. The produced file will use the record's bundle id (as produced
132 | * by `Bundle.id`) as its name, and will be located in the specified
133 | * `outDirectory`. The function will use the specified `bundleRootDirectory` as the root
134 | * where to read the files listed in `r`.
135 | *
136 | * The behaviour is unspecified unless:
137 | * - `r` specifies one 'one_file' and one 'screenshot' files
138 | * - the screenshot and one_file files exist and can be read in `bundleRootDirectory`
139 | * - `outDirectory` exists and is writable
140 | *
141 | * @returns a promise resolving once the ZIP file has been created, or rejecting
142 | * if an error happened.
143 | */
144 | export const makeZip = (
145 | r: Record.Record,
146 | bundleRootDirectory: string,
147 | outDirectory: string
148 | ): Promise => {
149 | const b = r.bundle;
150 | const zip = archiver('zip', { zlib: { level: 0 } });
151 | const out = `${outDirectory}/${Bundle.id(b)}.zip`;
152 |
153 | return new Promise((resolve, reject) => {
154 | const screenshotFile = r.bundle.find(e => e.kind === 'screenshot');
155 | const oneFileFile = r.bundle.find(e => e.kind === 'one_file');
156 | if (screenshotFile === undefined || oneFileFile === undefined) {
157 | reject(
158 | new Error(
159 | `This record's bundle is malformed (missing either screenshot or oneFile): ${pprint(
160 | r
161 | )}`
162 | )
163 | );
164 | return;
165 | }
166 | // presently the ZIP file only includes the full screenshot and one-file
167 | // HTML archive. isolating them like so isn't the most elegant.
168 | // maybe replace with a function from `Bundle`?
169 | const screenshotName = File.fileName(screenshotFile);
170 | const one_fileName = File.fileName(oneFileFile);
171 | const sidecarTextFile = generateAboutString(r);
172 |
173 | const stream = fs.createWriteStream(out);
174 | stream.on('error', e => {
175 | console.error(e);
176 | reject();
177 | });
178 | stream.on('close', () => {
179 | console.log(`${zip.pointer()} bytes written`);
180 | resolve();
181 | });
182 | zip
183 | .append(
184 | fs
185 | .createReadStream(`${bundleRootDirectory}/${screenshotName}`)
186 | .on('error', reject),
187 | {
188 | name: screenshotName,
189 | }
190 | )
191 | .append(
192 | fs
193 | .createReadStream(`${bundleRootDirectory}/${one_fileName}`)
194 | .on('error', reject),
195 | {
196 | name: one_fileName,
197 | }
198 | )
199 | .append(sidecarTextFile, { name: `about-this-export.txt` })
200 | .on('error', reject)
201 | .pipe(stream);
202 | zip.finalize();
203 | });
204 | };
205 |
206 | /**
207 | * Handles how to access a file depending on its source, i.e. dispatch between
208 | * the file system or an S3 bucket (and hopefully more to come)
209 | * @returns a Response containing the file
210 | */
211 | export const getFile = (id: string, res: Response) => {
212 | const source = sourceToFavour();
213 | if (source === 'directory') {
214 | const outDir = path.join(
215 | __dirname,
216 | `./../../${process.env.SOURCE_FILES_DIRECTORY}`
217 | );
218 | const options = {
219 | root: outDir,
220 | dotfiles: 'deny',
221 | };
222 | res.sendFile(`${id}`, options);
223 | } else if (source === 'bucket') {
224 | const result = S3.getFileInBucket(
225 | id,
226 | process.env.SOURCE_FILES_BUCKET as string
227 | );
228 | result.on('error', err => {
229 | const { name } = err;
230 | res.status(500).send(`Error getting file: S3 ${name}`);
231 | });
232 | result.pipe(res);
233 | }
234 | };
235 |
236 | /**
237 | * Handles dispatch between local and cloud resources.
238 | * If a bucket name is present in the config, then this
239 | * data source will be preferred over local directory.
240 | * @returns a string: 'bucket' or 'directory'
241 | */
242 | export const sourceToFavour = (): 'bucket' | 'directory' | null => {
243 | const hasBucket =
244 | process.env.SOURCE_FILES_BUCKET && process.env.SOURCE_FILES_BUCKET != '';
245 | const hasDirectory =
246 | process.env.SOURCE_FILES_DIRECTORY &&
247 | process.env.SOURCE_FILES_DIRECTORY != '';
248 | if (hasBucket === true) {
249 | return 'bucket';
250 | } else if (hasDirectory === true) {
251 | return 'directory';
252 | } else {
253 | return null;
254 | }
255 | };
256 |
257 | /**
258 | * if cfg.s3
259 | * ...
260 | * if cfg.local
261 | * ...
262 | **/
263 |
--------------------------------------------------------------------------------
/src/store/store.test.ts:
--------------------------------------------------------------------------------
1 | import 'buffer';
2 | import * as fs from 'fs';
3 | import * as fsp from 'fs/promises';
4 | import * as path from 'path';
5 | import Zip from 'node-stream-zip';
6 | import { config } from 'dotenv';
7 |
8 | import * as Bundle from '../types/Bundle';
9 | import * as Record from '../types/Record';
10 | import * as File from '../types/File';
11 |
12 | import * as Store from './index';
13 |
14 | describe('write a file locally', () => {
15 | let outDir = '';
16 | config();
17 | const old_env = process.env;
18 |
19 | beforeEach(() => {
20 | //jest.resetModules();
21 | process.env = { ...old_env };
22 | process.env.SOURCE_FILES_DIRECTORY = 'out';
23 | process.env.SOURCE_FILES_BUCKET = '';
24 | outDir = fs.mkdtempSync('out-');
25 | });
26 | afterEach(() => {
27 | fs.rmSync(outDir, { recursive: true, force: true });
28 | });
29 | afterAll(() => {
30 | process.env = old_env;
31 | });
32 |
33 | const validateFile = async (
34 | { kind, hash }: File.File,
35 | expectedPath: string,
36 | expectedData: string | Buffer
37 | ): Promise => {
38 | const name = `${hash}.${kind === 'one_file' ? 'html' : 'png'}`;
39 | const actual = await fsp.readFile(path.join(expectedPath, name));
40 | const expected = Buffer.from(expectedData);
41 | expect(actual.compare(expected)).toBe(0);
42 | };
43 |
44 | it('should write to disk the specified new bundle', async () => {
45 | const newBundle: Bundle.NewBundle = [
46 | { kind: 'one_file', data: 'jeejtuut' },
47 | { kind: 'screenshot', data: 'foobar' },
48 | ];
49 |
50 | const bundle = await Store.newBundle(newBundle, {
51 | type: 'local',
52 | directory: outDir,
53 | });
54 |
55 | expect(bundle.length).toBe(2);
56 | await Promise.all(
57 | bundle.map((file, index) =>
58 | validateFile(file, outDir, newBundle[index].data)
59 | )
60 | );
61 | });
62 |
63 | it('should create the specified directory if it doesnt exist', async () => {
64 | const nestedDir = path.join(outDir, 'jeej-tuut');
65 |
66 | const newBundle: Bundle.NewBundle = [
67 | { kind: 'one_file', data: 'jeejtuut' },
68 | { kind: 'screenshot', data: 'foobar' },
69 | ];
70 |
71 | const bundle = await Store.newBundle(newBundle, {
72 | type: 'local',
73 | directory: nestedDir,
74 | });
75 |
76 | expect(bundle.length).toBe(2);
77 | await Promise.all(
78 | bundle.map((file, index) =>
79 | validateFile(file, nestedDir, newBundle[index].data)
80 | )
81 | );
82 | });
83 |
84 | it('should be robust to writing the same file twice', async () => {
85 | const newBundle1: Bundle.NewBundle = [
86 | { kind: 'one_file', data: 'jeejtuut' },
87 | { kind: 'screenshot', data: 'foobar' },
88 | ];
89 | const newBundle2: Bundle.NewBundle = [
90 | { kind: 'one_file', data: 'jeejtuut' },
91 | { kind: 'screenshot', data: 'souce' },
92 | ];
93 |
94 | const bundle1 = await Store.newBundle(newBundle1, {
95 | type: 'local',
96 | directory: outDir,
97 | });
98 | const bundle2 = await Store.newBundle(newBundle2, {
99 | type: 'local',
100 | directory: outDir,
101 | });
102 | const bundle3 = await Store.newBundle(newBundle1, {
103 | type: 'local',
104 | directory: outDir,
105 | });
106 |
107 | expect(bundle1[0]).toEqual(bundle2[0]);
108 | expect(bundle1).toEqual(bundle3);
109 | });
110 | });
111 |
112 | describe('write a file to S3', () => {
113 | // or maybe test in src/s3 directly
114 | });
115 |
116 | describe('generateAboutString', () => {
117 | it('should throw if the record is missing the screenshot or one_file', () => {
118 | const bundles: Bundle.Bundle[] = [
119 | [],
120 | [{ hash: 'jeej', kind: 'one_file' }],
121 | [{ hash: 'tuut', kind: 'screenshot' }],
122 | ];
123 |
124 | bundles.forEach(bundle =>
125 | expect(() =>
126 | Store.generateAboutString({
127 | bundle,
128 | annotations: {
129 | description: '',
130 | },
131 | data: {
132 | title: 'foo',
133 | url: 'http://jeej.tuut',
134 | },
135 | })
136 | ).toThrow()
137 | );
138 | });
139 |
140 | it('should generate a string describing the given bundle', () => {
141 | const title = 'Win big money in no time thanks to this one simple trick';
142 | const url = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
143 | const oneFile: File.File = { hash: 'this-is-the-file', kind: 'one_file' };
144 | const screenshot: File.File = {
145 | hash: 'pretty-picture',
146 | kind: 'screenshot',
147 | };
148 | const ogNow = Date.now;
149 | Date.now = () => 42;
150 |
151 | const result = Store.generateAboutString({
152 | bundle: [oneFile, screenshot],
153 | annotations: {
154 | description: '',
155 | },
156 | data: {
157 | title,
158 | url,
159 | },
160 | });
161 |
162 | const expected = `THE DIGITAL EVIDENCE PRESERVATION TOOLKIT
163 | ============
164 | Working copy export generated on 42
165 |
166 | ${title}
167 | ${url}
168 |
169 | Files included:
170 | ${File.fileName(screenshot)}
171 | ${File.fileName(oneFile)}`;
172 |
173 | expect(result).toEqual(expected);
174 |
175 | Date.now = ogNow;
176 | });
177 | });
178 |
179 | describe('makeZip', () => {
180 | let bundleRootDir = '';
181 | let outDir = '';
182 |
183 | beforeEach(() => {
184 | bundleRootDir = fs.mkdtempSync('in-');
185 | outDir = fs.mkdtempSync('out-');
186 | });
187 |
188 | afterEach(() => {
189 | [bundleRootDir, outDir].forEach(p =>
190 | fs.rmSync(p, { force: true, recursive: true })
191 | );
192 | });
193 |
194 | it('should reject if the record is missing the screenshot or one_file', () => {
195 | const bundles: Bundle.Bundle[] = [
196 | [],
197 | [{ hash: 'jeej', kind: 'one_file' }],
198 | [{ hash: 'tuut', kind: 'screenshot' }],
199 | ];
200 |
201 | bundles.forEach(async bundle => {
202 | let zip = undefined;
203 | try {
204 | zip = await Store.makeZip(
205 | {
206 | bundle,
207 | annotations: {
208 | description: '',
209 | },
210 | data: {
211 | title: 'foo',
212 | url: 'http://jeej.tuut',
213 | },
214 | },
215 | bundleRootDir,
216 | outDir
217 | );
218 | } catch (e) {}
219 | expect(zip).toBeUndefined();
220 | });
221 | });
222 |
223 | it('should produce a zip of the given record, in the correct location', async () => {
224 | const oneFile: File.File = { hash: 'this-is-the-file', kind: 'one_file' };
225 | const screenshot: File.File = {
226 | hash: 'pretty-picture',
227 | kind: 'screenshot',
228 | };
229 | const oneFileName = File.fileName(oneFile);
230 | const screenshotName = File.fileName(screenshot);
231 |
232 | fs.writeFileSync(path.join(bundleRootDir, oneFileName), 'jeej');
233 | fs.writeFileSync(path.join(bundleRootDir, screenshotName), 'tuut');
234 |
235 | const record: Record.Record = {
236 | data: {
237 | title: 'Non Stop Nyan Cat',
238 | url: 'http://www.nyan.cat/',
239 | },
240 | annotations: {
241 | description: 'A cat farting an infinite rainbow',
242 | },
243 | bundle: [oneFile, screenshot],
244 | };
245 |
246 | await Store.makeZip(record, bundleRootDir, outDir);
247 |
248 | const zipPath = path.join(outDir, `${Bundle.id(record.bundle)}.zip`);
249 |
250 | // check for the zip's existence and contents
251 | // TODO: for this test to be complete, we should also check the files
252 | // contents themselves.
253 | const z = new Zip.async({ file: zipPath });
254 | const entries = await z.entries();
255 | expect(entries).toEqual({
256 | [oneFileName]: expect.any(Object),
257 | [screenshotName]: expect.any(Object),
258 | 'about-this-export.txt': expect.any(Object),
259 | });
260 | z.close();
261 | });
262 |
263 | it('should return a rejected promise when a file is missing on disk', async () => {
264 | const oneFile: File.File = { hash: 'this-is-the-file', kind: 'one_file' };
265 | const screenshot: File.File = {
266 | hash: 'pretty-picture',
267 | kind: 'screenshot',
268 | };
269 | const screenshotName = File.fileName(screenshot);
270 |
271 | // simulate a missing file
272 | // fs.writeFileSync(path.join(bundleRootDir, oneFileName), 'jeej');
273 | fs.writeFileSync(path.join(bundleRootDir, screenshotName), 'tuut');
274 |
275 | const record: Record.Record = {
276 | data: {
277 | title: 'Non Stop Nyan Cat',
278 | url: 'http://www.nyan.cat/',
279 | },
280 | annotations: {
281 | description: 'A cat farting an infinite rainbow',
282 | },
283 | bundle: [oneFile, screenshot],
284 | };
285 |
286 | await Store.makeZip(record, bundleRootDir, outDir)
287 | .then(() => {
288 | throw new Error('hello error, bad luck');
289 | })
290 | .catch(() => {
291 | // this is the expected behavior
292 | });
293 | });
294 | });
295 |
296 | describe('sourceToFavour', () => {
297 | config();
298 | const old_env = process.env;
299 |
300 | beforeEach(() => {
301 | jest.resetModules();
302 | process.env = { ...old_env };
303 | });
304 | afterAll(() => {
305 | process.env = old_env;
306 | });
307 |
308 | it('should return a value based on env', () => {
309 | process.env.SOURCE_FILES_DIRECTORY = 'out';
310 | const source = Store.sourceToFavour();
311 | expect(typeof source).toBe('string');
312 | });
313 | it('should be null if no config', () => {
314 | process.env.SOURCE_FILES_DIRECTORY = '';
315 | process.env.SOURCE_FILES_BUCKET = '';
316 | const source = Store.sourceToFavour();
317 | expect(source).toBe(null);
318 | });
319 | it('should favour bucket over local dir', () => {
320 | process.env.SOURCE_FILES_DIRECTORY = 'dir';
321 | process.env.SOURCE_FILES_BUCKET = 'bucket';
322 | const source = Store.sourceToFavour();
323 | expect(source).toBe('bucket');
324 | });
325 | it('should be dir if bucket is empty', () => {
326 | process.env.SOURCE_FILES_DIRECTORY = 'dir';
327 | process.env.SOURCE_FILES_BUCKET = '';
328 | const source = Store.sourceToFavour();
329 | expect(source).toBe('directory');
330 | });
331 | });
332 |
--------------------------------------------------------------------------------
/src/types/Annotations.ts:
--------------------------------------------------------------------------------
1 | import * as yup from 'yup';
2 |
3 | /**
4 | * Annotations are user-created data _about_ the record,
5 | * most probably after the original archive through the UI
6 | */
7 | export type Annotation = { description: string; [key: string]: any };
8 |
9 | const DocSchema = yup
10 | .object()
11 | .shape({
12 | description: yup.string(),
13 | })
14 | .strict()
15 | .noUnknown();
16 |
17 | export const validate = (obj: Annotation): Promise =>
18 | DocSchema.validate(obj);
19 |
--------------------------------------------------------------------------------
/src/types/Bundle.ts:
--------------------------------------------------------------------------------
1 | import { makeHash } from '../helpers';
2 | import * as File from './File';
3 |
4 | /**
5 | * A bundle is a list of files, which belong to a discrete
6 | * number of "kinds": screenshots, thumbnails, one-file downloads...
7 | */
8 | export type Bundle = File.File[];
9 |
10 | /**
11 | * list of File-like things with base64-encoded strings,
12 | * that should be stored but haven't been yet
13 | */
14 | export type NewBundle = Array;
15 |
16 | /**
17 | * Ensure the bundle (a list of files) is _always_ sorted
18 | * the same way to preserve identity.
19 | * @param b a Bundle
20 | * @returns a Bundle, sorted alphabetically by its ID
21 | */
22 | const sortedBundle = (b: Bundle): Bundle => {
23 | const sortedArray = [...b].sort((a, b) =>
24 | File.id(a).localeCompare(File.id(b))
25 | );
26 | return sortedArray;
27 | };
28 |
29 | /**
30 | * A bundle's ID is the sum of its composing parts' IDs (or hashes)
31 | * @param b a Bundle
32 | * @returns a string
33 | */
34 | // @TODO: confirm this is accurate
35 | export const id = (b: Bundle): string =>
36 | makeHash(sortedBundle(b).map(File.id).join(''));
37 |
--------------------------------------------------------------------------------
/src/types/Data.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Data saved from the archived webpage
3 | **/
4 | export type Data = { title: string; url: string };
5 |
--------------------------------------------------------------------------------
/src/types/File.test.ts:
--------------------------------------------------------------------------------
1 | import * as File from './File';
2 |
3 | describe('id', () => {
4 | it('should return the file hash', () => {
5 | const hash = 'I should have been a pair of ragged claws';
6 | expect(File.id({ kind: 'one_file', hash })).toBe(hash);
7 | });
8 | });
9 |
10 | describe('fileName', () => {
11 | it('should generate the file name by adding the correct extention to the hash', () => {
12 | const cases: Array = [
13 | { hash: 'The gate is straight', kind: 'one_file', expected: 'html' },
14 | { hash: 'Deep and wide', kind: 'screenshot', expected: 'png' },
15 | {
16 | hash: 'Break on through to the other side',
17 | kind: 'screenshot_thumbnail',
18 | expected: 'png',
19 | },
20 | ];
21 |
22 | cases.forEach(c => {
23 | expect(File.fileName(c)).toBe(`${c.hash}.${c.expected}`);
24 | });
25 | });
26 | });
27 |
--------------------------------------------------------------------------------
/src/types/File.ts:
--------------------------------------------------------------------------------
1 | type Hash = string;
2 |
3 | export type Process = 'screenshot' | 'one_file' | 'screenshot_thumbnail';
4 |
5 | export type File = { kind: Process; hash: Hash };
6 |
7 | export type NewFile = { kind: Process; data: Buffer | string };
8 |
9 | /**
10 | * A File's ID is its hash
11 | * @param a a File
12 | * @returns string
13 | */
14 | export const id = (a: File) => a.hash;
15 |
16 | /**
17 | * Generates a complete filename from a given file.
18 | * @returns A complete filename, with the correct extension according to the
19 | * file's kind.
20 | */
21 | export const fileName: (f: File) => string = ({ kind, hash }) =>
22 | `${hash}.${kind === 'one_file' ? 'html' : 'png'}`;
23 |
--------------------------------------------------------------------------------
/src/types/Record.ts:
--------------------------------------------------------------------------------
1 | import * as yup from 'yup';
2 | import * as Annotations from './Annotations';
3 | import * as Data from './Data';
4 | import * as Bundle from './Bundle';
5 |
6 | export type Record = {
7 | bundle: Bundle.Bundle;
8 | annotations: Annotations.Annotation;
9 | data: Data.Data;
10 | };
11 |
12 | export type FrontEndRecord = {
13 | title: string;
14 | sku: string;
15 | url: string;
16 | thumb_hash?: string;
17 | screenshot_hash?: string;
18 | one_file_hash?: string;
19 | description?: string;
20 | };
21 |
22 | const DocSchema = yup
23 | .object()
24 | .shape({
25 | bundle: yup
26 | .array()
27 | .of(yup.object().shape({ kind: yup.string(), hash: yup.string() })),
28 | annotations: yup.object().shape({ description: yup.string() }),
29 | data: yup.object().shape({ title: yup.string(), url: yup.string() }),
30 | })
31 | .strict()
32 | .noUnknown();
33 |
34 | export const validate = (r: Record): Promise => DocSchema.validate(r);
35 |
36 | type ArbitraryObject = { [key: string]: any };
37 |
38 | /**
39 | * A Record's ID is the same ID as its bundle
40 | * @param r a Record
41 | * @returns string
42 | */
43 | export const id = (r: Record) => Bundle.id(r.bundle);
44 |
45 | /**
46 | * Converts a Record data structure ahead of sending to the ledger
47 | * @param r a Record
48 | * @returns an un-nested Record structure
49 | */
50 | export const toLedger = (r: Record) => ({
51 | id: id(r),
52 | ...r.data,
53 | ...r.annotations,
54 | /**
55 | * [{kind: 'a', hash: 'aaa'}, {kind: 'b', hash: 'bbb'}]
56 | * --> { a: 'aaa', b: 'bbb' }
57 | */
58 | ...r.bundle.reduce((a, { kind, hash }) => ({ ...a, [kind]: hash }), {}),
59 | });
60 |
61 | /**
62 | * Converts a ledger data structure to a Record one
63 | * @param o an object from the ledger
64 | * @returns a Record
65 | */
66 | export const fromLedger = (o: ArbitraryObject): Record => {
67 | return {
68 | data: { url: o?.url, title: o?.title },
69 | annotations: { description: o?.description },
70 | bundle: [
71 | { kind: 'screenshot' as const, hash: o?.screenshot },
72 | {
73 | kind: 'screenshot_thumbnail' as const,
74 | hash: o?.screenshot_thumbnail,
75 | },
76 | { kind: 'one_file' as const, hash: o?.one_file },
77 | ],
78 | };
79 | console.warn(` got this junk from the ledger: ${o}`);
80 | };
81 |
82 | /**
83 | * Converts a Record data structure ahead of rendering by the frontend
84 | * @param r
85 | * @returns
86 | */
87 | export const toFrontend = (r: Record): FrontEndRecord => ({
88 | title: r.data.title,
89 | sku: id(r),
90 | url: r.data.url,
91 | thumb_hash: r.bundle.find(f => f.kind === 'screenshot_thumbnail')?.hash,
92 | screenshot_hash: r.bundle.find(f => f.kind === 'screenshot')?.hash,
93 | one_file_hash: r.bundle.find(f => f.kind === 'one_file')?.hash,
94 | description: r.annotations?.description,
95 | });
96 |
--------------------------------------------------------------------------------
/src/verify/index.ts:
--------------------------------------------------------------------------------
1 | import { makeHash } from '../helpers';
2 | import * as Record from '../types/Record';
3 | import * as Ledger from '../ledger';
4 |
5 | /**
6 | * Verify whether or not the specified `file` is present in the Ledger,
7 | *
8 | * @return a Promise resolving to:
9 | * - `null` if the file isn't present, or
10 | * - the corresponding record, converted to Frontend format, otherwise.
11 | **/
12 | export const verifyFile = async (
13 | file: Buffer
14 | ): Promise => {
15 | const hash = makeHash(file);
16 | const record = await Ledger.getDoc(hash, 'screenshot');
17 | return record === null ? record : Record.toFrontend(record);
18 | };
19 |
--------------------------------------------------------------------------------
/src/verify/verify.test.ts:
--------------------------------------------------------------------------------
1 | import { exportNamedDeclaration } from '@babel/types';
2 | import { Buffer } from 'buffer';
3 | import { ImportMock } from 'ts-mock-imports';
4 | import * as Ledger from '../ledger';
5 | import * as Record from '../types/Record';
6 |
7 | import * as Verify from './index';
8 |
9 | describe('verify', () => {
10 | const buffer = Buffer.from('foo-bar');
11 |
12 | afterEach(() => {
13 | ImportMock.restore();
14 | });
15 |
16 | it('should return null when the file could not be found on the ledger', async () => {
17 | ImportMock.mockFunction(Ledger, 'getDoc', Promise.resolve(null));
18 |
19 | const result = await Verify.verifyFile(buffer);
20 |
21 | expect(result).toBeNull();
22 | });
23 |
24 | it('should return the corresponding record when the file is found', async () => {
25 | const foundRecord: Record.Record = {
26 | data: {
27 | title: 'Async/Await - The modern Javascript tutorial',
28 | url: 'https://javascript.info/async-await',
29 | },
30 | annotations: {
31 | description: 'Learn this one weird trick',
32 | },
33 | bundle: [],
34 | };
35 |
36 | ImportMock.mockFunction(Ledger, 'getDoc', Promise.resolve(foundRecord));
37 |
38 | const result = await Verify.verifyFile(buffer);
39 |
40 | expect(result).toEqual(Record.toFrontend(foundRecord));
41 | });
42 | });
43 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "target": "es5",
4 | "module": "commonjs",
5 | "strict": true,
6 | "esModuleInterop": true,
7 | "skipLibCheck": true,
8 | "forceConsistentCasingInFileNames": true
9 | }
10 | }
11 |
--------------------------------------------------------------------------------
/ui/.dockerignore:
--------------------------------------------------------------------------------
1 | node_modules/
2 | out/
--------------------------------------------------------------------------------
/ui/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | node_modules
3 | /build
4 | /.svelte-kit
5 | /package
6 |
--------------------------------------------------------------------------------
/ui/.node-version:
--------------------------------------------------------------------------------
1 | 16.13.0
2 |
--------------------------------------------------------------------------------
/ui/.npmrc:
--------------------------------------------------------------------------------
1 | engine-strict=true
2 |
--------------------------------------------------------------------------------
/ui/.prettierrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "singleQuote": true,
3 | "arrowParens": "avoid",
4 | "jsxSingleQuote": true,
5 | "svelteSortOrder": "options-scripts-styles-markup"
6 | }
7 |
--------------------------------------------------------------------------------
/ui/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM node:16.13.0-alpine3.14 as builder
2 | LABEL version="0.1"
3 | LABEL description="GUI from the QLDB ledger"
4 |
5 | WORKDIR /app
6 | COPY . .
7 | RUN npm install
8 | RUN node node_modules/esbuild/install.js
9 | RUN npm run build
10 |
11 | FROM nginx:alpine
12 | COPY --from=builder /app/out /assets
13 |
14 | COPY ./default.conf /etc/nginx/conf.d/default.conf
--------------------------------------------------------------------------------
/ui/default.conf:
--------------------------------------------------------------------------------
1 | upstream endpoint {
2 | server endpoint:3000;
3 | }
4 |
5 | server {
6 | listen 80;
7 | client_max_body_size 64M;
8 | client_body_buffer_size 64M;
9 |
10 | location / {
11 | root /assets;
12 | try_files $uri $uri/ /index.html;
13 | }
14 |
15 | location /api {
16 | rewrite /api/(.*) /$1 break;
17 | proxy_pass http://endpoint;
18 | proxy_buffering off;
19 | proxy_request_buffering off;
20 | }
21 | }
--------------------------------------------------------------------------------
/ui/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "deptoolkit-ui",
3 | "version": "2.0.0",
4 | "engines": {
5 | "node": "^16"
6 | },
7 | "scripts": {
8 | "dev": "svelte-kit dev --port 8000",
9 | "build": "svelte-kit build",
10 | "preview": "svelte-kit preview",
11 | "check": "svelte-check --tsconfig ./tsconfig.json",
12 | "check:watch": "svelte-check --tsconfig ./tsconfig.json --watch"
13 | },
14 | "devDependencies": {
15 | "@sveltejs/adapter-node": "^1.0.0-next.67",
16 | "@sveltejs/adapter-static": "^1.0.0-next.26",
17 | "@sveltejs/kit": "^1.30.4",
18 | "@types/cookie": "^0.4.0",
19 | "attractions": "^3.3.0",
20 | "postcss": "^8.4.31",
21 | "prettier": "~2.2.1",
22 | "prettier-plugin-svelte": "^2.2.0",
23 | "sass": "^1.37.5",
24 | "svelte": "^4.2.19",
25 | "svelte-check": "^2.0.0",
26 | "svelte-feather-icons": "^3.5.0",
27 | "svelte-preprocess": "^4.7.4",
28 | "tslib": "^2.0.0",
29 | "typescript": "^4.0.0"
30 | },
31 | "type": "module",
32 | "dependencies": {
33 | "@fontsource/fira-mono": "^4.2.2",
34 | "@fontsource/roboto": "^4.5.0",
35 | "@lukeed/uuid": "^2.0.0",
36 | "@types/d3-scale": "^4.0.1",
37 | "cookie": "^0.4.1",
38 | "d3-scale": "^4.0.0",
39 | "d3-time": "^3.0.0",
40 | "d3-time-format": "^4.0.0"
41 | }
42 | }
43 |
--------------------------------------------------------------------------------
/ui/src/app.css:
--------------------------------------------------------------------------------
1 | @import '@fontsource/fira-mono';
2 | @import '@fontsource/roboto';
3 |
4 | :root {
5 | font-family: Roboto, Arial, -apple-system, BlinkMacSystemFont, 'Segoe UI',
6 | Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
7 | --font-mono: 'Fira Mono', monospace;
8 | --pure-white: #ffffff;
9 | --primary-color: #fff;
10 | --secondary-color: #d0dde9;
11 | --tertiary-color: #edf0f8;
12 | --accent-color: #b10dc9;
13 | --muted-grey: #ddd;
14 | --heading-color: rgba(0, 0, 0, 0.7);
15 | --text-color: #444444;
16 | --background-without-opacity: rgba(255, 255, 255, 0.7);
17 | --column-width: 42rem;
18 | --column-margin-top: 4rem;
19 | }
20 |
21 | body {
22 | min-height: 100vh;
23 | margin: 0;
24 | background-color: var(--primary-color);
25 | -moz-osx-font-smoothing: grayscale;
26 | text-rendering: optimizeLegibility;
27 | }
28 |
29 | body::before {
30 | content: '';
31 | width: 80vw;
32 | height: 100vh;
33 | position: absolute;
34 | top: 0;
35 | left: 10vw;
36 | z-index: -1;
37 | opacity: 0.05;
38 | }
39 |
40 | #svelte {
41 | min-height: 100vh;
42 | display: flex;
43 | flex-direction: column;
44 | }
45 |
46 | h1 {
47 | font-weight: 400;
48 | color: var(--accent-color);
49 | }
50 | h2,
51 | p {
52 | font-weight: 400;
53 | color: var(--heading-color);
54 | }
55 |
56 | p {
57 | line-height: 1;
58 | }
59 |
60 | a {
61 | color: var(--accent-color);
62 | text-decoration: none;
63 | }
64 |
65 | a:hover {
66 | text-decoration: underline;
67 | }
68 |
69 | h1 {
70 | font-size: 2rem;
71 | text-align: center;
72 | }
73 |
74 | h2 {
75 | font-size: 1rem;
76 | }
77 |
78 | pre {
79 | font-size: 16px;
80 | font-family: var(--font-mono);
81 | padding: 0.1em;
82 | overflow-x: auto;
83 | color: var(--text-color);
84 | }
85 |
86 | input,
87 | button {
88 | font-size: inherit;
89 | font-family: inherit;
90 | }
91 |
92 | button:focus:not(:focus-visible) {
93 | outline: none;
94 | }
95 |
96 | @media (min-width: 720px) {
97 | h1 {
98 | font-size: 2.4rem;
99 | }
100 | }
101 |
--------------------------------------------------------------------------------
/ui/src/app.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | %svelte.head%
9 |
10 |
11 |
73 | Content dragged-and-dropped into this page will be cross-checked against the
74 | database and will be sent over the internet to the Toolkit.
75 |
76 |
77 | The SHA256 signature of the items to cross-reference are generated upon
78 | reception server-side and looked up in the ledger.
79 |
80 |
Matches will be surfaced below. Non-matches won't appear.