Compare commits
49 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a3e1e62082 | ||
|
|
b90d24e5de | ||
|
|
39fe7b58a4 | ||
|
|
217a0294ac | ||
|
|
2c21d76245 | ||
|
|
83de634df1 | ||
|
|
7861eeeb07 | ||
|
|
9f5baf2aa1 | ||
|
|
8de5223dfc | ||
|
|
64200cc48f | ||
|
|
8ff4c8360f | ||
|
|
13fe2ec7ed | ||
|
|
57bae97ad7 | ||
|
|
c4df0914d9 | ||
|
|
9dd9fe9d9c | ||
|
|
75dbef8196 | ||
|
|
5f53228113 | ||
|
|
24114c1281 | ||
|
|
9702908b4d | ||
|
|
4e96d4f48e | ||
|
|
8f3be09c9a | ||
|
|
8360f6f4f7 | ||
|
|
e9c7412227 | ||
|
|
c105921694 | ||
|
|
20f6283fe4 | ||
|
|
16bdafd44b | ||
|
|
b10e46554f | ||
|
|
334ba3f178 | ||
|
|
2fcd62df7c | ||
|
|
4166d08be1 | ||
|
|
60b70082ba | ||
|
|
c02eb1d02c | ||
|
|
69a802af55 | ||
|
|
14654b1033 | ||
|
|
1265b35d8f | ||
|
|
f5e2ab423d | ||
|
|
05c6407298 | ||
|
|
0d2ac63c61 | ||
|
|
1bd6f39650 | ||
|
|
5478728253 | ||
|
|
4ed8b57674 | ||
|
|
a2500d45d7 | ||
|
|
d7306d3fe9 | ||
|
|
20dc64834a | ||
|
|
16faf90989 | ||
|
|
f320d28f37 | ||
|
|
1a1bff7287 | ||
|
|
9aedb0280b | ||
|
|
75e0d16ac5 |
4
.gitignore
vendored
4
.gitignore
vendored
@@ -15,3 +15,7 @@
|
||||
!/WeddingShare/wwwroot/uploads/default
|
||||
/WeddingShare/wwwroot/uploads/default/*
|
||||
/release notes
|
||||
/WeddingShare/weddingshare.db
|
||||
/WeddingShare/wedding-share.db
|
||||
/WeddingShare/config/wedding-share.db
|
||||
/WeddingShare/wwwroot/thumbnails
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
image: mcr.microsoft.com/dotnet/sdk:latest
|
||||
|
||||
stages:
|
||||
- test
|
||||
- build
|
||||
- push
|
||||
- release
|
||||
@@ -22,12 +23,20 @@ cache:
|
||||
before_script:
|
||||
- 'docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY'
|
||||
|
||||
test:
|
||||
stage: test
|
||||
script:
|
||||
- 'dotnet restore'
|
||||
- 'dotnet test'
|
||||
|
||||
build:
|
||||
stage: build
|
||||
script:
|
||||
- 'docker build -t $CI_REGISTRY_IMAGE:dev -t $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA -f WeddingShare/Dockerfile .'
|
||||
- 'docker push $CI_REGISTRY_IMAGE:dev'
|
||||
- 'docker push $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA'
|
||||
needs:
|
||||
- test
|
||||
|
||||
push_pre_release:
|
||||
variables:
|
||||
@@ -41,6 +50,8 @@ push_pre_release:
|
||||
- 'docker push $CI_REGISTRY_IMAGE:$CI_COMMIT_BRANCH'
|
||||
- 'docker tag $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA $CI_REGISTRY_IMAGE:pre_release'
|
||||
- 'docker push $CI_REGISTRY_IMAGE:pre_release'
|
||||
needs:
|
||||
- build
|
||||
|
||||
push_latest:
|
||||
variables:
|
||||
@@ -53,6 +64,8 @@ push_latest:
|
||||
- 'docker pull $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA'
|
||||
- 'docker tag $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA $CI_REGISTRY_IMAGE:latest'
|
||||
- 'docker push $CI_REGISTRY_IMAGE:latest'
|
||||
needs:
|
||||
- build
|
||||
|
||||
push_tag:
|
||||
variables:
|
||||
@@ -64,6 +77,8 @@ push_tag:
|
||||
- 'docker pull $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA'
|
||||
- 'docker tag $CI_REGISTRY_IMAGE:$CI_COMMIT_SHORT_SHA $CI_REGISTRY_IMAGE:$CI_COMMIT_REF_NAME'
|
||||
- 'docker push $CI_REGISTRY_IMAGE:$CI_COMMIT_REF_NAME'
|
||||
needs:
|
||||
- build
|
||||
|
||||
push_docker_hub:
|
||||
variables:
|
||||
@@ -77,4 +92,6 @@ push_docker_hub:
|
||||
- 'docker tag $CI_REGISTRY_IMAGE:latest $DOCKERHUB_USERNAME/wedding_share:latest'
|
||||
- 'docker push $DOCKERHUB_USERNAME/wedding_share:latest'
|
||||
- 'docker tag $CI_REGISTRY_IMAGE:latest $DOCKERHUB_USERNAME/wedding_share:$CI_COMMIT_REF_NAME'
|
||||
- 'docker push $DOCKERHUB_USERNAME/wedding_share:$CI_COMMIT_REF_NAME'
|
||||
- 'docker push $DOCKERHUB_USERNAME/wedding_share:$CI_COMMIT_REF_NAME'
|
||||
needs:
|
||||
- build
|
||||
674
LICENSE
Normal file
674
LICENSE
Normal file
@@ -0,0 +1,674 @@
|
||||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
The licenses for most software and other practical works are designed
|
||||
to take away your freedom to share and change the works. By contrast,
|
||||
the GNU General Public License is intended to guarantee your freedom to
|
||||
share and change all versions of a program--to make sure it remains free
|
||||
software for all its users. We, the Free Software Foundation, use the
|
||||
GNU General Public License for most of our software; it applies also to
|
||||
any other work released this way by its authors. You can apply it to
|
||||
your programs, too.
|
||||
|
||||
When we speak of free software, we are referring to freedom, not
|
||||
price. Our General Public Licenses are designed to make sure that you
|
||||
have the freedom to distribute copies of free software (and charge for
|
||||
them if you wish), that you receive source code or can get it if you
|
||||
want it, that you can change the software or use pieces of it in new
|
||||
free programs, and that you know you can do these things.
|
||||
|
||||
To protect your rights, we need to prevent others from denying you
|
||||
these rights or asking you to surrender the rights. Therefore, you have
|
||||
certain responsibilities if you distribute copies of the software, or if
|
||||
you modify it: responsibilities to respect the freedom of others.
|
||||
|
||||
For example, if you distribute copies of such a program, whether
|
||||
gratis or for a fee, you must pass on to the recipients the same
|
||||
freedoms that you received. You must make sure that they, too, receive
|
||||
or can get the source code. And you must show them these terms so they
|
||||
know their rights.
|
||||
|
||||
Developers that use the GNU GPL protect your rights with two steps:
|
||||
(1) assert copyright on the software, and (2) offer you this License
|
||||
giving you legal permission to copy, distribute and/or modify it.
|
||||
|
||||
For the developers' and authors' protection, the GPL clearly explains
|
||||
that there is no warranty for this free software. For both users' and
|
||||
authors' sake, the GPL requires that modified versions be marked as
|
||||
changed, so that their problems will not be attributed erroneously to
|
||||
authors of previous versions.
|
||||
|
||||
Some devices are designed to deny users access to install or run
|
||||
modified versions of the software inside them, although the manufacturer
|
||||
can do so. This is fundamentally incompatible with the aim of
|
||||
protecting users' freedom to change the software. The systematic
|
||||
pattern of such abuse occurs in the area of products for individuals to
|
||||
use, which is precisely where it is most unacceptable. Therefore, we
|
||||
have designed this version of the GPL to prohibit the practice for those
|
||||
products. If such problems arise substantially in other domains, we
|
||||
stand ready to extend this provision to those domains in future versions
|
||||
of the GPL, as needed to protect the freedom of users.
|
||||
|
||||
Finally, every program is threatened constantly by software patents.
|
||||
States should not allow patents to restrict development and use of
|
||||
software on general-purpose computers, but in those that do, we wish to
|
||||
avoid the special danger that patents applied to a free program could
|
||||
make it effectively proprietary. To prevent this, the GPL assures that
|
||||
patents cannot be used to render the program non-free.
|
||||
|
||||
The precise terms and conditions for copying, distribution and
|
||||
modification follow.
|
||||
|
||||
TERMS AND CONDITIONS
|
||||
|
||||
0. Definitions.
|
||||
|
||||
"This License" refers to version 3 of the GNU General Public License.
|
||||
|
||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||
works, such as semiconductor masks.
|
||||
|
||||
"The Program" refers to any copyrightable work licensed under this
|
||||
License. Each licensee is addressed as "you". "Licensees" and
|
||||
"recipients" may be individuals or organizations.
|
||||
|
||||
To "modify" a work means to copy from or adapt all or part of the work
|
||||
in a fashion requiring copyright permission, other than the making of an
|
||||
exact copy. The resulting work is called a "modified version" of the
|
||||
earlier work or a work "based on" the earlier work.
|
||||
|
||||
A "covered work" means either the unmodified Program or a work based
|
||||
on the Program.
|
||||
|
||||
To "propagate" a work means to do anything with it that, without
|
||||
permission, would make you directly or secondarily liable for
|
||||
infringement under applicable copyright law, except executing it on a
|
||||
computer or modifying a private copy. Propagation includes copying,
|
||||
distribution (with or without modification), making available to the
|
||||
public, and in some countries other activities as well.
|
||||
|
||||
To "convey" a work means any kind of propagation that enables other
|
||||
parties to make or receive copies. Mere interaction with a user through
|
||||
a computer network, with no transfer of a copy, is not conveying.
|
||||
|
||||
An interactive user interface displays "Appropriate Legal Notices"
|
||||
to the extent that it includes a convenient and prominently visible
|
||||
feature that (1) displays an appropriate copyright notice, and (2)
|
||||
tells the user that there is no warranty for the work (except to the
|
||||
extent that warranties are provided), that licensees may convey the
|
||||
work under this License, and how to view a copy of this License. If
|
||||
the interface presents a list of user commands or options, such as a
|
||||
menu, a prominent item in the list meets this criterion.
|
||||
|
||||
1. Source Code.
|
||||
|
||||
The "source code" for a work means the preferred form of the work
|
||||
for making modifications to it. "Object code" means any non-source
|
||||
form of a work.
|
||||
|
||||
A "Standard Interface" means an interface that either is an official
|
||||
standard defined by a recognized standards body, or, in the case of
|
||||
interfaces specified for a particular programming language, one that
|
||||
is widely used among developers working in that language.
|
||||
|
||||
The "System Libraries" of an executable work include anything, other
|
||||
than the work as a whole, that (a) is included in the normal form of
|
||||
packaging a Major Component, but which is not part of that Major
|
||||
Component, and (b) serves only to enable use of the work with that
|
||||
Major Component, or to implement a Standard Interface for which an
|
||||
implementation is available to the public in source code form. A
|
||||
"Major Component", in this context, means a major essential component
|
||||
(kernel, window system, and so on) of the specific operating system
|
||||
(if any) on which the executable work runs, or a compiler used to
|
||||
produce the work, or an object code interpreter used to run it.
|
||||
|
||||
The "Corresponding Source" for a work in object code form means all
|
||||
the source code needed to generate, install, and (for an executable
|
||||
work) run the object code and to modify the work, including scripts to
|
||||
control those activities. However, it does not include the work's
|
||||
System Libraries, or general-purpose tools or generally available free
|
||||
programs which are used unmodified in performing those activities but
|
||||
which are not part of the work. For example, Corresponding Source
|
||||
includes interface definition files associated with source files for
|
||||
the work, and the source code for shared libraries and dynamically
|
||||
linked subprograms that the work is specifically designed to require,
|
||||
such as by intimate data communication or control flow between those
|
||||
subprograms and other parts of the work.
|
||||
|
||||
The Corresponding Source need not include anything that users
|
||||
can regenerate automatically from other parts of the Corresponding
|
||||
Source.
|
||||
|
||||
The Corresponding Source for a work in source code form is that
|
||||
same work.
|
||||
|
||||
2. Basic Permissions.
|
||||
|
||||
All rights granted under this License are granted for the term of
|
||||
copyright on the Program, and are irrevocable provided the stated
|
||||
conditions are met. This License explicitly affirms your unlimited
|
||||
permission to run the unmodified Program. The output from running a
|
||||
covered work is covered by this License only if the output, given its
|
||||
content, constitutes a covered work. This License acknowledges your
|
||||
rights of fair use or other equivalent, as provided by copyright law.
|
||||
|
||||
You may make, run and propagate covered works that you do not
|
||||
convey, without conditions so long as your license otherwise remains
|
||||
in force. You may convey covered works to others for the sole purpose
|
||||
of having them make modifications exclusively for you, or provide you
|
||||
with facilities for running those works, provided that you comply with
|
||||
the terms of this License in conveying all material for which you do
|
||||
not control copyright. Those thus making or running the covered works
|
||||
for you must do so exclusively on your behalf, under your direction
|
||||
and control, on terms that prohibit them from making any copies of
|
||||
your copyrighted material outside their relationship with you.
|
||||
|
||||
Conveying under any other circumstances is permitted solely under
|
||||
the conditions stated below. Sublicensing is not allowed; section 10
|
||||
makes it unnecessary.
|
||||
|
||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||
|
||||
No covered work shall be deemed part of an effective technological
|
||||
measure under any applicable law fulfilling obligations under article
|
||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||
similar laws prohibiting or restricting circumvention of such
|
||||
measures.
|
||||
|
||||
When you convey a covered work, you waive any legal power to forbid
|
||||
circumvention of technological measures to the extent such circumvention
|
||||
is effected by exercising rights under this License with respect to
|
||||
the covered work, and you disclaim any intention to limit operation or
|
||||
modification of the work as a means of enforcing, against the work's
|
||||
users, your or third parties' legal rights to forbid circumvention of
|
||||
technological measures.
|
||||
|
||||
4. Conveying Verbatim Copies.
|
||||
|
||||
You may convey verbatim copies of the Program's source code as you
|
||||
receive it, in any medium, provided that you conspicuously and
|
||||
appropriately publish on each copy an appropriate copyright notice;
|
||||
keep intact all notices stating that this License and any
|
||||
non-permissive terms added in accord with section 7 apply to the code;
|
||||
keep intact all notices of the absence of any warranty; and give all
|
||||
recipients a copy of this License along with the Program.
|
||||
|
||||
You may charge any price or no price for each copy that you convey,
|
||||
and you may offer support or warranty protection for a fee.
|
||||
|
||||
5. Conveying Modified Source Versions.
|
||||
|
||||
You may convey a work based on the Program, or the modifications to
|
||||
produce it from the Program, in the form of source code under the
|
||||
terms of section 4, provided that you also meet all of these conditions:
|
||||
|
||||
a) The work must carry prominent notices stating that you modified
|
||||
it, and giving a relevant date.
|
||||
|
||||
b) The work must carry prominent notices stating that it is
|
||||
released under this License and any conditions added under section
|
||||
7. This requirement modifies the requirement in section 4 to
|
||||
"keep intact all notices".
|
||||
|
||||
c) You must license the entire work, as a whole, under this
|
||||
License to anyone who comes into possession of a copy. This
|
||||
License will therefore apply, along with any applicable section 7
|
||||
additional terms, to the whole of the work, and all its parts,
|
||||
regardless of how they are packaged. This License gives no
|
||||
permission to license the work in any other way, but it does not
|
||||
invalidate such permission if you have separately received it.
|
||||
|
||||
d) If the work has interactive user interfaces, each must display
|
||||
Appropriate Legal Notices; however, if the Program has interactive
|
||||
interfaces that do not display Appropriate Legal Notices, your
|
||||
work need not make them do so.
|
||||
|
||||
A compilation of a covered work with other separate and independent
|
||||
works, which are not by their nature extensions of the covered work,
|
||||
and which are not combined with it such as to form a larger program,
|
||||
in or on a volume of a storage or distribution medium, is called an
|
||||
"aggregate" if the compilation and its resulting copyright are not
|
||||
used to limit the access or legal rights of the compilation's users
|
||||
beyond what the individual works permit. Inclusion of a covered work
|
||||
in an aggregate does not cause this License to apply to the other
|
||||
parts of the aggregate.
|
||||
|
||||
6. Conveying Non-Source Forms.
|
||||
|
||||
You may convey a covered work in object code form under the terms
|
||||
of sections 4 and 5, provided that you also convey the
|
||||
machine-readable Corresponding Source under the terms of this License,
|
||||
in one of these ways:
|
||||
|
||||
a) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by the
|
||||
Corresponding Source fixed on a durable physical medium
|
||||
customarily used for software interchange.
|
||||
|
||||
b) Convey the object code in, or embodied in, a physical product
|
||||
(including a physical distribution medium), accompanied by a
|
||||
written offer, valid for at least three years and valid for as
|
||||
long as you offer spare parts or customer support for that product
|
||||
model, to give anyone who possesses the object code either (1) a
|
||||
copy of the Corresponding Source for all the software in the
|
||||
product that is covered by this License, on a durable physical
|
||||
medium customarily used for software interchange, for a price no
|
||||
more than your reasonable cost of physically performing this
|
||||
conveying of source, or (2) access to copy the
|
||||
Corresponding Source from a network server at no charge.
|
||||
|
||||
c) Convey individual copies of the object code with a copy of the
|
||||
written offer to provide the Corresponding Source. This
|
||||
alternative is allowed only occasionally and noncommercially, and
|
||||
only if you received the object code with such an offer, in accord
|
||||
with subsection 6b.
|
||||
|
||||
d) Convey the object code by offering access from a designated
|
||||
place (gratis or for a charge), and offer equivalent access to the
|
||||
Corresponding Source in the same way through the same place at no
|
||||
further charge. You need not require recipients to copy the
|
||||
Corresponding Source along with the object code. If the place to
|
||||
copy the object code is a network server, the Corresponding Source
|
||||
may be on a different server (operated by you or a third party)
|
||||
that supports equivalent copying facilities, provided you maintain
|
||||
clear directions next to the object code saying where to find the
|
||||
Corresponding Source. Regardless of what server hosts the
|
||||
Corresponding Source, you remain obligated to ensure that it is
|
||||
available for as long as needed to satisfy these requirements.
|
||||
|
||||
e) Convey the object code using peer-to-peer transmission, provided
|
||||
you inform other peers where the object code and Corresponding
|
||||
Source of the work are being offered to the general public at no
|
||||
charge under subsection 6d.
|
||||
|
||||
A separable portion of the object code, whose source code is excluded
|
||||
from the Corresponding Source as a System Library, need not be
|
||||
included in conveying the object code work.
|
||||
|
||||
A "User Product" is either (1) a "consumer product", which means any
|
||||
tangible personal property which is normally used for personal, family,
|
||||
or household purposes, or (2) anything designed or sold for incorporation
|
||||
into a dwelling. In determining whether a product is a consumer product,
|
||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||
product received by a particular user, "normally used" refers to a
|
||||
typical or common use of that class of product, regardless of the status
|
||||
of the particular user or of the way in which the particular user
|
||||
actually uses, or expects or is expected to use, the product. A product
|
||||
is a consumer product regardless of whether the product has substantial
|
||||
commercial, industrial or non-consumer uses, unless such uses represent
|
||||
the only significant mode of use of the product.
|
||||
|
||||
"Installation Information" for a User Product means any methods,
|
||||
procedures, authorization keys, or other information required to install
|
||||
and execute modified versions of a covered work in that User Product from
|
||||
a modified version of its Corresponding Source. The information must
|
||||
suffice to ensure that the continued functioning of the modified object
|
||||
code is in no case prevented or interfered with solely because
|
||||
modification has been made.
|
||||
|
||||
If you convey an object code work under this section in, or with, or
|
||||
specifically for use in, a User Product, and the conveying occurs as
|
||||
part of a transaction in which the right of possession and use of the
|
||||
User Product is transferred to the recipient in perpetuity or for a
|
||||
fixed term (regardless of how the transaction is characterized), the
|
||||
Corresponding Source conveyed under this section must be accompanied
|
||||
by the Installation Information. But this requirement does not apply
|
||||
if neither you nor any third party retains the ability to install
|
||||
modified object code on the User Product (for example, the work has
|
||||
been installed in ROM).
|
||||
|
||||
The requirement to provide Installation Information does not include a
|
||||
requirement to continue to provide support service, warranty, or updates
|
||||
for a work that has been modified or installed by the recipient, or for
|
||||
the User Product in which it has been modified or installed. Access to a
|
||||
network may be denied when the modification itself materially and
|
||||
adversely affects the operation of the network or violates the rules and
|
||||
protocols for communication across the network.
|
||||
|
||||
Corresponding Source conveyed, and Installation Information provided,
|
||||
in accord with this section must be in a format that is publicly
|
||||
documented (and with an implementation available to the public in
|
||||
source code form), and must require no special password or key for
|
||||
unpacking, reading or copying.
|
||||
|
||||
7. Additional Terms.
|
||||
|
||||
"Additional permissions" are terms that supplement the terms of this
|
||||
License by making exceptions from one or more of its conditions.
|
||||
Additional permissions that are applicable to the entire Program shall
|
||||
be treated as though they were included in this License, to the extent
|
||||
that they are valid under applicable law. If additional permissions
|
||||
apply only to part of the Program, that part may be used separately
|
||||
under those permissions, but the entire Program remains governed by
|
||||
this License without regard to the additional permissions.
|
||||
|
||||
When you convey a copy of a covered work, you may at your option
|
||||
remove any additional permissions from that copy, or from any part of
|
||||
it. (Additional permissions may be written to require their own
|
||||
removal in certain cases when you modify the work.) You may place
|
||||
additional permissions on material, added by you to a covered work,
|
||||
for which you have or can give appropriate copyright permission.
|
||||
|
||||
Notwithstanding any other provision of this License, for material you
|
||||
add to a covered work, you may (if authorized by the copyright holders of
|
||||
that material) supplement the terms of this License with terms:
|
||||
|
||||
a) Disclaiming warranty or limiting liability differently from the
|
||||
terms of sections 15 and 16 of this License; or
|
||||
|
||||
b) Requiring preservation of specified reasonable legal notices or
|
||||
author attributions in that material or in the Appropriate Legal
|
||||
Notices displayed by works containing it; or
|
||||
|
||||
c) Prohibiting misrepresentation of the origin of that material, or
|
||||
requiring that modified versions of such material be marked in
|
||||
reasonable ways as different from the original version; or
|
||||
|
||||
d) Limiting the use for publicity purposes of names of licensors or
|
||||
authors of the material; or
|
||||
|
||||
e) Declining to grant rights under trademark law for use of some
|
||||
trade names, trademarks, or service marks; or
|
||||
|
||||
f) Requiring indemnification of licensors and authors of that
|
||||
material by anyone who conveys the material (or modified versions of
|
||||
it) with contractual assumptions of liability to the recipient, for
|
||||
any liability that these contractual assumptions directly impose on
|
||||
those licensors and authors.
|
||||
|
||||
All other non-permissive additional terms are considered "further
|
||||
restrictions" within the meaning of section 10. If the Program as you
|
||||
received it, or any part of it, contains a notice stating that it is
|
||||
governed by this License along with a term that is a further
|
||||
restriction, you may remove that term. If a license document contains
|
||||
a further restriction but permits relicensing or conveying under this
|
||||
License, you may add to a covered work material governed by the terms
|
||||
of that license document, provided that the further restriction does
|
||||
not survive such relicensing or conveying.
|
||||
|
||||
If you add terms to a covered work in accord with this section, you
|
||||
must place, in the relevant source files, a statement of the
|
||||
additional terms that apply to those files, or a notice indicating
|
||||
where to find the applicable terms.
|
||||
|
||||
Additional terms, permissive or non-permissive, may be stated in the
|
||||
form of a separately written license, or stated as exceptions;
|
||||
the above requirements apply either way.
|
||||
|
||||
8. Termination.
|
||||
|
||||
You may not propagate or modify a covered work except as expressly
|
||||
provided under this License. Any attempt otherwise to propagate or
|
||||
modify it is void, and will automatically terminate your rights under
|
||||
this License (including any patent licenses granted under the third
|
||||
paragraph of section 11).
|
||||
|
||||
However, if you cease all violation of this License, then your
|
||||
license from a particular copyright holder is reinstated (a)
|
||||
provisionally, unless and until the copyright holder explicitly and
|
||||
finally terminates your license, and (b) permanently, if the copyright
|
||||
holder fails to notify you of the violation by some reasonable means
|
||||
prior to 60 days after the cessation.
|
||||
|
||||
Moreover, your license from a particular copyright holder is
|
||||
reinstated permanently if the copyright holder notifies you of the
|
||||
violation by some reasonable means, this is the first time you have
|
||||
received notice of violation of this License (for any work) from that
|
||||
copyright holder, and you cure the violation prior to 30 days after
|
||||
your receipt of the notice.
|
||||
|
||||
Termination of your rights under this section does not terminate the
|
||||
licenses of parties who have received copies or rights from you under
|
||||
this License. If your rights have been terminated and not permanently
|
||||
reinstated, you do not qualify to receive new licenses for the same
|
||||
material under section 10.
|
||||
|
||||
9. Acceptance Not Required for Having Copies.
|
||||
|
||||
You are not required to accept this License in order to receive or
|
||||
run a copy of the Program. Ancillary propagation of a covered work
|
||||
occurring solely as a consequence of using peer-to-peer transmission
|
||||
to receive a copy likewise does not require acceptance. However,
|
||||
nothing other than this License grants you permission to propagate or
|
||||
modify any covered work. These actions infringe copyright if you do
|
||||
not accept this License. Therefore, by modifying or propagating a
|
||||
covered work, you indicate your acceptance of this License to do so.
|
||||
|
||||
10. Automatic Licensing of Downstream Recipients.
|
||||
|
||||
Each time you convey a covered work, the recipient automatically
|
||||
receives a license from the original licensors, to run, modify and
|
||||
propagate that work, subject to this License. You are not responsible
|
||||
for enforcing compliance by third parties with this License.
|
||||
|
||||
An "entity transaction" is a transaction transferring control of an
|
||||
organization, or substantially all assets of one, or subdividing an
|
||||
organization, or merging organizations. If propagation of a covered
|
||||
work results from an entity transaction, each party to that
|
||||
transaction who receives a copy of the work also receives whatever
|
||||
licenses to the work the party's predecessor in interest had or could
|
||||
give under the previous paragraph, plus a right to possession of the
|
||||
Corresponding Source of the work from the predecessor in interest, if
|
||||
the predecessor has it or can get it with reasonable efforts.
|
||||
|
||||
You may not impose any further restrictions on the exercise of the
|
||||
rights granted or affirmed under this License. For example, you may
|
||||
not impose a license fee, royalty, or other charge for exercise of
|
||||
rights granted under this License, and you may not initiate litigation
|
||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||
any patent claim is infringed by making, using, selling, offering for
|
||||
sale, or importing the Program or any portion of it.
|
||||
|
||||
11. Patents.
|
||||
|
||||
A "contributor" is a copyright holder who authorizes use under this
|
||||
License of the Program or a work on which the Program is based. The
|
||||
work thus licensed is called the contributor's "contributor version".
|
||||
|
||||
A contributor's "essential patent claims" are all patent claims
|
||||
owned or controlled by the contributor, whether already acquired or
|
||||
hereafter acquired, that would be infringed by some manner, permitted
|
||||
by this License, of making, using, or selling its contributor version,
|
||||
but do not include claims that would be infringed only as a
|
||||
consequence of further modification of the contributor version. For
|
||||
purposes of this definition, "control" includes the right to grant
|
||||
patent sublicenses in a manner consistent with the requirements of
|
||||
this License.
|
||||
|
||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||
patent license under the contributor's essential patent claims, to
|
||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||
propagate the contents of its contributor version.
|
||||
|
||||
In the following three paragraphs, a "patent license" is any express
|
||||
agreement or commitment, however denominated, not to enforce a patent
|
||||
(such as an express permission to practice a patent or covenant not to
|
||||
sue for patent infringement). To "grant" such a patent license to a
|
||||
party means to make such an agreement or commitment not to enforce a
|
||||
patent against the party.
|
||||
|
||||
If you convey a covered work, knowingly relying on a patent license,
|
||||
and the Corresponding Source of the work is not available for anyone
|
||||
to copy, free of charge and under the terms of this License, through a
|
||||
publicly available network server or other readily accessible means,
|
||||
then you must either (1) cause the Corresponding Source to be so
|
||||
available, or (2) arrange to deprive yourself of the benefit of the
|
||||
patent license for this particular work, or (3) arrange, in a manner
|
||||
consistent with the requirements of this License, to extend the patent
|
||||
license to downstream recipients. "Knowingly relying" means you have
|
||||
actual knowledge that, but for the patent license, your conveying the
|
||||
covered work in a country, or your recipient's use of the covered work
|
||||
in a country, would infringe one or more identifiable patents in that
|
||||
country that you have reason to believe are valid.
|
||||
|
||||
If, pursuant to or in connection with a single transaction or
|
||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||
covered work, and grant a patent license to some of the parties
|
||||
receiving the covered work authorizing them to use, propagate, modify
|
||||
or convey a specific copy of the covered work, then the patent license
|
||||
you grant is automatically extended to all recipients of the covered
|
||||
work and works based on it.
|
||||
|
||||
A patent license is "discriminatory" if it does not include within
|
||||
the scope of its coverage, prohibits the exercise of, or is
|
||||
conditioned on the non-exercise of one or more of the rights that are
|
||||
specifically granted under this License. You may not convey a covered
|
||||
work if you are a party to an arrangement with a third party that is
|
||||
in the business of distributing software, under which you make payment
|
||||
to the third party based on the extent of your activity of conveying
|
||||
the work, and under which the third party grants, to any of the
|
||||
parties who would receive the covered work from you, a discriminatory
|
||||
patent license (a) in connection with copies of the covered work
|
||||
conveyed by you (or copies made from those copies), or (b) primarily
|
||||
for and in connection with specific products or compilations that
|
||||
contain the covered work, unless you entered into that arrangement,
|
||||
or that patent license was granted, prior to 28 March 2007.
|
||||
|
||||
Nothing in this License shall be construed as excluding or limiting
|
||||
any implied license or other defenses to infringement that may
|
||||
otherwise be available to you under applicable patent law.
|
||||
|
||||
12. No Surrender of Others' Freedom.
|
||||
|
||||
If conditions are imposed on you (whether by court order, agreement or
|
||||
otherwise) that contradict the conditions of this License, they do not
|
||||
excuse you from the conditions of this License. If you cannot convey a
|
||||
covered work so as to satisfy simultaneously your obligations under this
|
||||
License and any other pertinent obligations, then as a consequence you may
|
||||
not convey it at all. For example, if you agree to terms that obligate you
|
||||
to collect a royalty for further conveying from those to whom you convey
|
||||
the Program, the only way you could satisfy both those terms and this
|
||||
License would be to refrain entirely from conveying the Program.
|
||||
|
||||
13. Use with the GNU Affero General Public License.
|
||||
|
||||
Notwithstanding any other provision of this License, you have
|
||||
permission to link or combine any covered work with a work licensed
|
||||
under version 3 of the GNU Affero General Public License into a single
|
||||
combined work, and to convey the resulting work. The terms of this
|
||||
License will continue to apply to the part which is the covered work,
|
||||
but the special requirements of the GNU Affero General Public License,
|
||||
section 13, concerning interaction through a network will apply to the
|
||||
combination as such.
|
||||
|
||||
14. Revised Versions of this License.
|
||||
|
||||
The Free Software Foundation may publish revised and/or new versions of
|
||||
the GNU General Public License from time to time. Such new versions will
|
||||
be similar in spirit to the present version, but may differ in detail to
|
||||
address new problems or concerns.
|
||||
|
||||
Each version is given a distinguishing version number. If the
|
||||
Program specifies that a certain numbered version of the GNU General
|
||||
Public License "or any later version" applies to it, you have the
|
||||
option of following the terms and conditions either of that numbered
|
||||
version or of any later version published by the Free Software
|
||||
Foundation. If the Program does not specify a version number of the
|
||||
GNU General Public License, you may choose any version ever published
|
||||
by the Free Software Foundation.
|
||||
|
||||
If the Program specifies that a proxy can decide which future
|
||||
versions of the GNU General Public License can be used, that proxy's
|
||||
public statement of acceptance of a version permanently authorizes you
|
||||
to choose that version for the Program.
|
||||
|
||||
Later license versions may give you additional or different
|
||||
permissions. However, no additional obligations are imposed on any
|
||||
author or copyright holder as a result of your choosing to follow a
|
||||
later version.
|
||||
|
||||
15. Disclaimer of Warranty.
|
||||
|
||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||
|
||||
16. Limitation of Liability.
|
||||
|
||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||
SUCH DAMAGES.
|
||||
|
||||
17. Interpretation of Sections 15 and 16.
|
||||
|
||||
If the disclaimer of warranty and limitation of liability provided
|
||||
above cannot be given local legal effect according to their terms,
|
||||
reviewing courts shall apply local law that most closely approximates
|
||||
an absolute waiver of all civil liability in connection with the
|
||||
Program, unless a warranty or assumption of liability accompanies a
|
||||
copy of the Program in return for a fee.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
How to Apply These Terms to Your New Programs
|
||||
|
||||
If you develop a new program, and you want it to be of the greatest
|
||||
possible use to the public, the best way to achieve this is to make it
|
||||
free software which everyone can redistribute and change under these terms.
|
||||
|
||||
To do so, attach the following notices to the program. It is safest
|
||||
to attach them to the start of each source file to most effectively
|
||||
state the exclusion of warranty; and each file should have at least
|
||||
the "copyright" line and a pointer to where the full notice is found.
|
||||
|
||||
<one line to give the program's name and a brief idea of what it does.>
|
||||
Copyright (C) <year> <name of author>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
Also add information on how to contact you by electronic and paper mail.
|
||||
|
||||
If the program does terminal interaction, make it output a short
|
||||
notice like this when it starts in an interactive mode:
|
||||
|
||||
<program> Copyright (C) <year> <name of author>
|
||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||
This is free software, and you are welcome to redistribute it
|
||||
under certain conditions; type `show c' for details.
|
||||
|
||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||
parts of the General Public License. Of course, your program's commands
|
||||
might be different; for a GUI interface, you would use an "about box".
|
||||
|
||||
You should also get your employer (if you work as a programmer) or school,
|
||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||
For more information on this, and how to apply and follow the GNU GPL, see
|
||||
<https://www.gnu.org/licenses/>.
|
||||
|
||||
The GNU General Public License does not permit incorporating your program
|
||||
into proprietary programs. If your program is a subroutine library, you
|
||||
may consider it more useful to permit linking proprietary applications with
|
||||
the library. If this is what you want to do, use the GNU Lesser General
|
||||
Public License instead of this License. But first, please read
|
||||
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
||||
2
WeddingShare.UnitTests/GlobalUsings.cs
Normal file
2
WeddingShare.UnitTests/GlobalUsings.cs
Normal file
@@ -0,0 +1,2 @@
|
||||
global using NUnit.Framework;
|
||||
global using NSubstitute;
|
||||
14
WeddingShare.UnitTests/Helpers/ConfigurationHelper.cs
Normal file
14
WeddingShare.UnitTests/Helpers/ConfigurationHelper.cs
Normal file
@@ -0,0 +1,14 @@
|
||||
using Microsoft.Extensions.Configuration;
|
||||
|
||||
namespace WeddingShare.UnitTests.Helpers
|
||||
{
|
||||
internal class ConfigurationHelper
|
||||
{
|
||||
public static IConfiguration MockConfiguration(IDictionary<string, string?> settings)
|
||||
{
|
||||
return new ConfigurationBuilder()
|
||||
.AddInMemoryCollection(settings ?? new Dictionary<string, string?>())
|
||||
.Build();
|
||||
}
|
||||
}
|
||||
}
|
||||
23
WeddingShare.UnitTests/Helpers/JsonResponseHelper.cs
Normal file
23
WeddingShare.UnitTests/Helpers/JsonResponseHelper.cs
Normal file
@@ -0,0 +1,23 @@
|
||||
namespace WeddingShare.UnitTests.Helpers
|
||||
{
|
||||
internal class JsonResponseHelper
|
||||
{
|
||||
public static T GetPropertyValue<T>(object? obj, string propertyName, T defaultValue)
|
||||
{
|
||||
if (obj != null)
|
||||
{
|
||||
try
|
||||
{
|
||||
var val = obj?.GetType()?.GetProperty(propertyName)?.GetValue(obj, null);
|
||||
if (val != null)
|
||||
{
|
||||
return (T)val;
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
}
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
}
|
||||
}
|
||||
72
WeddingShare.UnitTests/Helpers/MockData.cs
Normal file
72
WeddingShare.UnitTests/Helpers/MockData.cs
Normal file
@@ -0,0 +1,72 @@
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.Extensions.Primitives;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.UnitTests.Helpers
|
||||
{
|
||||
internal class MockData
|
||||
{
|
||||
public static DefaultHttpContext MockHttpContext(Dictionary<string, StringValues>? form = null, IFormFileCollection? files = null)
|
||||
{
|
||||
var ctx = new DefaultHttpContext()
|
||||
{
|
||||
Session = new MockSession()
|
||||
};
|
||||
|
||||
ctx.Request.Form = new FormCollection(form, files);
|
||||
|
||||
return ctx;
|
||||
}
|
||||
|
||||
public static List<GalleryItemModel> MockGalleryItems(int count = 10, int? galleryId = null, GalleryItemState state = GalleryItemState.All)
|
||||
{
|
||||
var result = new List<GalleryItemModel>();
|
||||
|
||||
for (var i = 0; i < count; i++)
|
||||
{
|
||||
result.Add(MockGalleryItem(galleryId, state));
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public static GalleryItemModel MockGalleryItem(int? galleryId = null, GalleryItemState state = GalleryItemState.All)
|
||||
{
|
||||
var rand = new Random();
|
||||
|
||||
return new GalleryItemModel()
|
||||
{
|
||||
Id = rand.Next(),
|
||||
GalleryId = galleryId != null ? (int)galleryId : rand.Next(),
|
||||
Title = $"{Guid.NewGuid()}.{MockFileExtension()}",
|
||||
UploadedBy = rand.Next(2) % 2 == 0 ? Guid.NewGuid().ToString() : null,
|
||||
State = state == GalleryItemState.All ? (GalleryItemState)rand.Next(2) : state,
|
||||
};
|
||||
}
|
||||
|
||||
public static string MockFileExtension()
|
||||
{
|
||||
var rand = new Random();
|
||||
|
||||
string extension;
|
||||
switch (rand.Next(4))
|
||||
{
|
||||
case 0:
|
||||
extension = "jpg";
|
||||
break;
|
||||
case 1:
|
||||
extension = "jpeg";
|
||||
break;
|
||||
case 2:
|
||||
extension = "png";
|
||||
break;
|
||||
default:
|
||||
extension = "ffff";
|
||||
break;
|
||||
}
|
||||
|
||||
return rand.Next(2) % 2 == 0 ? extension.ToUpper() : extension.ToLower();
|
||||
}
|
||||
}
|
||||
}
|
||||
26
WeddingShare.UnitTests/Helpers/MockHttpMessageHandler.cs
Normal file
26
WeddingShare.UnitTests/Helpers/MockHttpMessageHandler.cs
Normal file
@@ -0,0 +1,26 @@
|
||||
using System.Net;
|
||||
using System.Net.Http.Json;
|
||||
|
||||
namespace WeddingShare.UnitTests.Helpers
|
||||
{
|
||||
public class MockHttpMessageHandler : HttpMessageHandler
|
||||
{
|
||||
private readonly HttpStatusCode _statusCode;
|
||||
private readonly object? _responseContent;
|
||||
|
||||
public MockHttpMessageHandler(HttpStatusCode statusCode, object? responseContent = null)
|
||||
{
|
||||
_statusCode = statusCode;
|
||||
_responseContent = responseContent;
|
||||
}
|
||||
|
||||
protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
|
||||
{
|
||||
return await Task.FromResult(new HttpResponseMessage
|
||||
{
|
||||
StatusCode = _statusCode,
|
||||
Content = JsonContent.Create(_responseContent)
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
60
WeddingShare.UnitTests/Helpers/MockSession.cs
Normal file
60
WeddingShare.UnitTests/Helpers/MockSession.cs
Normal file
@@ -0,0 +1,60 @@
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
using System.Text;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
|
||||
namespace WeddingShare.UnitTests.Helpers
|
||||
{
|
||||
internal class MockSession : ISession
|
||||
{
|
||||
private IDictionary<string, string> Values = new Dictionary<string, string>();
|
||||
|
||||
public bool IsAvailable => throw new NotImplementedException();
|
||||
|
||||
public string Id => throw new NotImplementedException();
|
||||
|
||||
public IEnumerable<string> Keys => throw new NotImplementedException();
|
||||
|
||||
public void Clear()
|
||||
{
|
||||
this.Values.Clear();
|
||||
}
|
||||
|
||||
public Task CommitAsync(CancellationToken cancellationToken = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public Task LoadAsync(CancellationToken cancellationToken = default)
|
||||
{
|
||||
throw new NotImplementedException();
|
||||
}
|
||||
|
||||
public void Remove(string key)
|
||||
{
|
||||
this.Values.Remove(key);
|
||||
}
|
||||
|
||||
public void Set(string key, byte[] value)
|
||||
{
|
||||
if (this.Values.ContainsKey(key))
|
||||
{
|
||||
this.Values.Remove(key);
|
||||
}
|
||||
|
||||
this.Values.Add(key, Encoding.UTF8.GetString(value));
|
||||
}
|
||||
|
||||
public bool TryGetValue(string key, [NotNullWhen(true)] out byte[]? value)
|
||||
{
|
||||
if (this.Values.ContainsKey(key))
|
||||
{
|
||||
value = Encoding.UTF8.GetBytes(this.Values[key]);
|
||||
return true;
|
||||
}
|
||||
|
||||
value = null;
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,279 @@
|
||||
using Microsoft.AspNetCore.Hosting;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.Extensions.Localization;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using Microsoft.Extensions.Primitives;
|
||||
using System.Net;
|
||||
using System.Text.Json;
|
||||
using WeddingShare.Controllers;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
using WeddingShare.Models;
|
||||
using WeddingShare.Models.Database;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class GalleryControllerTests
|
||||
{
|
||||
private readonly IWebHostEnvironment _env = Substitute.For<IWebHostEnvironment>();
|
||||
private readonly IConfigHelper _config = Substitute.For<IConfigHelper>();
|
||||
private readonly IGalleryHelper _gallery = Substitute.For<IGalleryHelper>();
|
||||
private readonly IDatabaseHelper _database = Substitute.For<IDatabaseHelper>();
|
||||
private readonly IFileHelper _file = Substitute.For<IFileHelper>();
|
||||
private readonly IDeviceDetector _deviceDetector = Substitute.For<IDeviceDetector>();
|
||||
private readonly IImageHelper _image = Substitute.For<IImageHelper>();
|
||||
private readonly INotificationHelper _notification = Substitute.For<INotificationHelper>();
|
||||
private readonly ILogger<GalleryController> _logger = Substitute.For<ILogger<GalleryController>>();
|
||||
private readonly IStringLocalizer<GalleryController> _localizer = Substitute.For<IStringLocalizer<GalleryController>>();
|
||||
|
||||
public GalleryControllerTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
_env.WebRootPath.Returns("/app/wwwroot");
|
||||
|
||||
_database.GetGallery("default").Returns(Task.FromResult<GalleryModel?>(new GalleryModel()
|
||||
{
|
||||
Id = 1,
|
||||
Name = "default",
|
||||
SecretKey = "password",
|
||||
ApprovedItems = 32,
|
||||
PendingItems = 50,
|
||||
TotalItems = 72
|
||||
}));
|
||||
_database.GetGallery("missing").Returns(Task.FromResult<GalleryModel?>(null));
|
||||
_database.AddGallery(Arg.Any<GalleryModel>()).Returns(Task.FromResult<GalleryModel?>(new GalleryModel()
|
||||
{
|
||||
Id = 101,
|
||||
Name = "missing",
|
||||
SecretKey = "123456",
|
||||
ApprovedItems = 0,
|
||||
PendingItems = 0,
|
||||
TotalItems = 0
|
||||
}));
|
||||
_database.AddGalleryItem(Arg.Any<GalleryItemModel>()).Returns(Task.FromResult<GalleryItemModel?>(MockData.MockGalleryItem()));
|
||||
|
||||
_database.GetAllGalleryItems(Arg.Any<int>(), GalleryItemState.All).Returns(Task.FromResult(MockData.MockGalleryItems(10, 1, GalleryItemState.All)));
|
||||
_database.GetAllGalleryItems(Arg.Any<int>(), GalleryItemState.Pending).Returns(Task.FromResult(MockData.MockGalleryItems(10, 1, GalleryItemState.Pending)));
|
||||
_database.GetAllGalleryItems(Arg.Any<int>(), GalleryItemState.Approved).Returns(Task.FromResult(MockData.MockGalleryItems(10, 1, GalleryItemState.Approved)));
|
||||
|
||||
_gallery.GetSecretKey(Arg.Any<string>()).Returns("password");
|
||||
_gallery.GetSecretKey("missing").Returns("123456");
|
||||
|
||||
_config.GetOrDefault("Settings", "Allowed_File_Types", Arg.Any<string>()).Returns(".jpg,.jpeg,.png");
|
||||
_config.GetOrDefault("Settings", "Default_Gallery_View", Arg.Any<int>()).Returns((int)ViewMode.Default);
|
||||
_config.GetOrDefault("Settings", "Require_Review", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Settings", "Max_File_Size_Mb", Arg.Any<int>()).Returns(10);
|
||||
|
||||
_notification.Send(Arg.Any<string>(), Arg.Any<string>()).Returns(Task.FromResult(true));
|
||||
|
||||
_localizer[Arg.Any<string>()].Returns(new LocalizedString("UnitTest", "UnitTest"));
|
||||
}
|
||||
|
||||
[TestCase(DeviceType.Desktop, "default", "password", ViewMode.Default, GalleryOrder.None)]
|
||||
[TestCase(DeviceType.Mobile, "blaa", "123456", ViewMode.Presentation, GalleryOrder.UploadedAsc)]
|
||||
[TestCase(DeviceType.Tablet, "missing", "123456", ViewMode.Slideshow, GalleryOrder.NameAsc)]
|
||||
public async Task GalleryController_Index(DeviceType deviceType, string id, string? key, ViewMode? mode, GalleryOrder order)
|
||||
{
|
||||
_deviceDetector.ParseDeviceType(Arg.Any<string>()).Returns(deviceType);
|
||||
_config.GetOrDefault("Settings", "Single_Gallery_Mode", Arg.Any<bool>()).Returns(false);
|
||||
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext();
|
||||
|
||||
ViewResult actual = (ViewResult)await controller.Index(id, key, mode, order);
|
||||
Assert.That(actual, Is.TypeOf<ViewResult>());
|
||||
Assert.That(actual?.Model, Is.Not.Null);
|
||||
|
||||
PhotoGallery model = (PhotoGallery)actual.Model;
|
||||
Assert.That(model?.GalleryId, Is.EqualTo(id));
|
||||
Assert.That(model.GalleryPath, Is.EqualTo($"/uploads/{id}"));
|
||||
Assert.That(model.ThumbnailsPath, Is.EqualTo($"/thumbnails"));
|
||||
Assert.That(model.ViewMode, Is.EqualTo(mode));
|
||||
Assert.That(model?.FileUploader?.GalleryId, Is.EqualTo(id));
|
||||
Assert.That(model?.FileUploader?.SecretKey, Is.EqualTo(key));
|
||||
Assert.That(model?.FileUploader?.UploadUrl, Is.EqualTo("/Gallery/UploadImage"));
|
||||
}
|
||||
|
||||
[TestCase(DeviceType.Desktop, ViewMode.Default, GalleryOrder.None)]
|
||||
[TestCase(DeviceType.Mobile, ViewMode.Presentation, GalleryOrder.UploadedAsc)]
|
||||
[TestCase(DeviceType.Tablet, ViewMode.Slideshow, GalleryOrder.NameAsc)]
|
||||
public async Task GalleryController_Index_SingleGalleryMode(DeviceType deviceType, ViewMode? mode, GalleryOrder order)
|
||||
{
|
||||
_deviceDetector.ParseDeviceType(Arg.Any<string>()).Returns(deviceType);
|
||||
_config.GetOrDefault("Settings", "Single_Gallery_Mode", Arg.Any<bool>()).Returns(true);
|
||||
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext();
|
||||
|
||||
ViewResult actual = (ViewResult)await controller.Index("default", "password", mode, order);
|
||||
Assert.That(actual, Is.TypeOf<ViewResult>());
|
||||
Assert.That(actual?.Model, Is.Not.Null);
|
||||
|
||||
PhotoGallery model = (PhotoGallery)actual.Model;
|
||||
Assert.That(model?.GalleryId, Is.EqualTo("default"));
|
||||
Assert.That(model.GalleryPath, Is.EqualTo($"/uploads/default"));
|
||||
Assert.That(model.ThumbnailsPath, Is.EqualTo($"/thumbnails"));
|
||||
Assert.That(model.ViewMode, Is.EqualTo(mode));
|
||||
Assert.That(model?.FileUploader?.GalleryId, Is.EqualTo("default"));
|
||||
Assert.That(model?.FileUploader?.SecretKey, Is.EqualTo("password"));
|
||||
Assert.That(model?.FileUploader?.UploadUrl, Is.EqualTo("/Gallery/UploadImage"));
|
||||
}
|
||||
|
||||
[TestCase(true, 1, null)]
|
||||
[TestCase(true, 3, "Bob")]
|
||||
[TestCase(false, 1, "")]
|
||||
[TestCase(false, 3, "Unit Testing")]
|
||||
public async Task GalleryController_UploadImage(bool requiresReview, int fileCount, string? uploadedBy)
|
||||
{
|
||||
_config.GetOrDefault("Settings", "Require_Review", Arg.Any<bool>()).Returns(requiresReview);
|
||||
|
||||
var files = new FormFileCollection();
|
||||
for (var i = 0; i < fileCount; i++)
|
||||
{
|
||||
files.Add(new FormFile(null, 0, 0, "TestFile_001", $"{Guid.NewGuid()}.jpg"));
|
||||
}
|
||||
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(
|
||||
form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", "default" },
|
||||
{ "SecretKey", "password" },
|
||||
{ "UploadedBy", uploadedBy }
|
||||
},
|
||||
files: files);
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.True);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(files.Count));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploadedBy", string.Empty), Is.EqualTo(!string.IsNullOrWhiteSpace(uploadedBy) ? uploadedBy : string.Empty));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.EqualTo(0));
|
||||
}
|
||||
|
||||
[TestCase(null)]
|
||||
[TestCase("")]
|
||||
public async Task GalleryController_UploadImage_InvalidGallery(string? id)
|
||||
{
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", id }
|
||||
});
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.False);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(0));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.GreaterThan(0));
|
||||
}
|
||||
|
||||
[TestCase(null)]
|
||||
[TestCase("")]
|
||||
public async Task GalleryController_UploadImage_InvalidSecretKey(string? key)
|
||||
{
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", "default" },
|
||||
{ "SecretKey", key }
|
||||
});
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.False);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(0));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.GreaterThan(0));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryController_UploadImage_MissingGallery()
|
||||
{
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", Guid.NewGuid().ToString() }
|
||||
});
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.False);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(0));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.GreaterThan(0));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryController_UploadImage_NoFiles()
|
||||
{
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", "default" },
|
||||
{ "SecretKey", "password" }
|
||||
});
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.False);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(0));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.GreaterThan(0));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryController_UploadImage_FileTooBig()
|
||||
{
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(
|
||||
form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", "default" },
|
||||
{ "SecretKey", "password" }
|
||||
},
|
||||
files: new FormFileCollection() {
|
||||
new FormFile(null, 0, int.MaxValue, "TestFile_001", $"{Guid.NewGuid()}.jpg")
|
||||
});
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.False);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(0));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.GreaterThan(0));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryController_UploadImage_InvalidFileType()
|
||||
{
|
||||
var controller = new GalleryController(_env, _config, _database, _file, _gallery, _deviceDetector, _image, _notification, _logger, _localizer);
|
||||
controller.ControllerContext.HttpContext = MockData.MockHttpContext(
|
||||
form: new Dictionary<string, StringValues>
|
||||
{
|
||||
{ "Id", "default" },
|
||||
{ "SecretKey", "password" }
|
||||
},
|
||||
files: new FormFileCollection() {
|
||||
new FormFile(null, 0, int.MaxValue, "TestFile_001", $"{Guid.NewGuid()}.blaa")
|
||||
});
|
||||
|
||||
JsonResult actual = (JsonResult)await controller.UploadImage();
|
||||
Assert.That(actual, Is.TypeOf<JsonResult>());
|
||||
Assert.That(actual?.Value, Is.Not.Null);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "success", false), Is.False);
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "uploaded", 0), Is.EqualTo(0));
|
||||
Assert.That(JsonResponseHelper.GetPropertyValue(actual.Value, "errors", new List<string>()).Count, Is.GreaterThan(0));
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,64 @@
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using WeddingShare.Controllers;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class HomeControllerTests
|
||||
{
|
||||
private readonly IConfigHelper _config = Substitute.For<IConfigHelper>();
|
||||
private readonly IGalleryHelper _gallery = Substitute.For<IGalleryHelper>();
|
||||
private readonly IDeviceDetector _deviceDetector = Substitute.For<IDeviceDetector>();
|
||||
private readonly ILogger<HomeController> _logger = Substitute.For<ILogger<HomeController>>();
|
||||
|
||||
public HomeControllerTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
}
|
||||
|
||||
[TestCase(DeviceType.Desktop, true, "", true)]
|
||||
[TestCase(DeviceType.Desktop, false, "", false)]
|
||||
[TestCase(DeviceType.Mobile, true, "", true)]
|
||||
[TestCase(DeviceType.Mobile, false, "", false)]
|
||||
[TestCase(DeviceType.Desktop, true, "123456", false)]
|
||||
[TestCase(DeviceType.Desktop, false, "Abc123!", false)]
|
||||
[TestCase(DeviceType.Mobile, true, "abc123!", false)]
|
||||
[TestCase(DeviceType.Mobile, false, "adsbsds", false)]
|
||||
public async Task HomeController_Index(DeviceType deviceType, bool singleGalleryMode, string secretKey, bool isRedirect)
|
||||
{
|
||||
_deviceDetector.ParseDeviceType(Arg.Any<string>()).Returns(deviceType);
|
||||
_config.GetOrDefault("Settings", "Single_Gallery_Mode", Arg.Any<bool>()).Returns(singleGalleryMode);
|
||||
_gallery.GetSecretKey(Arg.Any<string>()).Returns(secretKey);
|
||||
|
||||
var controller = new HomeController(_config, _gallery, _deviceDetector, _logger);
|
||||
controller.ControllerContext.HttpContext = new DefaultHttpContext()
|
||||
{
|
||||
Session = new MockSession()
|
||||
};
|
||||
|
||||
if (!isRedirect)
|
||||
{
|
||||
ViewResult actual = (ViewResult)await controller.Index();
|
||||
Assert.That(actual, Is.TypeOf<ViewResult>());
|
||||
}
|
||||
else
|
||||
{
|
||||
RedirectToActionResult actual = (RedirectToActionResult)await controller.Index();
|
||||
Assert.That(actual, Is.TypeOf<RedirectToActionResult>());
|
||||
Assert.That(actual.Permanent, Is.EqualTo(false));
|
||||
Assert.That(actual.ControllerName, Is.EqualTo("Gallery"));
|
||||
Assert.That(actual.ActionName, Is.EqualTo("Index"));
|
||||
Assert.That(actual.RouteValues, Is.Null);
|
||||
Assert.That(actual.Fragment, Is.Null);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
39
WeddingShare.UnitTests/Tests/Extensions/StringExtensions.cs
Normal file
39
WeddingShare.UnitTests/Tests/Extensions/StringExtensions.cs
Normal file
@@ -0,0 +1,39 @@
|
||||
using WeddingShare.Extensions;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class StringExtensionsTests
|
||||
{
|
||||
public StringExtensionsTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
}
|
||||
|
||||
[TestCase("abcdefgh", new [] { "g" }, "+", "abcdef+h")]
|
||||
[TestCase("abcdefgh", new [] { "b", "e" }, "-", "a-cd-fgh")]
|
||||
[TestCase("abcdefgh", new [] { "bc", "e" }, "-", "a-d-fgh")]
|
||||
[TestCase("abcdefgeh", new [] { "bc", "e" }, "$", "a$d$fg$h")]
|
||||
[TestCase("abcdefgeh", new[] { "ee", "z" }, "$", "abcdefgeh")]
|
||||
public void StringExtensions_Replace(string input, string[] oldChars, string newChar, string expected)
|
||||
{
|
||||
var actual = input.Replace(oldChars, newChar);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("abcdefge", "c", "abdefge")]
|
||||
[TestCase("abcdefge", "cd", "abefge")]
|
||||
[TestCase("abcdefge", "def", "abcge")]
|
||||
[TestCase("abcdefge", "e", "abcdfg")]
|
||||
[TestCase("abcdefg", "g", "abcdef")]
|
||||
[TestCase("abcdefg", "a", "bcdefg")]
|
||||
public void StringExtensions_Remove(string input, string value, string expected)
|
||||
{
|
||||
var actual = input.Remove(value);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
}
|
||||
}
|
||||
178
WeddingShare.UnitTests/Tests/Helpers/ConfigHelper.cs
Normal file
178
WeddingShare.UnitTests/Tests/Helpers/ConfigHelper.cs
Normal file
@@ -0,0 +1,178 @@
|
||||
using Microsoft.Extensions.Configuration;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class ConfigHelperTests
|
||||
{
|
||||
private readonly IConfiguration _configuration;
|
||||
private readonly IEnvironmentWrapper _environment = Substitute.For<IEnvironmentWrapper>();
|
||||
private readonly ILogger<ConfigHelper> _logger = Substitute.For<ILogger<ConfigHelper>>();
|
||||
|
||||
public ConfigHelperTests()
|
||||
{
|
||||
_environment.GetEnvironmentVariable("VERSION").Returns("v2.0.0");
|
||||
_environment.GetEnvironmentVariable("ENVKEY_1").Returns("EnvValue1");
|
||||
_environment.GetEnvironmentVariable("ENVKEY_2").Returns("EnvValue2");
|
||||
_environment.GetEnvironmentVariable("ENVKEY_3").Returns("EnvValue3");
|
||||
|
||||
_configuration = ConfigurationHelper.MockConfiguration(new Dictionary<string, string?>()
|
||||
{
|
||||
{ "Release:Version", "v1.0.0" },
|
||||
{ "Release:Plugin:Version", "v3.0.0" },
|
||||
|
||||
{ "String1:Key1", "Value1" },
|
||||
{ "String1:Key2", "Value2" },
|
||||
{ "String2:Key1", "Value3" },
|
||||
|
||||
{ "Int1:Key1", "1" },
|
||||
{ "Int1:Key2", "2" },
|
||||
{ "Int2:Key1", "3" },
|
||||
|
||||
{ "Long1:Key1", "4" },
|
||||
{ "Long1:Key2", "5" },
|
||||
{ "Long2:Key1", "6" },
|
||||
|
||||
{ "Decimal1:Key1", "4.12" },
|
||||
{ "Decimal1:Key2", "5.45" },
|
||||
{ "Decimal2:Key1", "6.733" },
|
||||
|
||||
{ "Double1:Key1", "4.12" },
|
||||
{ "Double1:Key2", "5.45" },
|
||||
{ "Double2:Key1", "6.733" },
|
||||
|
||||
{ "Boolean1:Key1", "true" },
|
||||
{ "Boolean1:Key2", "false" },
|
||||
{ "Boolean2:Key1", "true" },
|
||||
|
||||
{ "DateTime1:Key1", "1987-11-20 08:00:00" },
|
||||
{ "DateTime1:Key2", "2000-08-12 12:00:00" },
|
||||
{ "DateTime2:Key1", "2018-01-01 20:30:10" },
|
||||
});
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
}
|
||||
|
||||
[TestCase("ENVKEY:1", "EnvValue1")]
|
||||
[TestCase("ENVKEY:2", "EnvValue2")]
|
||||
[TestCase("ENVKEY:3", "EnvValue3")]
|
||||
[TestCase("ENVKEY:4", null)]
|
||||
[TestCase("VERSION", "v2.0.0")]
|
||||
public void ConfigHelper_GetEnvironmentVariable(string section, string? expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetEnvironmentVariable(section);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("String1", null, "Key1", "Value1")]
|
||||
[TestCase("String1", null, "Key2", "Value2")]
|
||||
[TestCase("String2", null, "Key1", "Value3")]
|
||||
[TestCase("String2", null, "Key2", null)]
|
||||
[TestCase("Release", null, "Version", "v1.0.0")]
|
||||
public void ConfigHelper_GetConfigValue(string section, string? subsection, string key, string? expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetConfigValue(section, subsection, key);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("String1", "Key1", "Value1")]
|
||||
[TestCase("String1", "Key2", "Value2")]
|
||||
[TestCase("String2", "Key1", "Value3")]
|
||||
[TestCase("String2", "Key2", null)]
|
||||
[TestCase("Release", "Version", "v1.0.0")]
|
||||
public void ConfigHelper_Get(string section, string key, string? expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).Get(section, key);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("String1", "Key1", "Default", "Value1")]
|
||||
[TestCase("String1", "Key2", "Default", "Value2")]
|
||||
[TestCase("String2", "Key1", "Default", "Value3")]
|
||||
[TestCase("String2", "Key2", "Default", "Default")]
|
||||
[TestCase("Release", "Version", "v0.0.0", "v1.0.0")]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, string defaultValue, string expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("String1", null, "Key1", "Default", "Value1")]
|
||||
[TestCase("String1", null, "Key2", "Default", "Value2")]
|
||||
[TestCase("String2", null, "Key1", "Default", "Value3")]
|
||||
[TestCase("String2", null, "Key2", "Default", "Default")]
|
||||
[TestCase("Release", null, "Version", "v0.0.0", "v1.0.0")]
|
||||
[TestCase("Release", "Plugin", "Version", "v0.0.0", "v3.0.0")]
|
||||
public void ConfigHelper_GetOrDefault(string section, string? subsection, string key, string defaultValue, string expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, subsection, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("Int1", "Key1", 999, 1)]
|
||||
[TestCase("Int1", "Key2", 999, 2)]
|
||||
[TestCase("Int2", "Key1", 999, 3)]
|
||||
[TestCase("Int2", "Key2", 999, 999)]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, int defaultValue, int expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("Long1", "Key1", 999, 4)]
|
||||
[TestCase("Long1", "Key2", 999, 5)]
|
||||
[TestCase("Long2", "Key1", 999, 6)]
|
||||
[TestCase("Long2", "Key2", 999, 999)]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, long defaultValue, long expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("Decimal1", "Key1", 999, 4.12)]
|
||||
[TestCase("Decimal1", "Key2", 999, 5.45)]
|
||||
[TestCase("Decimal2", "Key1", 999, 6.733)]
|
||||
[TestCase("Decimal2", "Key2", 999, 999)]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, decimal defaultValue, decimal expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("Double1", "Key1", 999, 4.12)]
|
||||
[TestCase("Double1", "Key2", 999, 5.45)]
|
||||
[TestCase("Double2", "Key1", 999, 6.733)]
|
||||
[TestCase("Double2", "Key2", 999, 999)]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, double defaultValue, double expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("Boolean1", "Key1", false, true)]
|
||||
[TestCase("Boolean1", "Key2", false, false)]
|
||||
[TestCase("Boolean2", "Key1", false, true)]
|
||||
[TestCase("Boolean2", "Key2", true, true)]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, bool defaultValue, bool expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase("DateTime1", "Key1", null, "1987-11-20 08:00:00")]
|
||||
[TestCase("DateTime1", "Key2", null, "2000-08-12 12:00:00")]
|
||||
[TestCase("DateTime2", "Key1", null, "2018-01-01 20:30:10")]
|
||||
[TestCase("DateTime2", "Key2", null, null)]
|
||||
[TestCase("DateTime3", "Key3", "2350-05-05 00:05:12", "2350-05-05 00:05:12")]
|
||||
public void ConfigHelper_GetOrDefault(string section, string key, DateTime? defaultValue, string? expected)
|
||||
{
|
||||
var actual = new ConfigHelper(_environment, _configuration, _logger).GetOrDefault(section, key, defaultValue);
|
||||
Assert.That(actual, Is.EqualTo(!string.IsNullOrWhiteSpace(expected) ? DateTime.Parse(expected) : null));
|
||||
}
|
||||
}
|
||||
}
|
||||
23
WeddingShare.UnitTests/Tests/Helpers/DeviceDetector.cs
Normal file
23
WeddingShare.UnitTests/Tests/Helpers/DeviceDetector.cs
Normal file
@@ -0,0 +1,23 @@
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class DeviceDetectorTests
|
||||
{
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
}
|
||||
|
||||
[TestCase("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/131.0.0.0 Safari/537.36", DeviceType.Desktop)]
|
||||
[TestCase("Mozilla/5.0 (Linux; Android 7.0; SM-T827R4 Build/NRD90M) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.116 Safari/537.36", DeviceType.Tablet)]
|
||||
[TestCase("Mozilla/5.0 (Linux; Android 13; SM-G998B) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Mobile Safari/537.36", DeviceType.Mobile)]
|
||||
[TestCase("Googlebot-Image/1.0", DeviceType.Desktop)]
|
||||
public async Task DeviceDetector_ParseDeviceType(string userAgent, DeviceType expected)
|
||||
{
|
||||
var actual = await new DeviceDetector().ParseDeviceType(userAgent);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
}
|
||||
}
|
||||
108
WeddingShare.UnitTests/Tests/Helpers/EmailHelper.cs
Normal file
108
WeddingShare.UnitTests/Tests/Helpers/EmailHelper.cs
Normal file
@@ -0,0 +1,108 @@
|
||||
using System.Net.Mail;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using NSubstitute;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class EmailHelperTests
|
||||
{
|
||||
private readonly IConfigHelper _config = Substitute.For<IConfigHelper>();
|
||||
private readonly ISmtpClientWrapper _smtp = Substitute.For<ISmtpClientWrapper>();
|
||||
private readonly ILogger<EmailHelper> _logger = Substitute.For<ILogger<EmailHelper>>();
|
||||
|
||||
public EmailHelperTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
_smtp.SendMailAsync(Arg.Any<SmtpClient>(), Arg.Any<MailMessage>()).Returns(Task.FromResult(true));
|
||||
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Enabled", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Recipient", Arg.Any<string>()).Returns("unit@test.com");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Host", Arg.Any<string>()).Returns("https://unit.test.com/");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Port", Arg.Any<int>()).Returns(999);
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Username", Arg.Any<string>()).Returns("Unit");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Password", Arg.Any<string>()).Returns("Test");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "From", Arg.Any<string>()).Returns("unittest@test.com");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "DisplayName", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "UseSSL", Arg.Any<bool>()).Returns(true);
|
||||
}
|
||||
|
||||
[TestCase("unit", "test")]
|
||||
public async Task EmailHelper_Success(string title, string message)
|
||||
{
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send(title, message);
|
||||
Assert.That(actual, Is.EqualTo(true));
|
||||
}
|
||||
|
||||
[TestCase(true, true)]
|
||||
[TestCase(false, false)]
|
||||
public async Task EmailHelper_Enabled(bool enabled, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Enabled", Arg.Any<bool>()).Returns(enabled);
|
||||
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("", false)]
|
||||
[TestCase("blaa@blaa.com", true)]
|
||||
public async Task EmailHelper_Recipient(string recipient, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Recipient", Arg.Any<string>()).Returns(recipient);
|
||||
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("", false)]
|
||||
[TestCase("https://unit.test.com/", true)]
|
||||
public async Task EmailHelper_Host(string host, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Host", Arg.Any<string>()).Returns(host);
|
||||
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(-100, false)]
|
||||
[TestCase(-1, false)]
|
||||
[TestCase(0, false)]
|
||||
[TestCase(1, true)]
|
||||
public async Task EmailHelper_Port(int port, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Port", Arg.Any<int>()).Returns(port);
|
||||
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("", false)]
|
||||
[TestCase("blaa@blaa.com", true)]
|
||||
public async Task EmailHelper_From(string from, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "From", Arg.Any<string>()).Returns(from);
|
||||
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, true)]
|
||||
[TestCase("", true)]
|
||||
[TestCase("UnitTest", true)]
|
||||
public async Task EmailHelper_DisplayName(string displayName, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "DisplayName", Arg.Any<string>()).Returns(displayName);
|
||||
|
||||
var actual = await new EmailHelper(_config, _smtp, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
}
|
||||
}
|
||||
110
WeddingShare.UnitTests/Tests/Helpers/GalleryHelper.cs
Normal file
110
WeddingShare.UnitTests/Tests/Helpers/GalleryHelper.cs
Normal file
@@ -0,0 +1,110 @@
|
||||
using Microsoft.Extensions.Logging;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Models.Database;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class GalleryHelperTests
|
||||
{
|
||||
private readonly IDatabaseHelper _database = Substitute.For<IDatabaseHelper>();
|
||||
|
||||
public GalleryHelperTests()
|
||||
{
|
||||
_database.GetGallery("Gallery1").Returns(new GalleryModel() { SecretKey = "001" });
|
||||
_database.GetGallery("Gallery2").Returns(new GalleryModel() { SecretKey = "002" });
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryHelper_GetSecretKey_DefaultEnvKey()
|
||||
{
|
||||
var environment = Substitute.For<IEnvironmentWrapper>();
|
||||
environment.GetEnvironmentVariable("SECRET_KEY").Returns("123");
|
||||
|
||||
var configuration = ConfigurationHelper.MockConfiguration(new Dictionary<string, string?>()
|
||||
{
|
||||
{ "Secret_Key_Gallery2", "002" }
|
||||
});
|
||||
|
||||
var config = new ConfigHelper(environment, configuration, Substitute.For<ILogger<ConfigHelper>>());
|
||||
|
||||
var actual = await new GalleryHelper(config, _database).GetSecretKey("Gallery3");
|
||||
Assert.That(actual, Is.EqualTo("123"));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryHelper_GetSecretKey_GalleryEnvKey()
|
||||
{
|
||||
var environment = Substitute.For<IEnvironmentWrapper>();
|
||||
environment.GetEnvironmentVariable("SECRET_KEY").Returns("123");
|
||||
environment.GetEnvironmentVariable("SECRET_KEY_GALLERY1").Returns("001");
|
||||
|
||||
var configuration = ConfigurationHelper.MockConfiguration(new Dictionary<string, string?>()
|
||||
{
|
||||
{ "Secret_Key_Gallery2", "002" }
|
||||
});
|
||||
|
||||
var config = new ConfigHelper(environment, configuration, Substitute.For<ILogger<ConfigHelper>>());
|
||||
|
||||
var actual = await new GalleryHelper(config, _database).GetSecretKey("Gallery1");
|
||||
Assert.That(actual, Is.EqualTo("001"));
|
||||
}
|
||||
|
||||
[TestCase("Gallery1", "001")]
|
||||
[TestCase("Gallery2", "002")]
|
||||
[TestCase("Gallery3", null)]
|
||||
public async Task GalleryHelper_GetSecretKey_Database(string galleryId, string key)
|
||||
{
|
||||
var environment = Substitute.For<IEnvironmentWrapper>();
|
||||
environment.GetEnvironmentVariable(Arg.Any<string>()).Returns(string.Empty);
|
||||
|
||||
var configuration = ConfigurationHelper.MockConfiguration(new Dictionary<string, string?>());
|
||||
|
||||
var config = new ConfigHelper(environment, configuration, Substitute.For<ILogger<ConfigHelper>>());
|
||||
|
||||
var actual = await new GalleryHelper(config, _database).GetSecretKey(galleryId);
|
||||
Assert.That(actual, Is.EqualTo(key));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryHelper_GetConfig_DefaultEnvKey()
|
||||
{
|
||||
var environment = Substitute.For<IEnvironmentWrapper>();
|
||||
environment.GetEnvironmentVariable("SECRET_KEY").Returns("123");
|
||||
|
||||
var configuration = ConfigurationHelper.MockConfiguration(new Dictionary<string, string?>()
|
||||
{
|
||||
{ "Secret_Key_Gallery2", "002" }
|
||||
});
|
||||
|
||||
var config = new ConfigHelper(environment, configuration, Substitute.For<ILogger<ConfigHelper>>());
|
||||
|
||||
var actual = new GalleryHelper(config, _database).GetConfig("Gallery3", "Secret_Key");
|
||||
Assert.That(actual, Is.EqualTo("123"));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task GalleryHelper_GetConfig_GalleryEnvKey()
|
||||
{
|
||||
var environment = Substitute.For<IEnvironmentWrapper>();
|
||||
environment.GetEnvironmentVariable("SECRET_KEY").Returns("123");
|
||||
environment.GetEnvironmentVariable("SECRET_KEY_GALLERY1").Returns("001");
|
||||
|
||||
var configuration = ConfigurationHelper.MockConfiguration(new Dictionary<string, string?>()
|
||||
{
|
||||
{ "Secret_Key_Gallery2", "002" }
|
||||
});
|
||||
|
||||
var config = new ConfigHelper(environment, configuration, Substitute.For<ILogger<ConfigHelper>>());
|
||||
|
||||
var actual = new GalleryHelper(config, _database).GetConfig("Gallery1", "Secret_Key");
|
||||
Assert.That(actual, Is.EqualTo("001"));
|
||||
}
|
||||
}
|
||||
}
|
||||
88
WeddingShare.UnitTests/Tests/Helpers/GotifyHelper.cs
Normal file
88
WeddingShare.UnitTests/Tests/Helpers/GotifyHelper.cs
Normal file
@@ -0,0 +1,88 @@
|
||||
using System.Net;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class GotifyHelperTests
|
||||
{
|
||||
private readonly IConfigHelper _config = Substitute.For<IConfigHelper>();
|
||||
private readonly IHttpClientFactory _clientFactory = Substitute.For<IHttpClientFactory>();
|
||||
private readonly ILogger<GotifyHelper> _logger = Substitute.For<ILogger<GotifyHelper>>();
|
||||
|
||||
public GotifyHelperTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
var client = new HttpClient(new MockHttpMessageHandler(HttpStatusCode.OK));
|
||||
client.BaseAddress = new Uri("https://unit.test.com/");
|
||||
|
||||
_clientFactory.CreateClient(Arg.Any<string>()).Returns(client);
|
||||
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Enabled", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Endpoint", Arg.Any<string>()).Returns("https://unit.test.com/");
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Token", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Priority", Arg.Any<int>()).Returns(4);
|
||||
}
|
||||
|
||||
[TestCase("unit", "test")]
|
||||
public async Task GotifyHelper_Success(string title, string message)
|
||||
{
|
||||
var actual = await new GotifyHelper(_config, _clientFactory, _logger).Send(title, message);
|
||||
Assert.That(actual, Is.EqualTo(true));
|
||||
}
|
||||
|
||||
[TestCase(true, true)]
|
||||
[TestCase(false, false)]
|
||||
public async Task GotifyHelper_Enabled(bool enabled, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Enabled", Arg.Any<bool>()).Returns(enabled);
|
||||
|
||||
var actual = await new GotifyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("http://unittest.com", true)]
|
||||
[TestCase("https://unittest.com", true)]
|
||||
public async Task GotifyHelper_Endpoint(string? endpoint, bool expected)
|
||||
{
|
||||
var client = new HttpClient(new MockHttpMessageHandler(HttpStatusCode.OK));
|
||||
client.BaseAddress = endpoint != null ? new Uri(endpoint) : null;
|
||||
|
||||
_clientFactory.CreateClient(Arg.Any<string>()).Returns(client);
|
||||
|
||||
var actual = await new GotifyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("", false)]
|
||||
[TestCase("UnitTest", true)]
|
||||
public async Task GotifyHelper_Token(string? token, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Token", Arg.Any<string>()).Returns(token);
|
||||
|
||||
var actual = await new GotifyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(-100, false)]
|
||||
[TestCase(-1, false)]
|
||||
[TestCase(0, false)]
|
||||
[TestCase(1, true)]
|
||||
[TestCase(100, true)]
|
||||
public async Task GotifyHelper_Priority(int priority, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Priority", Arg.Any<int>()).Returns(priority);
|
||||
|
||||
var actual = await new GotifyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
}
|
||||
}
|
||||
42
WeddingShare.UnitTests/Tests/Helpers/ImageHelper.cs
Normal file
42
WeddingShare.UnitTests/Tests/Helpers/ImageHelper.cs
Normal file
@@ -0,0 +1,42 @@
|
||||
using Microsoft.Extensions.Logging;
|
||||
using SixLabors.ImageSharp;
|
||||
using SixLabors.ImageSharp.PixelFormats;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class ImageHelperTests
|
||||
{
|
||||
private readonly IFileHelper _fileHelper = Substitute.For<IFileHelper>();
|
||||
private readonly ILogger<ImageHelper> _logger = Substitute.For<ILogger<ImageHelper>>();
|
||||
private readonly IDictionary<ImageOrientation, Image?> _imageCollection;
|
||||
|
||||
public ImageHelperTests()
|
||||
{
|
||||
_imageCollection = new Dictionary<ImageOrientation, Image?>()
|
||||
{
|
||||
{ ImageOrientation.Square, new Image<Rgba32>(100, 100) },
|
||||
{ ImageOrientation.Landscape, new Image<Rgba32>(200, 100) },
|
||||
{ ImageOrientation.Portrait, new Image<Rgba32>(100, 200) }
|
||||
};
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
}
|
||||
|
||||
[TestCase(ImageOrientation.Square)]
|
||||
[TestCase(ImageOrientation.Landscape)]
|
||||
[TestCase(ImageOrientation.Portrait)]
|
||||
public void ImageHelper_GetOrientation(ImageOrientation orientation)
|
||||
{
|
||||
var image = _imageCollection[orientation];
|
||||
Assert.IsNotNull(image);
|
||||
|
||||
var actual = new ImageHelper(_fileHelper, _logger).GetOrientation(image);
|
||||
Assert.That(actual, Is.EqualTo(orientation));
|
||||
}
|
||||
}
|
||||
}
|
||||
95
WeddingShare.UnitTests/Tests/Helpers/NotificationBroker.cs
Normal file
95
WeddingShare.UnitTests/Tests/Helpers/NotificationBroker.cs
Normal file
@@ -0,0 +1,95 @@
|
||||
using System.Net;
|
||||
using System.Net.Mail;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class NotificationBrokerTests
|
||||
{
|
||||
private readonly IConfigHelper _config = Substitute.For<IConfigHelper>();
|
||||
private readonly IHttpClientFactory _clientFactory = Substitute.For<IHttpClientFactory>();
|
||||
private readonly ISmtpClientWrapper _smtp = Substitute.For<ISmtpClientWrapper>();
|
||||
private readonly ILoggerFactory _logger = Substitute.For<ILoggerFactory>();
|
||||
|
||||
public NotificationBrokerTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
var client = new HttpClient(new MockHttpMessageHandler(HttpStatusCode.OK));
|
||||
client.BaseAddress = new Uri("https://unit.test.com/");
|
||||
|
||||
_clientFactory.CreateClient(Arg.Any<string>()).Returns(client);
|
||||
|
||||
_smtp.SendMailAsync(Arg.Any<SmtpClient>(), Arg.Any<MailMessage>()).Returns(Task.FromResult(true));
|
||||
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Enabled", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Recipient", Arg.Any<string>()).Returns("unit@test.com");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Host", Arg.Any<string>()).Returns("https://unit.test.com/");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Port", Arg.Any<int>()).Returns(999);
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Username", Arg.Any<string>()).Returns("Unit");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Password", Arg.Any<string>()).Returns("Test");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "From", Arg.Any<string>()).Returns("unittest@test.com");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "DisplayName", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Smtp", "UseSSL", Arg.Any<bool>()).Returns(true);
|
||||
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Enabled", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Endpoint", Arg.Any<string>()).Returns("https://unit.test.com/");
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Token", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Topic", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Priority", Arg.Any<int>()).Returns(4);
|
||||
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Enabled", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Endpoint", Arg.Any<string>()).Returns("https://unit.test.com/");
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Token", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Priority", Arg.Any<int>()).Returns(4);
|
||||
}
|
||||
|
||||
[TestCase(false, false, false, true)]
|
||||
[TestCase(true, false, false, true)]
|
||||
[TestCase(false, true, false, true)]
|
||||
[TestCase(false, false, true, true)]
|
||||
[TestCase(true, true, true, true)]
|
||||
public async Task NotificationBroker_Success(bool smtp, bool ntfy, bool gotify, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Enabled", Arg.Any<bool>()).Returns(smtp);
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Enabled", Arg.Any<bool>()).Returns(ntfy);
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Enabled", Arg.Any<bool>()).Returns(gotify);
|
||||
|
||||
var actual = await new NotificationBroker(_config, _smtp, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task NotificationBroker_Issue_Smtp()
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Smtp", "Host", Arg.Any<string>()).Returns(string.Empty);
|
||||
|
||||
var actual = await new NotificationBroker(_config, _smtp, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(false));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task NotificationBroker_Issue_Ntfy()
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Endpoint", Arg.Any<string>()).Returns(string.Empty);
|
||||
|
||||
var actual = await new NotificationBroker(_config, _smtp, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(false));
|
||||
}
|
||||
|
||||
[TestCase()]
|
||||
public async Task NotificationBroker_Issue_Gotify()
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Gotify", "Endpoint", Arg.Any<string>()).Returns(string.Empty);
|
||||
|
||||
var actual = await new NotificationBroker(_config, _smtp, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(false));
|
||||
}
|
||||
}
|
||||
}
|
||||
101
WeddingShare.UnitTests/Tests/Helpers/NtfyHelper.cs
Normal file
101
WeddingShare.UnitTests/Tests/Helpers/NtfyHelper.cs
Normal file
@@ -0,0 +1,101 @@
|
||||
using System.Net;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class NtfyHelperTests
|
||||
{
|
||||
private readonly IConfigHelper _config = Substitute.For<IConfigHelper>();
|
||||
private readonly IHttpClientFactory _clientFactory = Substitute.For<IHttpClientFactory>();
|
||||
private readonly ILogger<NtfyHelper> _logger = Substitute.For<ILogger<NtfyHelper>>();
|
||||
|
||||
public NtfyHelperTests()
|
||||
{
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
var client = new HttpClient(new MockHttpMessageHandler(HttpStatusCode.OK));
|
||||
client.BaseAddress = new Uri("https://unit.test.com/");
|
||||
|
||||
_clientFactory.CreateClient(Arg.Any<string>()).Returns(client);
|
||||
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Enabled", Arg.Any<bool>()).Returns(true);
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Endpoint", Arg.Any<string>()).Returns("https://unit.test.com/");
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Token", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Topic", Arg.Any<string>()).Returns("UnitTest");
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Priority", Arg.Any<int>()).Returns(4);
|
||||
}
|
||||
|
||||
[TestCase("unit", "test")]
|
||||
public async Task NtfyHelper_Success(string title, string message)
|
||||
{
|
||||
var actual = await new NtfyHelper(_config, _clientFactory, _logger).Send(title, message);
|
||||
Assert.That(actual, Is.EqualTo(true));
|
||||
}
|
||||
|
||||
[TestCase(true, true)]
|
||||
[TestCase(false, false)]
|
||||
public async Task NtfyHelper_Enabled(bool enabled, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Enabled", Arg.Any<bool>()).Returns(enabled);
|
||||
|
||||
var actual = await new NtfyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("http://unittest.com", true)]
|
||||
[TestCase("https://unittest.com", true)]
|
||||
public async Task NtfyHelper_Endpoint(string? endpoint, bool expected)
|
||||
{
|
||||
var client = new HttpClient(new MockHttpMessageHandler(HttpStatusCode.OK));
|
||||
client.BaseAddress = endpoint != null ? new Uri(endpoint) : null;
|
||||
|
||||
_clientFactory.CreateClient(Arg.Any<string>()).Returns(client);
|
||||
|
||||
var actual = await new NtfyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("", false)]
|
||||
[TestCase("UnitTest", true)]
|
||||
public async Task NtfyHelper_Token(string? token, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Token", Arg.Any<string>()).Returns(token);
|
||||
|
||||
var actual = await new NtfyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(null, false)]
|
||||
[TestCase("", false)]
|
||||
[TestCase("UnitTest", true)]
|
||||
public async Task NtfyHelper_Topic(string? topic, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Topic", Arg.Any<string>()).Returns(topic);
|
||||
|
||||
var actual = await new NtfyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
|
||||
[TestCase(-100, false)]
|
||||
[TestCase(-1, false)]
|
||||
[TestCase(0, false)]
|
||||
[TestCase(1, true)]
|
||||
[TestCase(100, true)]
|
||||
public async Task NtfyHelper_Priority(int priority, bool expected)
|
||||
{
|
||||
_config.GetOrDefault("Notifications", "Ntfy", "Priority", Arg.Any<int>()).Returns(priority);
|
||||
|
||||
var actual = await new NtfyHelper(_config, _clientFactory, _logger).Send("unit", "test");
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
}
|
||||
}
|
||||
46
WeddingShare.UnitTests/Tests/Helpers/UrlHelper.cs
Normal file
46
WeddingShare.UnitTests/Tests/Helpers/UrlHelper.cs
Normal file
@@ -0,0 +1,46 @@
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.UnitTests.Helpers;
|
||||
|
||||
namespace WeddingShare.UnitTests.Tests.Helpers
|
||||
{
|
||||
public class UrlHelperTests
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
|
||||
public UrlHelperTests()
|
||||
{
|
||||
_config = Substitute.For<IConfigHelper>();
|
||||
}
|
||||
|
||||
[SetUp]
|
||||
public void Setup()
|
||||
{
|
||||
_config.GetOrDefault("Settings", "Force_Https", Arg.Any<bool>()).Returns(false);
|
||||
}
|
||||
|
||||
[TestCase("http", "unittest.com", null, "http://unittest.com/")]
|
||||
[TestCase("https", "unittest.org", null, "https://unittest.org/")]
|
||||
[TestCase("http", "www.unittest.com", null, "http://www.unittest.com/")]
|
||||
[TestCase("https", "mobile.unittest.org", null, "https://mobile.unittest.org/")]
|
||||
[TestCase("http", "unittest.com", "", "http://unittest.com/")]
|
||||
[TestCase("https", "unittest.org", "", "https://unittest.org/")]
|
||||
[TestCase("http", "www.unittest.com", "", "http://www.unittest.com/")]
|
||||
[TestCase("https", "mobile.unittest.org", "", "https://mobile.unittest.org/")]
|
||||
[TestCase("http", "unittest.com", "/unittest", "http://unittest.com/unittest")]
|
||||
[TestCase("https", "unittest.org", "?unit=test", "https://unittest.org/?unit=test")]
|
||||
[TestCase("http", "www.unittest.com", "?unit=test&blaa=test", "http://www.unittest.com/?unit=test&blaa=test")]
|
||||
[TestCase("https", "mobile.unittest.org", "/unittest?unit=test&blaa=test", "https://mobile.unittest.org/unittest?unit=test&blaa=test")]
|
||||
public async Task UrlHelper_Success(string scheme, string host, string? querystring, string expected)
|
||||
{
|
||||
_config.GetOrDefault("Settings", "Base_Url", Arg.Any<string>()).Returns(host);
|
||||
|
||||
var mockContext = MockData.MockHttpContext();
|
||||
mockContext.Request.Scheme = scheme;
|
||||
mockContext.Request.Host = new HostString(host);
|
||||
|
||||
var actual = UrlHelper.Generate(mockContext, _config, querystring);
|
||||
Assert.That(actual, Is.EqualTo(expected));
|
||||
}
|
||||
}
|
||||
}
|
||||
26
WeddingShare.UnitTests/WeddingShare.UnitTests.csproj
Normal file
26
WeddingShare.UnitTests/WeddingShare.UnitTests.csproj
Normal file
@@ -0,0 +1,26 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
|
||||
<IsPackable>false</IsPackable>
|
||||
<IsTestProject>true</IsTestProject>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.6.0" />
|
||||
<PackageReference Include="NSubstitute" Version="5.3.0" />
|
||||
<PackageReference Include="NUnit" Version="3.13.3" />
|
||||
<PackageReference Include="NUnit3TestAdapter" Version="4.2.1" />
|
||||
<PackageReference Include="NUnit.Analyzers" Version="3.6.1" />
|
||||
<PackageReference Include="coverlet.collector" Version="6.0.0" />
|
||||
<PackageReference Include="SixLabors.ImageSharp" Version="3.1.6" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\WeddingShare\WeddingShare.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
@@ -5,6 +5,8 @@ VisualStudioVersion = 17.8.34511.84
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "WeddingShare", "WeddingShare\WeddingShare.csproj", "{1E5B4227-C87D-42DA-9946-4F86C9034F35}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WeddingShare.UnitTests", "WeddingShare.UnitTests\WeddingShare.UnitTests.csproj", "{B1BAAF2F-454D-4B57-AD6B-9701D73EB8C6}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
@@ -15,6 +17,10 @@ Global
|
||||
{1E5B4227-C87D-42DA-9946-4F86C9034F35}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{1E5B4227-C87D-42DA-9946-4F86C9034F35}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{1E5B4227-C87D-42DA-9946-4F86C9034F35}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{B1BAAF2F-454D-4B57-AD6B-9701D73EB8C6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{B1BAAF2F-454D-4B57-AD6B-9701D73EB8C6}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{B1BAAF2F-454D-4B57-AD6B-9701D73EB8C6}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{B1BAAF2F-454D-4B57-AD6B-9701D73EB8C6}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
|
||||
52
WeddingShare/Attributes/AllowGuestCreateAttribute.cs
Normal file
52
WeddingShare/Attributes/AllowGuestCreateAttribute.cs
Normal file
@@ -0,0 +1,52 @@
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.AspNetCore.Mvc.Filters;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
|
||||
namespace WeddingShare.Attributes
|
||||
{
|
||||
public class AllowGuestCreateAttribute : ActionFilterAttribute
|
||||
{
|
||||
public override void OnActionExecuting(ActionExecutingContext filterContext)
|
||||
{
|
||||
try
|
||||
{
|
||||
var request = filterContext.HttpContext.Request;
|
||||
|
||||
var galleryId = (request.Query.ContainsKey("id") && !string.IsNullOrWhiteSpace(request.Query["id"])) ? request.Query["id"].ToString().ToLower() : "default";
|
||||
|
||||
if (!string.Equals("default", galleryId, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
var user = filterContext?.HttpContext?.User;
|
||||
if (user?.Identity == null || !user.Identity.IsAuthenticated)
|
||||
{
|
||||
var configHelper = filterContext.HttpContext.RequestServices.GetService<IConfigHelper>();
|
||||
if (configHelper != null)
|
||||
{
|
||||
if (configHelper.GetOrDefault("Settings", "Disable_Guest_Gallery_Creation", true))
|
||||
{
|
||||
var databaseHelper = filterContext.HttpContext.RequestServices.GetService<IDatabaseHelper>();
|
||||
if (databaseHelper != null)
|
||||
{
|
||||
var gallery = databaseHelper.GetGallery(galleryId).Result;
|
||||
if (gallery == null)
|
||||
{
|
||||
filterContext.Result = new RedirectToActionResult("Index", "Error", new { Reason = ErrorCode.GalleryCreationNotAllowed }, false);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
var logger = filterContext.HttpContext.RequestServices.GetService<ILogger<RequiresSecretKeyAttribute>>();
|
||||
if (logger != null)
|
||||
{
|
||||
logger.LogError(ex, $"Failed to check guest creation - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
44
WeddingShare/Attributes/RequiresSecretKeyAttribute.cs
Normal file
44
WeddingShare/Attributes/RequiresSecretKeyAttribute.cs
Normal file
@@ -0,0 +1,44 @@
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.AspNetCore.Mvc.Filters;
|
||||
using WeddingShare.Helpers;
|
||||
|
||||
namespace WeddingShare.Attributes
|
||||
{
|
||||
public class RequiresSecretKeyAttribute : ActionFilterAttribute
|
||||
{
|
||||
public override void OnActionExecuting(ActionExecutingContext filterContext)
|
||||
{
|
||||
try
|
||||
{
|
||||
var request = filterContext.HttpContext.Request;
|
||||
|
||||
var galleryHelper = filterContext.HttpContext.RequestServices.GetService<IGalleryHelper>();
|
||||
if (galleryHelper != null)
|
||||
{
|
||||
var galleryId = (request.Query.ContainsKey("id") && !string.IsNullOrWhiteSpace(request.Query["id"])) ? request.Query["id"].ToString().ToLower() : "default";
|
||||
var secretKey = galleryHelper.GetSecretKey(galleryId).Result;
|
||||
|
||||
var key = request.Query.ContainsKey("key") ? request.Query["key"].ToString() : string.Empty;
|
||||
if (!string.IsNullOrWhiteSpace(secretKey) && !string.Equals(secretKey, key))
|
||||
{
|
||||
var logger = filterContext.HttpContext.RequestServices.GetService<ILogger<RequiresSecretKeyAttribute>>();
|
||||
if (logger != null)
|
||||
{
|
||||
logger.LogWarning($"A request was made to an endpoint with an invalid secure key");
|
||||
}
|
||||
|
||||
filterContext.Result = new RedirectToActionResult("Index", "Error", new { Reason = ErrorCode.InvalidSecretKey }, false);
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
var logger = filterContext.HttpContext.RequestServices.GetService<ILogger<RequiresSecretKeyAttribute>>();
|
||||
if (logger != null)
|
||||
{
|
||||
logger.LogError(ex, $"Failed to validate secure key - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
153
WeddingShare/BackgroundWorkers/DirectoryScanner.cs
Normal file
153
WeddingShare/BackgroundWorkers/DirectoryScanner.cs
Normal file
@@ -0,0 +1,153 @@
|
||||
using NCrontab;
|
||||
using SixLabors.ImageSharp;
|
||||
using SixLabors.ImageSharp.Processing;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.BackgroundWorkers
|
||||
{
|
||||
public sealed class DirectoryScanner(IWebHostEnvironment hostingEnvironment, IConfigHelper configHelper, IDatabaseHelper databaseHelper, IFileHelper fileHelper, IImageHelper imageHelper, ILogger<DirectoryScanner> logger) : BackgroundService
|
||||
{
|
||||
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||
{
|
||||
var cron = configHelper.GetOrDefault("BackgroundServices", "Directory_Scanner_Interval", "*/30 * * * *");
|
||||
var schedule = CrontabSchedule.Parse(cron, new CrontabSchedule.ParseOptions() { IncludingSeconds = cron.Split(new[] { ' ' }, StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries).Length == 6 });
|
||||
|
||||
await Task.Delay((int)TimeSpan.FromSeconds(10).TotalMilliseconds, stoppingToken);
|
||||
await ScanForFiles();
|
||||
|
||||
while (!stoppingToken.IsCancellationRequested)
|
||||
{
|
||||
var now = DateTime.Now;
|
||||
var nextExecutionTime = schedule.GetNextOccurrence(now);
|
||||
var waitTime = nextExecutionTime - now;
|
||||
await Task.Delay(waitTime, stoppingToken);
|
||||
|
||||
await ScanForFiles();
|
||||
}
|
||||
}
|
||||
|
||||
private async Task ScanForFiles()
|
||||
{
|
||||
await Task.Run(async () =>
|
||||
{
|
||||
var allowedFileTypes = configHelper.GetOrDefault("Settings", "Allowed_File_Types", ".jpg,.jpeg,.png").Split(',', StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries);
|
||||
|
||||
var thumbnailsDirectory = Path.Combine(hostingEnvironment.WebRootPath, "thumbnails");
|
||||
fileHelper.CreateDirectoryIfNotExists(thumbnailsDirectory);
|
||||
|
||||
var uploadsDirectory = Path.Combine(hostingEnvironment.WebRootPath, "uploads");
|
||||
if (fileHelper.DirectoryExists(uploadsDirectory))
|
||||
{
|
||||
var searchPattern = !configHelper.GetOrDefault("Settings", "Single_Gallery_Mode", false) ? "*" : "default";
|
||||
var galleries = fileHelper.GetDirectories(uploadsDirectory, searchPattern, SearchOption.TopDirectoryOnly)?.Where(x => !Path.GetFileName(x).StartsWith("."));
|
||||
if (galleries != null)
|
||||
{
|
||||
foreach (var gallery in galleries)
|
||||
{
|
||||
try
|
||||
{
|
||||
var id = Path.GetFileName(gallery).ToLower();
|
||||
var galleryItem = await databaseHelper.GetGallery(id);
|
||||
if (galleryItem == null)
|
||||
{
|
||||
galleryItem = await databaseHelper.AddGallery(new GalleryModel()
|
||||
{
|
||||
Name = id
|
||||
});
|
||||
}
|
||||
|
||||
if (galleryItem != null)
|
||||
{
|
||||
var galleryItems = await databaseHelper.GetAllGalleryItems(galleryItem.Id);
|
||||
|
||||
if (Path.Exists(gallery))
|
||||
{
|
||||
var approvedFiles = fileHelper.GetFiles(gallery, "*.*", SearchOption.TopDirectoryOnly).Where(x => allowedFileTypes.Any(y => string.Equals(Path.GetExtension(x).Trim('.'), y.Trim('.'), StringComparison.OrdinalIgnoreCase)));
|
||||
if (approvedFiles != null)
|
||||
{
|
||||
foreach (var file in approvedFiles)
|
||||
{
|
||||
try
|
||||
{
|
||||
var filename = Path.GetFileName(file);
|
||||
if (!galleryItems.Exists(x => string.Equals(x.Title, filename, StringComparison.OrdinalIgnoreCase)))
|
||||
{
|
||||
await databaseHelper.AddGalleryItem(new GalleryItemModel()
|
||||
{
|
||||
GalleryId = galleryItem.Id,
|
||||
Title = filename,
|
||||
State = GalleryItemState.Approved
|
||||
});
|
||||
}
|
||||
|
||||
var thumbnailPath = Path.Combine(thumbnailsDirectory, $"{Path.GetFileNameWithoutExtension(file)}.webp");
|
||||
if (!fileHelper.FileExists(thumbnailPath))
|
||||
{
|
||||
await imageHelper.GenerateThumbnail(file, thumbnailPath, configHelper.GetOrDefault("Settings", "Thumbnail_Size", 720));
|
||||
}
|
||||
else
|
||||
{
|
||||
using (var img = await Image.LoadAsync(thumbnailPath))
|
||||
{
|
||||
var width = img.Width;
|
||||
|
||||
img.Mutate(x => x.AutoOrient());
|
||||
|
||||
if (width != img.Width)
|
||||
{
|
||||
await img.SaveAsWebpAsync(thumbnailPath);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
logger.LogError(ex, $"An error occurred while scanning file '{file}'");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (Path.Exists(Path.Combine(gallery, "Pending")))
|
||||
{
|
||||
var pendingFiles = fileHelper.GetFiles(Path.Combine(gallery, "Pending"), "*.*", SearchOption.TopDirectoryOnly).Where(x => allowedFileTypes.Any(y => string.Equals(Path.GetExtension(x).Trim('.'), y.Trim('.'), StringComparison.OrdinalIgnoreCase)));
|
||||
if (pendingFiles != null)
|
||||
{
|
||||
foreach (var file in pendingFiles)
|
||||
{
|
||||
try
|
||||
{
|
||||
var filename = Path.GetFileName(file);
|
||||
if (!galleryItems.Exists(x => string.Equals(x.Title, filename, StringComparison.OrdinalIgnoreCase)))
|
||||
{
|
||||
await databaseHelper.AddGalleryItem(new GalleryItemModel()
|
||||
{
|
||||
GalleryId = galleryItem.Id,
|
||||
Title = filename,
|
||||
State = GalleryItemState.Pending
|
||||
});
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
logger.LogError(ex, $"An error occurred while scanning file '{file}'");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
logger.LogError(ex, $"An error occurred while scanning directory '{gallery}'");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
61
WeddingShare/BackgroundWorkers/NotificationReport.cs
Normal file
61
WeddingShare/BackgroundWorkers/NotificationReport.cs
Normal file
@@ -0,0 +1,61 @@
|
||||
using System.Text;
|
||||
using NCrontab;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
|
||||
namespace WeddingShare.BackgroundWorkers
|
||||
{
|
||||
public sealed class NotificationReport(IConfigHelper configHelper, IDatabaseHelper databaseHelper, ISmtpClientWrapper smtpHelper, ILoggerFactory loggerFactory) : BackgroundService
|
||||
{
|
||||
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||
{
|
||||
if (configHelper.GetOrDefault("Settings", "Email_Report", true) && configHelper.GetOrDefault("Notifications", "Smtp", "Enabled", false))
|
||||
{
|
||||
var cron = configHelper.GetOrDefault("BackgroundServices", "Email_Report_Interval", "0 0 * * *");
|
||||
var schedule = CrontabSchedule.Parse(cron, new CrontabSchedule.ParseOptions() { IncludingSeconds = cron.Split(new[] { ' ' }, StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries).Length == 6 });
|
||||
|
||||
while (!stoppingToken.IsCancellationRequested)
|
||||
{
|
||||
var now = DateTime.Now;
|
||||
var nextExecutionTime = schedule.GetNextOccurrence(now);
|
||||
var waitTime = nextExecutionTime - now;
|
||||
await Task.Delay(waitTime, stoppingToken);
|
||||
|
||||
await SendReport();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private async Task SendReport()
|
||||
{
|
||||
await Task.Run(async () =>
|
||||
{
|
||||
var pendingItems = await databaseHelper.GetPendingGalleryItems();
|
||||
if (pendingItems != null && pendingItems.Any())
|
||||
{
|
||||
var builder = new StringBuilder();
|
||||
builder.AppendLine($"<h1>You have items pending review!</h1>");
|
||||
|
||||
foreach (var item in pendingItems.GroupBy(x => x.GalleryName).OrderBy(x => x.Key))
|
||||
{
|
||||
try
|
||||
{
|
||||
builder.AppendLine($"<p style=\"font-size: 16pt;\">{item.Key} - Pending Items ({item.Count()})</p>");
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
loggerFactory.CreateLogger<NotificationReport>().LogError(ex, $"Failed to build gallery report for id '{item?.Key}' - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
var sent = await new EmailHelper(configHelper, smtpHelper, loggerFactory.CreateLogger<EmailHelper>()).Send("Pending Items Report", builder.ToString());
|
||||
if (!sent)
|
||||
{
|
||||
loggerFactory.CreateLogger<NotificationReport>().LogWarning($"Failed to send notification report");
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
24
WeddingShare/Configurations/DatabaseConfiguration.cs
Normal file
24
WeddingShare/Configurations/DatabaseConfiguration.cs
Normal file
@@ -0,0 +1,24 @@
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Helpers.Dbup;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.Configurations
|
||||
{
|
||||
internal static class DatabaseConfiguration
|
||||
{
|
||||
public static void AddDatabaseConfiguration(this IServiceCollection services, ConfigHelper config)
|
||||
{
|
||||
switch (config.GetOrDefault("Database", "Database_Type", "sqlite")?.ToLower())
|
||||
{
|
||||
case "sqlite":
|
||||
services.AddSingleton<IDatabaseHelper, SQLiteDatabaseHelper>();
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
services.AddHostedService<DbupMigrator>();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
using WeddingShare.Helpers;
|
||||
|
||||
namespace WeddingShare.Configurations
|
||||
{
|
||||
internal static class DependencyInjectionConfiguration
|
||||
{
|
||||
public static void AddDependencyInjectionConfiguration(this IServiceCollection services)
|
||||
{
|
||||
services.AddSingleton<IConfigHelper, ConfigHelper>();
|
||||
services.AddSingleton<IEnvironmentWrapper, EnvironmentWrapper>();
|
||||
services.AddSingleton<IGalleryHelper, GalleryHelper>();
|
||||
services.AddSingleton<IImageHelper, ImageHelper>();
|
||||
services.AddSingleton<IFileHelper, FileHelper>();
|
||||
services.AddSingleton<IDeviceDetector, DeviceDetector>();
|
||||
services.AddSingleton<ISmtpClientWrapper, SmtpClientWrapper>();
|
||||
}
|
||||
}
|
||||
}
|
||||
50
WeddingShare/Configurations/NotificationConfiguration.cs
Normal file
50
WeddingShare/Configurations/NotificationConfiguration.cs
Normal file
@@ -0,0 +1,50 @@
|
||||
using System.Net.Http.Headers;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
|
||||
namespace WeddingShare.Configurations
|
||||
{
|
||||
internal static class NotificationConfiguration
|
||||
{
|
||||
private const int CLIENT_DEFAULT_TIMEOUT = 10;
|
||||
|
||||
public static void AddNotificationConfiguration(this IServiceCollection services, ConfigHelper config)
|
||||
{
|
||||
services.AddSingleton<INotificationHelper, NotificationBroker>();
|
||||
services.AddNtfyConfiguration(config);
|
||||
services.AddGotifyConfiguration(config);
|
||||
}
|
||||
|
||||
public static void AddNtfyConfiguration(this IServiceCollection services, ConfigHelper config)
|
||||
{
|
||||
services.AddHttpClient("NtfyClient", (serviceProvider, httpClient) =>
|
||||
{
|
||||
var endpoint = config.GetOrDefault("Notifications", "Ntfy", "Endpoint", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(endpoint))
|
||||
{
|
||||
var token = config.GetOrDefault("Notifications", "Ntfy", "Token", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(token))
|
||||
{
|
||||
httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", token);
|
||||
}
|
||||
|
||||
httpClient.BaseAddress = new Uri(endpoint);
|
||||
httpClient.Timeout = TimeSpan.FromSeconds(CLIENT_DEFAULT_TIMEOUT);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public static void AddGotifyConfiguration(this IServiceCollection services, ConfigHelper config)
|
||||
{
|
||||
services.AddHttpClient("GotifyClient", (serviceProvider, httpClient) =>
|
||||
{
|
||||
var endpoint = config.GetOrDefault("Notifications", "Gotify", "Endpoint", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(endpoint))
|
||||
{
|
||||
httpClient.BaseAddress = new Uri(endpoint);
|
||||
httpClient.Timeout = TimeSpan.FromSeconds(CLIENT_DEFAULT_TIMEOUT);
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,13 +1,17 @@
|
||||
using System.IO.Compression;
|
||||
using System.Security.Claims;
|
||||
using Microsoft.AspNetCore.Authentication.Cookies;
|
||||
using Microsoft.AspNetCore.Authentication;
|
||||
using Microsoft.AspNetCore.Authentication.Cookies;
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.Extensions.Localization;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Views.Admin;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
using WeddingShare.Models;
|
||||
using WeddingShare.Models.Database;
|
||||
using WeddingShare.Views.Admin;
|
||||
|
||||
namespace WeddingShare.Controllers
|
||||
{
|
||||
@@ -16,19 +20,43 @@ namespace WeddingShare.Controllers
|
||||
{
|
||||
private readonly IWebHostEnvironment _hostingEnvironment;
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly IDatabaseHelper _database;
|
||||
private readonly IDeviceDetector _deviceDetector;
|
||||
private readonly IFileHelper _fileHelper;
|
||||
private readonly IImageHelper _imageHelper;
|
||||
private readonly INotificationHelper _notificationHelper;
|
||||
private readonly ILogger _logger;
|
||||
private readonly IStringLocalizer<AdminController> _localizer;
|
||||
|
||||
private readonly string UploadsDirectory;
|
||||
private readonly string ThumbnailsDirectory;
|
||||
|
||||
public AdminController(IWebHostEnvironment hostingEnvironment, IConfigHelper config, ILogger<AdminController> logger, IStringLocalizer<AdminController> localizer)
|
||||
public AdminController(IWebHostEnvironment hostingEnvironment, IConfigHelper config, IDatabaseHelper database, IDeviceDetector deviceDetector, IFileHelper fileHelper, IImageHelper imageHelper, INotificationHelper notificationHelper, ILogger<AdminController> logger, IStringLocalizer<AdminController> localizer)
|
||||
{
|
||||
_hostingEnvironment = hostingEnvironment;
|
||||
_config = config;
|
||||
_database = database;
|
||||
_deviceDetector = deviceDetector;
|
||||
_fileHelper = fileHelper;
|
||||
_imageHelper = imageHelper;
|
||||
_notificationHelper = notificationHelper;
|
||||
_logger = logger;
|
||||
_localizer = localizer;
|
||||
|
||||
UploadsDirectory = Path.Combine(_hostingEnvironment.WebRootPath, "uploads");
|
||||
ThumbnailsDirectory = Path.Combine(_hostingEnvironment.WebRootPath, "thumbnails");
|
||||
}
|
||||
|
||||
[AllowAnonymous]
|
||||
[HttpGet]
|
||||
public async Task<IActionResult> Login()
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
return RedirectToAction("Index", "Admin");
|
||||
}
|
||||
|
||||
return View();
|
||||
}
|
||||
|
||||
[AllowAnonymous]
|
||||
@@ -37,19 +65,46 @@ namespace WeddingShare.Controllers
|
||||
{
|
||||
try
|
||||
{
|
||||
var passkey = _config.Get("Settings", "Admin_Password");
|
||||
if (string.IsNullOrEmpty(passkey) || string.Equals(passkey, model?.Password))
|
||||
var user = await _database.GetUser(model.Username);
|
||||
if (user != null && !user.IsLockedOut)
|
||||
{
|
||||
var claims = new List<Claim>
|
||||
if (await _database.ValidateCredentials(user.Username, model.Password))
|
||||
{
|
||||
new Claim(ClaimTypes.Name, "Admin")
|
||||
};
|
||||
if (user.FailedLogins > 0)
|
||||
{
|
||||
await _database.ResetLockoutCount(user.Id);
|
||||
}
|
||||
|
||||
var claimsIdentity = new ClaimsIdentity(claims, CookieAuthenticationDefaults.AuthenticationScheme);
|
||||
var claims = new List<Claim>
|
||||
{
|
||||
new Claim(ClaimTypes.Name, user.Username.ToLower())
|
||||
};
|
||||
|
||||
await this.HttpContext.SignInAsync(CookieAuthenticationDefaults.AuthenticationScheme, new ClaimsPrincipal(claimsIdentity));
|
||||
var claimsIdentity = new ClaimsIdentity(claims, CookieAuthenticationDefaults.AuthenticationScheme);
|
||||
|
||||
return Json(new { success = true });
|
||||
await this.HttpContext.SignInAsync(CookieAuthenticationDefaults.AuthenticationScheme, new ClaimsPrincipal(claimsIdentity));
|
||||
|
||||
return Json(new { success = true });
|
||||
}
|
||||
else
|
||||
{
|
||||
if (_config.GetOrDefault("Notifications", "Alerts", "Failed_Login", true))
|
||||
{
|
||||
await _notificationHelper.Send("Invalid Login Detected", $"An invalid login attempt was made for account '{model?.Username}'.", UrlHelper.Generate(HttpContext, _config, "/Admin"));
|
||||
}
|
||||
|
||||
var failedAttempts = await _database.IncrementLockoutCount(user.Id);
|
||||
if (failedAttempts >= _config.GetOrDefault("Settings", "Account", "Lockout_Attempts", 5))
|
||||
{
|
||||
var timeout = _config.GetOrDefault("Settings", "Account", "Lockout_Mins", 60);
|
||||
await _database.SetLockout(user.Id, DateTime.UtcNow.AddMinutes(timeout));
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Alerts", "Account_Lockout", true))
|
||||
{
|
||||
await _notificationHelper.Send("Account Lockout", $"Account '{model?.Username}' has been locked out for {timeout} minutes due to too many failed login attempts.", UrlHelper.Generate(HttpContext, _config, "/Admin"));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
@@ -57,7 +112,7 @@ namespace WeddingShare.Controllers
|
||||
_logger.LogError(ex, $"{_localizer["Login_Failed"].Value} - {ex?.Message}");
|
||||
}
|
||||
|
||||
return Json(new { success = false, message = "Invalid password" });
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[Authorize]
|
||||
@@ -69,7 +124,7 @@ namespace WeddingShare.Controllers
|
||||
}
|
||||
|
||||
[HttpGet]
|
||||
public IActionResult Index()
|
||||
public async Task<IActionResult> Index()
|
||||
{
|
||||
if (User?.Identity == null || !User.Identity.IsAuthenticated)
|
||||
{
|
||||
@@ -78,12 +133,28 @@ namespace WeddingShare.Controllers
|
||||
|
||||
var model = new IndexModel();
|
||||
|
||||
var deviceType = HttpContext.Session.GetString("DeviceType");
|
||||
if (string.IsNullOrWhiteSpace(deviceType))
|
||||
{
|
||||
deviceType = (await _deviceDetector.ParseDeviceType(Request.Headers["User-Agent"].ToString())).ToString();
|
||||
HttpContext.Session.SetString("DeviceType", deviceType ?? "Desktop");
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
if (Directory.Exists(UploadsDirectory))
|
||||
if (!_config.GetOrDefault("Settings", "Single_Gallery_Mode", false))
|
||||
{
|
||||
model.Galleries = Directory.GetDirectories(UploadsDirectory)?.Select(x => new KeyValuePair<string, string>(Path.GetFileName(x), x))?.ToList();
|
||||
model.PendingRequests = model.Galleries?.SelectMany(x => Directory.GetFiles(Path.Combine(x.Value, "Pending"), "*.*", SearchOption.TopDirectoryOnly))?.Select(x => x.Replace(UploadsDirectory, string.Empty))?.ToList();
|
||||
model.Galleries = await _database.GetAllGalleries();
|
||||
model.PendingRequests = await _database.GetPendingGalleryItems();
|
||||
}
|
||||
else
|
||||
{
|
||||
var gallery = await _database.GetGallery("default");
|
||||
if (gallery != null)
|
||||
{
|
||||
model.Galleries = new List<GalleryModel>() { gallery };
|
||||
model.PendingRequests = await _database.GetPendingGalleryItems(gallery.Id);
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
@@ -95,23 +166,43 @@ namespace WeddingShare.Controllers
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
public IActionResult ReviewPhoto(string galleryId, string photoId, ReviewAction action)
|
||||
public async Task<IActionResult> ReviewPhoto(int id, ReviewAction action)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
var galleryDir = Path.Combine(UploadsDirectory, galleryId);
|
||||
var reviewFile = Path.Combine(galleryDir, "Pending", photoId);
|
||||
if (System.IO.File.Exists(reviewFile))
|
||||
{
|
||||
var review = await _database.GetPendingGalleryItem(id);
|
||||
if (review != null)
|
||||
{
|
||||
var galleryDir = Path.Combine(UploadsDirectory, review.GalleryName);
|
||||
var reviewFile = Path.Combine(galleryDir, "Pending", review.Title);
|
||||
if (action == ReviewAction.APPROVED)
|
||||
{
|
||||
System.IO.File.Move(reviewFile, Path.Combine(galleryDir, Path.GetFileName(reviewFile)));
|
||||
_fileHelper.CreateDirectoryIfNotExists(ThumbnailsDirectory);
|
||||
|
||||
await _imageHelper.GenerateThumbnail(reviewFile, Path.Combine(ThumbnailsDirectory, $"{Path.GetFileNameWithoutExtension(reviewFile)}.webp"), _config.GetOrDefault("Settings", "Thumbnail_Size", 720));
|
||||
|
||||
_fileHelper.MoveFileIfExists(reviewFile, Path.Combine(galleryDir, review.Title));
|
||||
|
||||
review.State = GalleryItemState.Approved;
|
||||
await _database.EditGalleryItem(review);
|
||||
}
|
||||
else if (action == ReviewAction.REJECTED)
|
||||
{
|
||||
System.IO.File.Delete(reviewFile);
|
||||
var retain = _config.GetOrDefault("Settings", "Retain_Rejected_Items", false);
|
||||
if (retain)
|
||||
{
|
||||
var rejectedDir = Path.Combine(galleryDir, "Rejected");
|
||||
_fileHelper.CreateDirectoryIfNotExists(rejectedDir);
|
||||
_fileHelper.MoveFileIfExists(reviewFile, Path.Combine(rejectedDir, review.Title));
|
||||
}
|
||||
else
|
||||
{
|
||||
_fileHelper.DeleteFileIfExists(reviewFile);
|
||||
}
|
||||
|
||||
await _database.DeleteGalleryItem(review);
|
||||
}
|
||||
else if (action == ReviewAction.UNKNOWN)
|
||||
{
|
||||
@@ -133,5 +224,442 @@ namespace WeddingShare.Controllers
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
public async Task<IActionResult> BulkReview(ReviewAction action)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
var items = await _database.GetPendingGalleryItems();
|
||||
if (items != null && items.Any())
|
||||
{
|
||||
foreach (var review in items)
|
||||
{
|
||||
var galleryDir = Path.Combine(UploadsDirectory, review.GalleryName);
|
||||
var reviewFile = Path.Combine(galleryDir, "Pending", review.Title);
|
||||
if (action == ReviewAction.APPROVED)
|
||||
{
|
||||
_fileHelper.CreateDirectoryIfNotExists(ThumbnailsDirectory);
|
||||
|
||||
await _imageHelper.GenerateThumbnail(reviewFile, Path.Combine(ThumbnailsDirectory, $"{Path.GetFileNameWithoutExtension(reviewFile)}.webp"), _config.GetOrDefault("Settings", "Thumbnail_Size", 720));
|
||||
|
||||
_fileHelper.MoveFileIfExists(reviewFile, Path.Combine(galleryDir, review.Title));
|
||||
|
||||
review.State = GalleryItemState.Approved;
|
||||
await _database.EditGalleryItem(review);
|
||||
}
|
||||
else if (action == ReviewAction.REJECTED)
|
||||
{
|
||||
var retain = _config.GetOrDefault("Settings", "Retain_Rejected_Items", false);
|
||||
if (retain)
|
||||
{
|
||||
var rejectedDir = Path.Combine(galleryDir, "Rejected");
|
||||
_fileHelper.CreateDirectoryIfNotExists(rejectedDir);
|
||||
_fileHelper.MoveFileIfExists(reviewFile, Path.Combine(rejectedDir, review.Title));
|
||||
}
|
||||
else
|
||||
{
|
||||
_fileHelper.DeleteFileIfExists(reviewFile);
|
||||
}
|
||||
|
||||
await _database.DeleteGalleryItem(review);
|
||||
}
|
||||
else if (action == ReviewAction.UNKNOWN)
|
||||
{
|
||||
throw new Exception(_localizer["Unknown_Review_Action"].Value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = true, action });
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Reviewing_Photo"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
public async Task<IActionResult> AddGallery(GalleryModel model)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
if (!string.IsNullOrWhiteSpace(model?.Name))
|
||||
{
|
||||
try
|
||||
{
|
||||
var check = await _database.GetGallery(model.Name);
|
||||
if (check == null)
|
||||
{
|
||||
return Json(new { success = string.Equals(model?.Name, (await _database.AddGallery(model))?.Name, StringComparison.OrdinalIgnoreCase) });
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Gallery_Name_Already_Exists"].Value });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Add_Gallery"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Name_Cannot_Be_Blank"].Value });
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpPut]
|
||||
public async Task<IActionResult> EditGallery(GalleryModel model)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
if (!string.IsNullOrWhiteSpace(model?.Name))
|
||||
{
|
||||
try
|
||||
{
|
||||
var check = await _database.GetGallery(model.Name);
|
||||
if (check == null || model.Id == check.Id)
|
||||
{
|
||||
var gallery = await _database.GetGallery(model.Id);
|
||||
if (gallery != null)
|
||||
{
|
||||
gallery.Name = model.Name;
|
||||
gallery.SecretKey = model.SecretKey;
|
||||
|
||||
return Json(new { success = string.Equals(model?.Name, (await _database.EditGallery(gallery))?.Name, StringComparison.OrdinalIgnoreCase) });
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Failed_Edit_Gallery"].Value });
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Gallery_Name_Already_Exists"].Value });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Edit_Gallery"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Name_Cannot_Be_Blank"].Value });
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpDelete]
|
||||
public async Task<IActionResult> WipeGallery(int id)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
var gallery = await _database.GetGallery(id);
|
||||
if (gallery != null)
|
||||
{
|
||||
var galleryDir = Path.Combine(UploadsDirectory, gallery.Name);
|
||||
if (_fileHelper.DirectoryExists(galleryDir))
|
||||
{
|
||||
foreach (var photo in _fileHelper.GetFiles(galleryDir, "*.*", SearchOption.AllDirectories))
|
||||
{
|
||||
var thumbnail = Path.Combine(ThumbnailsDirectory, $"{Path.GetFileNameWithoutExtension(photo)}.webp");
|
||||
_fileHelper.DeleteFileIfExists(thumbnail);
|
||||
}
|
||||
|
||||
_fileHelper.DeleteDirectoryIfExists(galleryDir);
|
||||
_fileHelper.CreateDirectoryIfNotExists(galleryDir);
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Alerts", "Destructive_Action", true))
|
||||
{
|
||||
await _notificationHelper.Send("Destructive Action Performed", $"The destructive action 'Wipe' was performed on gallery '{gallery.Name}'.", UrlHelper.Generate(HttpContext, _config, "/Admin"));
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = await _database.WipeGallery(gallery) });
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Failed_Wipe_Gallery"].Value });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Wipe_Gallery"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpDelete]
|
||||
public async Task<IActionResult> WipeAllGalleries()
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
if (_fileHelper.DirectoryExists(UploadsDirectory))
|
||||
{
|
||||
foreach (var gallery in _fileHelper.GetDirectories(UploadsDirectory, "*", SearchOption.TopDirectoryOnly))
|
||||
{
|
||||
_fileHelper.DeleteDirectoryIfExists(gallery);
|
||||
}
|
||||
|
||||
foreach (var thumbnail in _fileHelper.GetFiles(ThumbnailsDirectory, "*.*", SearchOption.AllDirectories))
|
||||
{
|
||||
_fileHelper.DeleteFileIfExists(thumbnail);
|
||||
}
|
||||
|
||||
_fileHelper.CreateDirectoryIfNotExists(Path.Combine(UploadsDirectory, "default"));
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Alerts", "Destructive_Action", true))
|
||||
{
|
||||
await _notificationHelper.Send("Destructive Action Performed", $"The destructive action 'Wipe' was performed on all galleries'.", UrlHelper.Generate(HttpContext, _config, "/Admin"));
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = await _database.WipeAllGalleries() });
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Wipe_Galleries"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpDelete]
|
||||
public async Task<IActionResult> DeleteGallery(int id)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
var gallery = await _database.GetGallery(id);
|
||||
if (gallery != null)
|
||||
{
|
||||
var galleryDir = Path.Combine(UploadsDirectory, gallery.Name);
|
||||
_fileHelper.DeleteDirectoryIfExists(galleryDir);
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Alerts", "Destructive_Action", true))
|
||||
{
|
||||
await _notificationHelper.Send("Destructive Action Performed", $"The destructive action 'Delete' was performed on gallery '{gallery.Name}'.", UrlHelper.Generate(HttpContext, _config, "/Admin"));
|
||||
}
|
||||
|
||||
return Json(new { success = await _database.DeleteGallery(gallery) });
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Failed_Delete_Gallery"].Value });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Delete_Gallery"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpDelete]
|
||||
public async Task<IActionResult> DeletePhoto(int id)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
var photo = await _database.GetGalleryItem(id);
|
||||
if (photo != null)
|
||||
{
|
||||
var gallery = await _database.GetGallery(photo.GalleryId);
|
||||
if (gallery != null)
|
||||
{
|
||||
var photoPath = Path.Combine(UploadsDirectory, gallery.Name, photo.Title);
|
||||
_fileHelper.DeleteFileIfExists(photoPath);
|
||||
|
||||
return Json(new { success = await _database.DeleteGalleryItem(photo) });
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Failed_Delete_Gallery"].Value });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Delete_Gallery"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
public async Task<IActionResult> DownloadGallery(int id)
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
try
|
||||
{
|
||||
var gallery = await _database.GetGallery(id);
|
||||
if (gallery != null)
|
||||
{
|
||||
var galleryDir = Path.Combine(UploadsDirectory, gallery.Name);
|
||||
if (_fileHelper.DirectoryExists(galleryDir))
|
||||
{
|
||||
var tempZipDir = $"Temp";
|
||||
_fileHelper.CreateDirectoryIfNotExists(tempZipDir);
|
||||
|
||||
var tempZipFile = Path.Combine(tempZipDir, $"{gallery.Name}-{DateTime.Now.ToString("yyyy-MM-dd_HH-mm-ss")}.zip");
|
||||
ZipFile.CreateFromDirectory(galleryDir, tempZipFile, CompressionLevel.Optimal, false);
|
||||
|
||||
byte[] bytes = await _fileHelper.ReadAllBytes(tempZipFile);
|
||||
_fileHelper.DeleteFileIfExists(tempZipFile);
|
||||
|
||||
return Json(new { success = true, filename = Path.GetFileName(tempZipFile), content = Convert.ToBase64String(bytes, 0, bytes.Length) });
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, message = _localizer["Failed_Download_Gallery"].Value });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Download_Gallery"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpGet]
|
||||
public async Task<IActionResult> ExportBackup()
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
var tempDir = $"Temp";
|
||||
var exportDir = Path.Combine(tempDir, "Export");
|
||||
|
||||
try
|
||||
{
|
||||
if (_fileHelper.DirectoryExists(UploadsDirectory))
|
||||
{
|
||||
_fileHelper.CreateDirectoryIfNotExists(tempDir);
|
||||
_fileHelper.DeleteDirectoryIfExists(exportDir);
|
||||
_fileHelper.CreateDirectoryIfNotExists(exportDir);
|
||||
|
||||
var dbExport = Path.Combine(exportDir, $"WeddingShare.bak");
|
||||
var exported = await _database.Export($"Data Source={dbExport}");
|
||||
if (exported)
|
||||
{
|
||||
var uploadsZip = Path.Combine(exportDir, $"Uploads.bak");
|
||||
ZipFile.CreateFromDirectory(UploadsDirectory, uploadsZip, CompressionLevel.Optimal, false);
|
||||
|
||||
var thumbnailsZip = Path.Combine(exportDir, $"Thumbnails.bak");
|
||||
ZipFile.CreateFromDirectory(ThumbnailsDirectory, thumbnailsZip, CompressionLevel.Optimal, false);
|
||||
|
||||
var exportZipFile = Path.Combine(tempDir, $"WeddingShare-{DateTime.Now.ToString("yyyy-MM-dd_HH-mm-ss")}.zip");
|
||||
_fileHelper.DeleteFileIfExists(exportZipFile);
|
||||
|
||||
ZipFile.CreateFromDirectory(exportDir, exportZipFile, CompressionLevel.Optimal, false);
|
||||
_fileHelper.DeleteFileIfExists(dbExport);
|
||||
_fileHelper.DeleteFileIfExists(uploadsZip);
|
||||
_fileHelper.DeleteFileIfExists(thumbnailsZip);
|
||||
|
||||
byte[] bytes = await _fileHelper.ReadAllBytes(exportZipFile);
|
||||
_fileHelper.DeleteFileIfExists(exportZipFile);
|
||||
|
||||
return Json(new { success = true, filename = Path.GetFileName(exportZipFile), content = Convert.ToBase64String(bytes, 0, bytes.Length) });
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Failed_Export"].Value} - {ex?.Message}");
|
||||
}
|
||||
finally
|
||||
{
|
||||
_fileHelper.DeleteDirectoryIfExists(exportDir);
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
public async Task<IActionResult> ImportBackup()
|
||||
{
|
||||
if (User?.Identity != null && User.Identity.IsAuthenticated)
|
||||
{
|
||||
var tempDir = $"Temp";
|
||||
var importDir = Path.Combine(tempDir, "Import");
|
||||
|
||||
try
|
||||
{
|
||||
var files = Request?.Form?.Files;
|
||||
if (files != null && files.Count > 0)
|
||||
{
|
||||
foreach (IFormFile file in files)
|
||||
{
|
||||
var extension = Path.GetExtension(file.FileName)?.Trim('.');
|
||||
if (string.Equals("zip", extension, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
_fileHelper.CreateDirectoryIfNotExists(tempDir);
|
||||
|
||||
var filePath = Path.Combine(tempDir, "Import.zip");
|
||||
if (!string.IsNullOrWhiteSpace(filePath))
|
||||
{
|
||||
await _fileHelper.SaveFile(file, filePath, FileMode.Create);
|
||||
|
||||
_fileHelper.DeleteDirectoryIfExists(importDir);
|
||||
_fileHelper.CreateDirectoryIfNotExists(importDir);
|
||||
|
||||
ZipFile.ExtractToDirectory(filePath, importDir, true);
|
||||
_fileHelper.DeleteFileIfExists(filePath);
|
||||
|
||||
var uploadsZip = Path.Combine(importDir, "Uploads.bak");
|
||||
ZipFile.ExtractToDirectory(uploadsZip, UploadsDirectory, true);
|
||||
|
||||
var thumbnailsZip = Path.Combine(importDir, "Thumbnails.bak");
|
||||
ZipFile.ExtractToDirectory(thumbnailsZip, ThumbnailsDirectory, true);
|
||||
|
||||
var dbImport = Path.Combine(importDir, "WeddingShare.bak");
|
||||
var imported = await _database.Import($"Data Source={dbImport}");
|
||||
|
||||
return Json(new { success = imported });
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"{_localizer["Import_Failed"].Value} - {ex?.Message}");
|
||||
}
|
||||
finally
|
||||
{
|
||||
_fileHelper.DeleteDirectoryIfExists(importDir);
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = false });
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,30 +1,19 @@
|
||||
using System.Diagnostics;
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using WeddingShare.Models;
|
||||
|
||||
namespace WeddingShare.Controllers
|
||||
{
|
||||
[AllowAnonymous]
|
||||
public class ErrorController : Controller
|
||||
{
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public ErrorController(ILogger<ErrorController> logger)
|
||||
public ErrorController()
|
||||
{
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
|
||||
public IActionResult Index()
|
||||
{
|
||||
return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier });
|
||||
}
|
||||
|
||||
[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
|
||||
public IActionResult AccessDenied()
|
||||
{
|
||||
return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier });
|
||||
return View();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,10 +1,15 @@
|
||||
using System.Diagnostics;
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.Extensions.Localization;
|
||||
using System.Net;
|
||||
using WeddingShare.Attributes;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Extensions;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Helpers.Notifications;
|
||||
using WeddingShare.Models;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.Controllers
|
||||
{
|
||||
@@ -13,152 +18,226 @@ namespace WeddingShare.Controllers
|
||||
{
|
||||
private readonly IWebHostEnvironment _hostingEnvironment;
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly IDatabaseHelper _database;
|
||||
private readonly IFileHelper _fileHelper;
|
||||
private readonly IGalleryHelper _gallery;
|
||||
private readonly IDeviceDetector _deviceDetector;
|
||||
private readonly IImageHelper _imageHelper;
|
||||
private readonly INotificationHelper _notificationHelper;
|
||||
private readonly ILogger _logger;
|
||||
private readonly IStringLocalizer<GalleryController> _localizer;
|
||||
|
||||
private readonly string UploadsDirectory;
|
||||
private readonly string ThumbnailsDirectory;
|
||||
|
||||
public GalleryController(IWebHostEnvironment hostingEnvironment, IConfigHelper config, ILogger<GalleryController> logger, IStringLocalizer<GalleryController> localizer)
|
||||
public GalleryController(IWebHostEnvironment hostingEnvironment, IConfigHelper config, IDatabaseHelper database, IFileHelper fileHelper, IGalleryHelper galleryHelper, IDeviceDetector deviceDetector, IImageHelper imageHelper, INotificationHelper notificationHelper, ILogger<GalleryController> logger, IStringLocalizer<GalleryController> localizer)
|
||||
{
|
||||
_hostingEnvironment = hostingEnvironment;
|
||||
_config = config;
|
||||
_database = database;
|
||||
_fileHelper = fileHelper;
|
||||
_gallery = galleryHelper;
|
||||
_deviceDetector = deviceDetector;
|
||||
_imageHelper = imageHelper;
|
||||
_notificationHelper = notificationHelper;
|
||||
_logger = logger;
|
||||
_localizer = localizer;
|
||||
|
||||
UploadsDirectory = Path.Combine(_hostingEnvironment.WebRootPath, "uploads");
|
||||
ThumbnailsDirectory = Path.Combine(_hostingEnvironment.WebRootPath, "thumbnails");
|
||||
}
|
||||
|
||||
[HttpGet]
|
||||
public IActionResult Index(string id, string? key)
|
||||
[RequiresSecretKey]
|
||||
[AllowGuestCreate]
|
||||
[ResponseCache(Duration = 0, Location = ResponseCacheLocation.None, NoStore = true)]
|
||||
public async Task<IActionResult> Index(string id = "default", string? key = null, ViewMode? mode = null, GalleryOrder order = GalleryOrder.None)
|
||||
{
|
||||
if (_config.GetOrDefault("Settings", "Single_Gallery_Mode", false))
|
||||
id = (!string.IsNullOrWhiteSpace(id) && !_config.GetOrDefault("Settings", "Single_Gallery_Mode", false)) ? id.ToLower() : "default";
|
||||
|
||||
try
|
||||
{
|
||||
id = "default";
|
||||
ViewBag.ViewMode = mode ?? (ViewMode)_config.GetOrDefault("Settings", "Default_Gallery_View", (int)ViewMode.Default);
|
||||
}
|
||||
catch
|
||||
{
|
||||
ViewBag.ViewMode = ViewMode.Default;
|
||||
}
|
||||
|
||||
id = id.ToLower();
|
||||
|
||||
var secretKey = _config.Get("Settings", $"Secret_Key_{id}");
|
||||
if (string.IsNullOrEmpty(secretKey))
|
||||
var deviceType = HttpContext.Session.GetString("DeviceType");
|
||||
if (string.IsNullOrWhiteSpace(deviceType))
|
||||
{
|
||||
secretKey = _config.Get("Settings", "Secret_Key");
|
||||
deviceType = (await _deviceDetector.ParseDeviceType(Request.Headers["User-Agent"].ToString())).ToString();
|
||||
HttpContext.Session.SetString("DeviceType", deviceType ?? "Desktop");
|
||||
}
|
||||
|
||||
if (!string.IsNullOrEmpty(secretKey) && !string.Equals(secretKey, key))
|
||||
{
|
||||
_logger.LogWarning(_localizer["Invalid_Security_Key_Warning"].Value);
|
||||
ViewBag.ErrorMessage = _localizer["Invalid_Gallery_Key"].Value;
|
||||
|
||||
return View("~/Views/Home/Index.cshtml");
|
||||
}
|
||||
else if (string.IsNullOrEmpty(id))
|
||||
{
|
||||
ViewBag.ErrorMessage = _localizer["Invalid_Gallery_Id"].Value;
|
||||
|
||||
return View("~/Views/Home/Index.cshtml");
|
||||
}
|
||||
|
||||
ViewBag.SecretKey = key;
|
||||
|
||||
var dd = new DeviceDetectorNET.DeviceDetector(Request.Headers["User-Agent"].ToString());
|
||||
dd.Parse();
|
||||
|
||||
ViewBag.IsMobile = dd.IsParsed() && dd.IsMobile();
|
||||
ViewBag.IsMobile = !string.Equals("Desktop", deviceType, StringComparison.OrdinalIgnoreCase);
|
||||
|
||||
var galleryPath = Path.Combine(UploadsDirectory, id);
|
||||
var allowedFileTypes = _config.GetOrDefault("Settings", "Allowed_File_Types", ".jpg,.jpeg,.png").Split(',', StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries);
|
||||
var files = Directory.Exists(galleryPath) ? Directory.GetFiles(galleryPath, "*.*", SearchOption.TopDirectoryOnly)?.Where(x => allowedFileTypes.Any(y => string.Equals(Path.GetExtension(x).Trim('.'), y.Trim('.'), StringComparison.OrdinalIgnoreCase))) : null;
|
||||
var pendingPath = Path.Combine(galleryPath, "Pending");
|
||||
var images = new PhotoGallery(_config.GetOrDefault("Settings", "Gallery_Columns", 4))
|
||||
{
|
||||
GalleryId = id,
|
||||
GalleryPath = $"/{galleryPath.Remove(_hostingEnvironment.WebRootPath).Replace('\\', '/').TrimStart('/')}",
|
||||
Images = files?.OrderByDescending(x => new FileInfo(x).CreationTimeUtc)?.Select(x => Path.GetFileName(x))?.ToList(),
|
||||
PendingCount = Directory.Exists(pendingPath) ? Directory.GetFiles(pendingPath, "*.*", SearchOption.TopDirectoryOnly).Length : 0,
|
||||
FileUploader = !_config.GetOrDefault("Settings", "Disable_Upload", false) || (User?.Identity != null && User.Identity.IsAuthenticated) ? new FileUploader(id, "/Gallery/UploadImage") : null
|
||||
};
|
||||
_fileHelper.CreateDirectoryIfNotExists(galleryPath);
|
||||
_fileHelper.CreateDirectoryIfNotExists(Path.Combine(galleryPath, "Pending"));
|
||||
|
||||
return View(images);
|
||||
GalleryModel? gallery = await _database.GetGallery(id);
|
||||
if (gallery == null)
|
||||
{
|
||||
gallery = await _database.AddGallery(new GalleryModel()
|
||||
{
|
||||
Name = id.ToLower(),
|
||||
SecretKey = key
|
||||
});
|
||||
}
|
||||
|
||||
if (gallery != null)
|
||||
{
|
||||
var secretKey = await _gallery.GetSecretKey(gallery.Name);
|
||||
ViewBag.SecretKey = secretKey;
|
||||
|
||||
var allowedFileTypes = _config.GetOrDefault("Settings", "Allowed_File_Types", ".jpg,.jpeg,.png").Split(',', StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries);
|
||||
var images = (await _database.GetAllGalleryItems(gallery.Id, GalleryItemState.Approved))?.Where(x => allowedFileTypes.Any(y => string.Equals(Path.GetExtension(x.Title).Trim('.'), y.Trim('.'), StringComparison.OrdinalIgnoreCase)));
|
||||
switch (order)
|
||||
{
|
||||
case GalleryOrder.UploadedAsc:
|
||||
images = images?.OrderBy(x => x.Id);
|
||||
break;
|
||||
case GalleryOrder.UploadedDesc:
|
||||
images = images?.OrderByDescending(x => x.Id);
|
||||
break;
|
||||
case GalleryOrder.NameAsc:
|
||||
images = images?.OrderByDescending(x => x.Title);
|
||||
break;
|
||||
case GalleryOrder.NameDesc:
|
||||
images = images?.OrderBy(x => x.Title);
|
||||
break;
|
||||
case GalleryOrder.Random:
|
||||
images = images?.OrderBy(x => Guid.NewGuid());
|
||||
break;
|
||||
default:
|
||||
images = images?.OrderByDescending(x => x.Id);
|
||||
break;
|
||||
}
|
||||
|
||||
var model = new PhotoGallery()
|
||||
{
|
||||
GalleryId = id,
|
||||
GalleryPath = $"/{galleryPath.Remove(_hostingEnvironment.WebRootPath).Replace('\\', '/').TrimStart('/')}",
|
||||
ThumbnailsPath = $"/{ThumbnailsDirectory.Remove(_hostingEnvironment.WebRootPath).Replace('\\', '/').TrimStart('/')}",
|
||||
Images = images?.Select(x => new PhotoGalleryImage() { Id = x.Id, Name = Path.GetFileName(x.Title), Path = x.Title, UploadedBy = x.UploadedBy })?.ToList(),
|
||||
PendingCount = gallery?.PendingItems ?? 0,
|
||||
FileUploader = !_gallery.GetConfig(id, "Disable_Upload", false) || (User?.Identity != null && User.Identity.IsAuthenticated) ? new FileUploader(id, secretKey, "/Gallery/UploadImage") : null,
|
||||
ViewMode = (ViewMode)ViewBag.ViewMode
|
||||
};
|
||||
|
||||
return View(model);
|
||||
}
|
||||
|
||||
return View(new PhotoGallery());
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
public async Task<IActionResult> UploadImage()
|
||||
{
|
||||
Response.StatusCode = (int)HttpStatusCode.BadRequest;
|
||||
|
||||
try
|
||||
{
|
||||
var secretKey = _config.Get("Settings", "Secret_Key");
|
||||
var key = Request?.Form?.FirstOrDefault(x => string.Equals("SecretKey", x.Key, StringComparison.OrdinalIgnoreCase)).Value;
|
||||
if (!string.IsNullOrEmpty(secretKey) && !string.Equals(secretKey, key))
|
||||
string galleryId = (Request?.Form?.FirstOrDefault(x => string.Equals("Id", x.Key, StringComparison.OrdinalIgnoreCase)).Value)?.ToString()?.ToLower() ?? string.Empty;
|
||||
if (string.IsNullOrWhiteSpace(galleryId))
|
||||
{
|
||||
_logger.LogWarning(_localizer["Invalid_Security_Key_Warning"].Value);
|
||||
throw new UnauthorizedAccessException(_localizer["Invalid_Access_Token"].Value);
|
||||
return Json(new { success = false, uploaded = 0, errors = new List<string>() { _localizer["Invalid_Gallery_Id"].Value } });
|
||||
}
|
||||
|
||||
string galleryId = Request?.Form?.FirstOrDefault(x => string.Equals("GalleryId", x.Key, StringComparison.OrdinalIgnoreCase)).Value ?? string.Empty;
|
||||
if (string.IsNullOrEmpty(galleryId))
|
||||
|
||||
var gallery = await _database.GetGallery(galleryId);
|
||||
if (gallery != null)
|
||||
{
|
||||
return Json(new { success = true, uploaded = 0, errors = new List<string>() { _localizer["Invalid_Gallery_Id"].Value } });
|
||||
}
|
||||
|
||||
galleryId = galleryId.ToLower();
|
||||
|
||||
var galleryPath = Path.Combine(UploadsDirectory, galleryId);
|
||||
var files = Request?.Form?.Files;
|
||||
if (files != null && files.Count > 0)
|
||||
{
|
||||
if (!Directory.Exists(galleryPath))
|
||||
var secretKey = await _gallery.GetSecretKey(galleryId);
|
||||
string key = (Request?.Form?.FirstOrDefault(x => string.Equals("SecretKey", x.Key, StringComparison.OrdinalIgnoreCase)).Value)?.ToString() ?? string.Empty;
|
||||
if (!string.IsNullOrWhiteSpace(secretKey) && !string.Equals(secretKey, key))
|
||||
{
|
||||
Directory.CreateDirectory(galleryPath);
|
||||
return Json(new { success = false, uploaded = 0, errors = new List<string>() { _localizer["Invalid_Secret_Key_Warning"].Value } });
|
||||
}
|
||||
|
||||
var requiresReview = _config.GetOrDefault("Settings", "Require_Review", true);
|
||||
if (requiresReview)
|
||||
{
|
||||
galleryPath = Path.Combine(galleryPath, "Pending");
|
||||
if (!Directory.Exists(galleryPath))
|
||||
{
|
||||
Directory.CreateDirectory(galleryPath);
|
||||
}
|
||||
}
|
||||
|
||||
var uploaded = 0;
|
||||
var errors = new List<string>();
|
||||
foreach (IFormFile file in files)
|
||||
string uploadedBy = (Request?.Form?.FirstOrDefault(x => string.Equals("UploadedBy", x.Key, StringComparison.OrdinalIgnoreCase)).Value)?.ToString() ?? string.Empty;
|
||||
|
||||
var files = Request?.Form?.Files;
|
||||
if (files != null && files.Count > 0)
|
||||
{
|
||||
try
|
||||
{
|
||||
var extension = Path.GetExtension(file.FileName);
|
||||
var maxFilesSize = _config.GetOrDefault("Settings", "Max_File_Size_Mb", 10) * 1000000;
|
||||
var requiresReview = _config.GetOrDefault("Settings", "Require_Review", true);
|
||||
|
||||
var allowedFileTypes = _config.GetOrDefault("Settings", "Allowed_File_Types", ".jpg,.jpeg,.png").Split(',', StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries);
|
||||
if (!allowedFileTypes.Any(x => string.Equals(x.Trim('.'), extension.Trim('.'), StringComparison.OrdinalIgnoreCase)))
|
||||
var uploaded = 0;
|
||||
var errors = new List<string>();
|
||||
foreach (IFormFile file in files)
|
||||
{
|
||||
try
|
||||
{
|
||||
errors.Add($"{_localizer["File_Upload_Failed"].Value} '{Path.GetFileName(file.FileName)}'. {_localizer["Invalid_File_Type"].Value}");
|
||||
}
|
||||
else if (file.Length > maxFilesSize)
|
||||
{
|
||||
errors.Add($"{_localizer["File_Upload_Failed"].Value} '{Path.GetFileName(file.FileName)}'. {_localizer["Max_File_Size"].Value} {maxFilesSize} bytes");
|
||||
}
|
||||
else
|
||||
{
|
||||
var filePath = Path.Combine(galleryPath, $"{Guid.NewGuid()}{Path.GetExtension(file.FileName)}");
|
||||
if (!string.IsNullOrEmpty(filePath))
|
||||
var extension = Path.GetExtension(file.FileName);
|
||||
var maxFilesSize = _config.GetOrDefault("Settings", "Max_File_Size_Mb", 10) * 1000000;
|
||||
|
||||
var allowedFileTypes = _config.GetOrDefault("Settings", "Allowed_File_Types", ".jpg,.jpeg,.png").Split(',', StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries);
|
||||
if (!allowedFileTypes.Any(x => string.Equals(x.Trim('.'), extension.Trim('.'), StringComparison.OrdinalIgnoreCase)))
|
||||
{
|
||||
using (var fs = new FileStream(filePath, FileMode.Create))
|
||||
errors.Add($"{_localizer["File_Upload_Failed"].Value} '{Path.GetFileName(file.FileName)}'. {_localizer["Invalid_File_Type"].Value}");
|
||||
}
|
||||
else if (file.Length > maxFilesSize)
|
||||
{
|
||||
errors.Add($"{_localizer["File_Upload_Failed"].Value} '{Path.GetFileName(file.FileName)}'. {_localizer["Max_File_Size"].Value} {maxFilesSize} bytes");
|
||||
}
|
||||
else
|
||||
{
|
||||
var fileName = $"{Guid.NewGuid()}{Path.GetExtension(file.FileName)}";
|
||||
var galleryPath = requiresReview ? Path.Combine(UploadsDirectory, gallery.Name, "Pending") : Path.Combine(UploadsDirectory, gallery.Name);
|
||||
|
||||
_fileHelper.CreateDirectoryIfNotExists(galleryPath);
|
||||
|
||||
var filePath = Path.Combine(galleryPath, fileName);
|
||||
if (!string.IsNullOrWhiteSpace(filePath))
|
||||
{
|
||||
await file.CopyToAsync(fs);
|
||||
uploaded++;
|
||||
await _fileHelper.SaveFile(file, filePath, FileMode.Create);
|
||||
|
||||
if (!requiresReview)
|
||||
{
|
||||
_fileHelper.CreateDirectoryIfNotExists(ThumbnailsDirectory);
|
||||
await _imageHelper.GenerateThumbnail(filePath, Path.Combine(ThumbnailsDirectory, $"{Path.GetFileNameWithoutExtension(filePath)}.webp"), _config.GetOrDefault("Settings", "Thumbnail_Size", 720));
|
||||
}
|
||||
|
||||
var item = await _database.AddGalleryItem(new GalleryItemModel()
|
||||
{
|
||||
GalleryId = gallery.Id,
|
||||
Title = fileName,
|
||||
UploadedBy = uploadedBy,
|
||||
State = requiresReview ? GalleryItemState.Pending : GalleryItemState.Approved
|
||||
});
|
||||
|
||||
if (item?.Id > 0)
|
||||
{
|
||||
uploaded++;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"{_localizer["Save_To_Gallery_Failed"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"{_localizer["Save_To_Gallery_Failed"].Value} - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return Json(new { success = true, uploaded, requiresReview, errors });
|
||||
Response.StatusCode = (int)HttpStatusCode.OK;
|
||||
|
||||
var filesUploaded = uploaded > 0;
|
||||
if (filesUploaded && requiresReview && _config.GetOrDefault("Notifications", "Alerts", "Pending_Review", true))
|
||||
{
|
||||
await _notificationHelper.Send("New Items Pending Review", $"{uploaded} new item(s) have been uploaded to gallery '{gallery.Name}' by '{(!string.IsNullOrWhiteSpace(uploadedBy) ? uploadedBy : "Anonymous")}' and are awaiting your review.", UrlHelper.Generate(HttpContext, _config, "/Admin"));
|
||||
}
|
||||
|
||||
return Json(new { success = filesUploaded, uploaded, uploadedBy, requiresReview, errors });
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, uploaded = 0, errors = new List<string>() { _localizer["No_Files_For_Upload"].Value } });
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return Json(new { success = false, uploaded = 0, errors = new List<string>() { _localizer["Gallery_Does_Not_Exist"].Value } });
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
|
||||
@@ -7,15 +7,45 @@ namespace WeddingShare.Controllers
|
||||
[AllowAnonymous]
|
||||
public class HomeController : Controller
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly IGalleryHelper _gallery;
|
||||
private readonly IDeviceDetector _deviceDetector;
|
||||
private readonly ILogger _logger;
|
||||
public HomeController(IConfigHelper config, ILogger<HomeController> logger)
|
||||
|
||||
public HomeController(IConfigHelper config, IGalleryHelper gallery, IDeviceDetector deviceDetector, ILogger<HomeController> logger)
|
||||
{
|
||||
_config = config;
|
||||
_gallery = gallery;
|
||||
_deviceDetector = deviceDetector;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
[HttpGet]
|
||||
public IActionResult Index()
|
||||
public async Task<IActionResult> Index()
|
||||
{
|
||||
try
|
||||
{
|
||||
var deviceType = HttpContext.Session.GetString("DeviceType");
|
||||
if (string.IsNullOrWhiteSpace(deviceType))
|
||||
{
|
||||
deviceType = (await _deviceDetector.ParseDeviceType(Request.Headers["User-Agent"].ToString())).ToString();
|
||||
HttpContext.Session.SetString("DeviceType", deviceType ?? "Desktop");
|
||||
}
|
||||
|
||||
if (_config.GetOrDefault("Settings", "Single_Gallery_Mode", false))
|
||||
{
|
||||
var key = await _gallery.GetSecretKey("default");
|
||||
if (string.IsNullOrWhiteSpace(key))
|
||||
{
|
||||
return RedirectToAction("Index", "Gallery");
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"An error occurred loading the homepage - {ex?.Message}");
|
||||
}
|
||||
|
||||
return View();
|
||||
}
|
||||
}
|
||||
|
||||
8
WeddingShare/Enums/DatabaseType.cs
Normal file
8
WeddingShare/Enums/DatabaseType.cs
Normal file
@@ -0,0 +1,8 @@
|
||||
namespace WeddingShare.Enums
|
||||
{
|
||||
public enum DatabaseType
|
||||
{
|
||||
Unknown,
|
||||
SQLite
|
||||
}
|
||||
}
|
||||
10
WeddingShare/Enums/DeviceType.cs
Normal file
10
WeddingShare/Enums/DeviceType.cs
Normal file
@@ -0,0 +1,10 @@
|
||||
namespace WeddingShare.Enums
|
||||
{
|
||||
public enum DeviceType
|
||||
{
|
||||
Unknown,
|
||||
Desktop,
|
||||
Tablet,
|
||||
Mobile
|
||||
}
|
||||
}
|
||||
9
WeddingShare/Enums/GalleryItemState.cs
Normal file
9
WeddingShare/Enums/GalleryItemState.cs
Normal file
@@ -0,0 +1,9 @@
|
||||
namespace WeddingShare.Enums
|
||||
{
|
||||
public enum GalleryItemState
|
||||
{
|
||||
Pending = 0,
|
||||
Approved = 1,
|
||||
All = 99
|
||||
}
|
||||
}
|
||||
12
WeddingShare/Enums/GalleryOrder.cs
Normal file
12
WeddingShare/Enums/GalleryOrder.cs
Normal file
@@ -0,0 +1,12 @@
|
||||
namespace WeddingShare.Enums
|
||||
{
|
||||
public enum GalleryOrder
|
||||
{
|
||||
None,
|
||||
UploadedAsc,
|
||||
UploadedDesc,
|
||||
NameAsc,
|
||||
NameDesc,
|
||||
Random
|
||||
}
|
||||
}
|
||||
10
WeddingShare/Enums/ImageOrientation.cs
Normal file
10
WeddingShare/Enums/ImageOrientation.cs
Normal file
@@ -0,0 +1,10 @@
|
||||
namespace WeddingShare.Enums
|
||||
{
|
||||
public enum ImageOrientation
|
||||
{
|
||||
None,
|
||||
Square,
|
||||
Portrait,
|
||||
Landscape
|
||||
}
|
||||
}
|
||||
9
WeddingShare/Enums/ViewMode.cs
Normal file
9
WeddingShare/Enums/ViewMode.cs
Normal file
@@ -0,0 +1,9 @@
|
||||
namespace WeddingShare.Enums
|
||||
{
|
||||
public enum ViewMode
|
||||
{
|
||||
Default,
|
||||
Presentation,
|
||||
Slideshow
|
||||
}
|
||||
}
|
||||
@@ -3,24 +3,34 @@
|
||||
public interface IConfigHelper
|
||||
{
|
||||
string? GetEnvironmentVariable(string key);
|
||||
string? GetConfigValue(string section, string key);
|
||||
string? GetConfigValue(string section, string? subsection, string key);
|
||||
string? Get(string section, string key);
|
||||
string? Get(string section, string? subsection, string key);
|
||||
string GetOrDefault(string section, string key, string defaultValue);
|
||||
string GetOrDefault(string section, string? subsection, string key, string defaultValue);
|
||||
int GetOrDefault(string section, string key, int defaultValue);
|
||||
int GetOrDefault(string section, string? subsection, string key, int defaultValue);
|
||||
long GetOrDefault(string section, string key, long defaultValue);
|
||||
long GetOrDefault(string section, string? subsection, string key, long defaultValue);
|
||||
decimal GetOrDefault(string section, string key, decimal defaultValue);
|
||||
decimal GetOrDefault(string section, string? subsection, string key, decimal defaultValue);
|
||||
double GetOrDefault(string section, string key, double defaultValue);
|
||||
double GetOrDefault(string section, string? subsection, string key, double defaultValue);
|
||||
bool GetOrDefault(string section, string key, bool defaultValue);
|
||||
bool GetOrDefault(string section, string? subsection, string key, bool defaultValue);
|
||||
DateTime? GetOrDefault(string section, string key, DateTime? defaultValue);
|
||||
DateTime? GetOrDefault(string section, string? subsection, string key, DateTime? defaultValue);
|
||||
}
|
||||
|
||||
public class ConfigHelper : IConfigHelper
|
||||
{
|
||||
private readonly IEnvironmentWrapper _environment;
|
||||
private readonly IConfiguration _configuration;
|
||||
private readonly ILogger<ConfigHelper> _logger;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public ConfigHelper(IConfiguration config, ILogger<ConfigHelper> logger)
|
||||
{
|
||||
public ConfigHelper(IEnvironmentWrapper environment, IConfiguration config, ILogger<ConfigHelper> logger)
|
||||
{
|
||||
_environment = environment;
|
||||
_configuration = config;
|
||||
_logger = logger;
|
||||
}
|
||||
@@ -29,8 +39,8 @@
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = Environment.GetEnvironmentVariable(key.Replace(":", "_").ToUpper());
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = _environment.GetEnvironmentVariable(key.Replace(":", "_").Trim('_').ToUpper());
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return value;
|
||||
}
|
||||
@@ -43,12 +53,15 @@
|
||||
return null;
|
||||
}
|
||||
|
||||
public string? GetConfigValue(string section, string key)
|
||||
public string? GetConfigValue(string section, string? subsection, string key)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = _configuration.GetValue<string>(!string.IsNullOrEmpty(section) ? $"{section}:{key}" : key);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
section = !string.IsNullOrWhiteSpace(subsection) ? $"{section}:{subsection}" : section;
|
||||
|
||||
var configKey = !string.IsNullOrWhiteSpace(section) ? $"{section}:{key}" : key;
|
||||
var value = _configuration.GetValue<string>(configKey);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return value;
|
||||
}
|
||||
@@ -62,17 +75,22 @@
|
||||
}
|
||||
|
||||
public string? Get(string section, string key)
|
||||
{
|
||||
return this.Get(section, null, key);
|
||||
}
|
||||
|
||||
public string? Get(string section, string? subsection, string key)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = !IsProtectedVariable($"{section}_{key}") ? this.GetEnvironmentVariable(key) : string.Empty;
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = !IsProtectedVariable($"{CombineSections(section, subsection)}_{key}") ? this.GetEnvironmentVariable($"{subsection}_{key}") : string.Empty;
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return value;
|
||||
}
|
||||
|
||||
value = this.GetConfigValue(section, key);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
value = this.GetConfigValue(section, subsection, key);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return value;
|
||||
}
|
||||
@@ -86,11 +104,16 @@
|
||||
}
|
||||
|
||||
public string GetOrDefault(string section, string key, string defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public string GetOrDefault(string section, string? subsection, string key, string defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.Get(section, key);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.Get(section, subsection, key);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return value;
|
||||
}
|
||||
@@ -101,11 +124,16 @@
|
||||
}
|
||||
|
||||
public int GetOrDefault(string section, string key, int defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public int GetOrDefault(string section, string? subsection, string key, int defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetOrDefault(section, key, string.Empty);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.GetOrDefault(section, subsection, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToInt32(value);
|
||||
}
|
||||
@@ -116,11 +144,16 @@
|
||||
}
|
||||
|
||||
public long GetOrDefault(string section, string key, long defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public long GetOrDefault(string section, string? subsection, string key, long defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetOrDefault(section, key, string.Empty);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.GetOrDefault(section, subsection, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToInt64(value);
|
||||
}
|
||||
@@ -131,11 +164,16 @@
|
||||
}
|
||||
|
||||
public decimal GetOrDefault(string section, string key, decimal defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public decimal GetOrDefault(string section, string? subsection, string key, decimal defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetOrDefault(section, key, string.Empty);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.GetOrDefault(section, subsection, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToDecimal(value);
|
||||
}
|
||||
@@ -146,11 +184,16 @@
|
||||
}
|
||||
|
||||
public double GetOrDefault(string section, string key, double defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public double GetOrDefault(string section, string? subsection, string key, double defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetOrDefault(section, key, string.Empty);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.GetOrDefault(section, subsection, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToDouble(value);
|
||||
}
|
||||
@@ -161,11 +204,16 @@
|
||||
}
|
||||
|
||||
public bool GetOrDefault(string section, string key, bool defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public bool GetOrDefault(string section, string? subsection, string key, bool defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetOrDefault(section, key, string.Empty);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.GetOrDefault(section, subsection, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToBoolean(value);
|
||||
}
|
||||
@@ -176,11 +224,16 @@
|
||||
}
|
||||
|
||||
public DateTime? GetOrDefault(string section, string key, DateTime? defaultValue)
|
||||
{
|
||||
return this.GetOrDefault(section, null, key, defaultValue);
|
||||
}
|
||||
|
||||
public DateTime? GetOrDefault(string section, string? subsection, string key, DateTime? defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetOrDefault(section, key, string.Empty);
|
||||
if (!string.IsNullOrEmpty(value))
|
||||
var value = this.GetOrDefault(section, subsection, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToDateTime(value);
|
||||
}
|
||||
@@ -190,6 +243,11 @@
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
private string CombineSections(string section, string? subsection)
|
||||
{
|
||||
return $"{section}_{subsection}".Trim('_');
|
||||
}
|
||||
|
||||
private bool IsProtectedVariable(string key)
|
||||
{
|
||||
switch (key.Replace(":", "_").Trim('_').ToUpper())
|
||||
|
||||
38
WeddingShare/Helpers/Database/IDatabaseHelper.cs
Normal file
38
WeddingShare/Helpers/Database/IDatabaseHelper.cs
Normal file
@@ -0,0 +1,38 @@
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.Helpers.Database
|
||||
{
|
||||
public interface IDatabaseHelper
|
||||
{
|
||||
Task<List<GalleryModel>> GetAllGalleries();
|
||||
Task<GalleryModel?> GetGallery(int id);
|
||||
Task<GalleryModel?> GetGallery(string name);
|
||||
Task<GalleryModel?> AddGallery(GalleryModel model);
|
||||
Task<GalleryModel?> EditGallery(GalleryModel model);
|
||||
Task<bool> WipeGallery(GalleryModel model);
|
||||
Task<bool> WipeAllGalleries();
|
||||
Task<bool> DeleteGallery(GalleryModel model);
|
||||
Task<List<GalleryItemModel>> GetAllGalleryItems(int galleryId, GalleryItemState state = GalleryItemState.All);
|
||||
Task<int> GetPendingGalleryItemCount(int? galleryId = null);
|
||||
Task<List<PendingGalleryItemModel>> GetPendingGalleryItems(int? galleryId = null);
|
||||
Task<PendingGalleryItemModel?> GetPendingGalleryItem(int id);
|
||||
Task<GalleryItemModel?> GetGalleryItem(int id);
|
||||
Task<GalleryItemModel?> AddGalleryItem(GalleryItemModel model);
|
||||
Task<GalleryItemModel?> EditGalleryItem(GalleryItemModel model);
|
||||
Task<bool> DeleteGalleryItem(GalleryItemModel model);
|
||||
Task<bool> InitAdminAccount(UserModel model);
|
||||
Task<bool> ValidateCredentials(string username, string password);
|
||||
Task<UserModel?> GetUser(int id);
|
||||
Task<UserModel?> GetUser(string name);
|
||||
Task<UserModel?> AddUser(UserModel model);
|
||||
Task<UserModel?> EditUser(UserModel model);
|
||||
Task<bool> DeleteUser(UserModel model);
|
||||
Task<bool> ChangePassword(UserModel model);
|
||||
Task<int> IncrementLockoutCount(int id);
|
||||
Task<bool> SetLockout(int id, DateTime? datetime);
|
||||
Task<bool> ResetLockoutCount(int id);
|
||||
Task<bool> Import(string path);
|
||||
Task<bool> Export(string path);
|
||||
}
|
||||
}
|
||||
898
WeddingShare/Helpers/Database/SQLiteDatabaseHelper.cs
Normal file
898
WeddingShare/Helpers/Database/SQLiteDatabaseHelper.cs
Normal file
@@ -0,0 +1,898 @@
|
||||
using System.Data;
|
||||
using Microsoft.Data.Sqlite;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.Helpers.Database
|
||||
{
|
||||
public class SQLiteDatabaseHelper : IDatabaseHelper
|
||||
{
|
||||
private readonly string _connString;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public SQLiteDatabaseHelper(IConfigHelper config, ILogger<SQLiteDatabaseHelper> logger)
|
||||
{
|
||||
_connString = config.GetOrDefault("Database", "Connection_String", "Data Source=./config/wedding-share.db");
|
||||
_logger = logger;
|
||||
|
||||
_logger.LogInformation($"Using SQLite connection string: '{_connString}'");
|
||||
}
|
||||
|
||||
#region Gallery
|
||||
public async Task<List<GalleryModel>> GetAllGalleries()
|
||||
{
|
||||
List<GalleryModel> result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT g.*, COUNT(gi.`id`) AS `total`, SUM(CASE WHEN gi.`state`=@ApprovedState THEN 1 ELSE 0 END) AS `approved`, SUM(CASE WHEN gi.`state`=@PendingState THEN 1 ELSE 0 END) AS `pending` FROM `galleries` AS g LEFT JOIN `gallery_items` AS gi ON g.`id` = gi.`gallery_id` GROUP BY g.`id` ORDER BY `name` ASC;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("PendingState", (int)GalleryItemState.Pending);
|
||||
cmd.Parameters.AddWithValue("ApprovedState", (int)GalleryItemState.Approved);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = await ReadGalleries(await cmd.ExecuteReaderAsync());
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryModel?> GetGallery(int id)
|
||||
{
|
||||
GalleryModel? result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT g.*, COUNT(gi.`id`) AS `total`, SUM(CASE WHEN gi.`state`=@ApprovedState THEN 1 ELSE 0 END) AS `approved`, SUM(CASE WHEN gi.`state`=@PendingState THEN 1 ELSE 0 END) AS `pending` FROM `galleries` AS g LEFT JOIN `gallery_items` AS gi ON g.`id` = gi.`gallery_id` WHERE g.`id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", id);
|
||||
cmd.Parameters.AddWithValue("PendingState", (int)GalleryItemState.Pending);
|
||||
cmd.Parameters.AddWithValue("ApprovedState", (int)GalleryItemState.Approved);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (await ReadGalleries(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryModel?> GetGallery(string name)
|
||||
{
|
||||
GalleryModel? result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT g.*, COUNT(gi.`id`) AS `total`, SUM(CASE WHEN gi.`state`=@ApprovedState THEN 1 ELSE 0 END) AS `approved`, SUM(CASE WHEN gi.`state`=@PendingState THEN 1 ELSE 0 END) AS `pending` FROM `galleries` AS g LEFT JOIN `gallery_items` AS gi ON g.`id` = gi.`gallery_id` WHERE g.`name`=@Name;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Name", name?.ToLower());
|
||||
cmd.Parameters.AddWithValue("ApprovedState", (int)GalleryItemState.Approved);
|
||||
cmd.Parameters.AddWithValue("PendingState", (int)GalleryItemState.Pending);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (await ReadGalleries(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryModel?> AddGallery(GalleryModel model)
|
||||
{
|
||||
GalleryModel? result = null;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"INSERT INTO `galleries` (`name`, `secret_key`) VALUES (@Name, @SecretKey); SELECT g.*, COUNT(gi.`id`) AS `total`, SUM(CASE WHEN gi.`state`=@ApprovedState THEN 1 ELSE 0 END) AS `approved`, SUM(CASE WHEN gi.`state`=@PendingState THEN 1 ELSE 0 END) AS `pending` FROM `galleries` AS g LEFT JOIN `gallery_items` AS gi ON g.`id` = gi.`gallery_id` WHERE g.`id`=last_insert_rowid();", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Name", model.Name.ToLower());
|
||||
cmd.Parameters.AddWithValue("SecretKey", !string.IsNullOrWhiteSpace(model.SecretKey) ? model.SecretKey : DBNull.Value);
|
||||
cmd.Parameters.AddWithValue("ApprovedState", (int)GalleryItemState.Approved);
|
||||
cmd.Parameters.AddWithValue("PendingState", (int)GalleryItemState.Pending);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await ReadGalleries(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryModel?> EditGallery(GalleryModel model)
|
||||
{
|
||||
GalleryModel? result = null;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"UPDATE `galleries` SET `name`=@Name, `secret_key`=@SecretKey WHERE `id`=@Id; SELECT g.*, COUNT(gi.`id`) AS `total`, SUM(CASE WHEN gi.`state`=@ApprovedState THEN 1 ELSE 0 END) AS `approved`, SUM(CASE WHEN gi.`state`=@PendingState THEN 1 ELSE 0 END) AS `pending` FROM `galleries` AS g LEFT JOIN `gallery_items` AS gi ON g.`id` = gi.`gallery_id` WHERE g.`id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
cmd.Parameters.AddWithValue("Name", model.Name?.ToLower());
|
||||
cmd.Parameters.AddWithValue("SecretKey", !string.IsNullOrWhiteSpace(model.SecretKey) ? model.SecretKey : DBNull.Value);
|
||||
cmd.Parameters.AddWithValue("ApprovedState", (int)GalleryItemState.Approved);
|
||||
cmd.Parameters.AddWithValue("PendingState", (int)GalleryItemState.Pending);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await ReadGalleries(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> WipeGallery(GalleryModel model)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"DELETE FROM `gallery_items` WHERE `gallery_id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await cmd.ExecuteNonQueryAsync()) > 0;
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> WipeAllGalleries()
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"DELETE FROM `gallery_items`; DELETE FROM `galleries` WHERE `id` > 1;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await cmd.ExecuteNonQueryAsync()) > 0;
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> DeleteGallery(GalleryModel model)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"DELETE FROM `gallery_items` WHERE `gallery_id`=@Id; DELETE FROM `galleries` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await cmd.ExecuteNonQueryAsync()) > 0;
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Gallery Items
|
||||
public async Task<List<GalleryItemModel>> GetAllGalleryItems(int galleryId, GalleryItemState state = GalleryItemState.All)
|
||||
{
|
||||
List<GalleryItemModel> result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT * FROM `gallery_items` WHERE `gallery_id`=@Id {(state != GalleryItemState.All ? "AND `state`=@State" : string.Empty)} ORDER BY `id` ASC;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", galleryId);
|
||||
cmd.Parameters.AddWithValue("State", state);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = await ReadGalleryItems(await cmd.ExecuteReaderAsync());
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<List<PendingGalleryItemModel>> GetPendingGalleryItems(int? galleryId = null)
|
||||
{
|
||||
List<PendingGalleryItemModel> result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT g.`name` AS `gallery_name`, gi.* FROM `gallery_items` AS gi LEFT JOIN `galleries` AS g ON g.`id` = gi.`gallery_id` WHERE gi.`state`=@State {(galleryId != null ? "AND gi.`gallery_id`=@GalleryId" : string.Empty)};", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("GalleryId", galleryId);
|
||||
cmd.Parameters.AddWithValue("State", (int)GalleryItemState.Pending);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = await ReadPendingGalleryItems(await cmd.ExecuteReaderAsync());
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<PendingGalleryItemModel?> GetPendingGalleryItem(int id)
|
||||
{
|
||||
PendingGalleryItemModel? result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT g.`name` AS `gallery_name`, gi.* FROM `gallery_items` AS gi LEFT JOIN `galleries` AS g ON g.`id` = gi.`gallery_id` WHERE gi.`id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (await ReadPendingGalleryItems(await cmd.ExecuteReaderAsync())).FirstOrDefault();
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<int> GetPendingGalleryItemCount(int? galleryId = null)
|
||||
{
|
||||
int result = 0;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT COUNT(`id`) FROM `gallery_items` {(galleryId != null ? "WHERE `gallery_id`=@GalleryId" : string.Empty)};", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("GalleryId", galleryId);
|
||||
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (int)(await cmd.ExecuteScalarAsync() ?? 0);
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryItemModel?> GetGalleryItem(int id)
|
||||
{
|
||||
GalleryItemModel? result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT * FROM `gallery_items` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (await ReadGalleryItems(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryItemModel?> AddGalleryItem(GalleryItemModel model)
|
||||
{
|
||||
GalleryItemModel? result = null;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"INSERT INTO `gallery_items` (`gallery_id`, `title`, `state`, `uploaded_by`) VALUES (@GalleryId, @Title, @State, @UploadedBy); SELECT * FROM `gallery_items` WHERE `id`=last_insert_rowid();", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("GalleryId", model.GalleryId);
|
||||
cmd.Parameters.AddWithValue("Title", model.Title);
|
||||
cmd.Parameters.AddWithValue("State", (int)model.State);
|
||||
cmd.Parameters.AddWithValue("UploadedBy", !string.IsNullOrWhiteSpace(model.UploadedBy) ? model.UploadedBy : DBNull.Value);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await ReadGalleryItems(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<GalleryItemModel?> EditGalleryItem(GalleryItemModel model)
|
||||
{
|
||||
GalleryItemModel? result = null;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"UPDATE `gallery_items` SET `title`=@Title, `state`=@State, `uploaded_by`=@UploadedBy WHERE `id`=@Id; SELECT * FROM `gallery_items` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
cmd.Parameters.AddWithValue("Title", model.Title);
|
||||
cmd.Parameters.AddWithValue("State", (int)model.State);
|
||||
cmd.Parameters.AddWithValue("UploadedBy", !string.IsNullOrWhiteSpace(model.UploadedBy) ? model.UploadedBy : DBNull.Value);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await ReadGalleryItems(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> DeleteGalleryItem(GalleryItemModel model)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"DELETE FROM `gallery_items` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await cmd.ExecuteNonQueryAsync()) > 0;
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Users
|
||||
public async Task<bool> InitAdminAccount(UserModel model)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"UPDATE `users` SET `username`=@Username, `password`=@Password WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", 1);
|
||||
cmd.Parameters.AddWithValue("Username", model.Username.ToLower());
|
||||
cmd.Parameters.AddWithValue("Password", model.Password);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = await cmd.ExecuteNonQueryAsync() > 0;
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> ValidateCredentials(string username, string password)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT COUNT(`id`) FROM `users` WHERE `username`=@Username AND `password`=@Password;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", 1);
|
||||
cmd.Parameters.AddWithValue("Username", username.ToLower());
|
||||
cmd.Parameters.AddWithValue("Password", password);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (long)(await cmd.ExecuteScalarAsync() ?? 0) > 0;
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<UserModel?> GetUser(int id)
|
||||
{
|
||||
UserModel? result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT * FROM `users` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (await ReadUsers(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<UserModel?> GetUser(string username)
|
||||
{
|
||||
UserModel? result;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"SELECT * FROM `users` WHERE `username`=@Username;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Username", username.ToLower());
|
||||
|
||||
await conn.OpenAsync();
|
||||
result = (await ReadUsers(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<UserModel?> AddUser(UserModel model)
|
||||
{
|
||||
UserModel? result = null;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"INSERT INTO `users` (`username`, `email`, `password`) VALUES (@Username, @Email, @Password); SELECT SELECT * FROM `users` WHERE `id`=last_insert_rowid();", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Username", model.Username.ToLower());
|
||||
cmd.Parameters.AddWithValue("Email", model.Email);
|
||||
cmd.Parameters.AddWithValue("Password", model.Password);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await ReadUsers(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<UserModel?> EditUser(UserModel model)
|
||||
{
|
||||
UserModel? result = null;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"UPDATE `users` SET `username`=@Username, `email`=@Email, `failed_logins`=@FailedLogins, `lockout_until`=@LockoutUntil WHERE `id`=@Id; SELECT * FROM `users` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
cmd.Parameters.AddWithValue("Username", model.Username.ToLower());
|
||||
cmd.Parameters.AddWithValue("Email", !string.IsNullOrEmpty(model.Email) ? model.Email : DBNull.Value);
|
||||
cmd.Parameters.AddWithValue("FailedLogins", model.FailedLogins);
|
||||
cmd.Parameters.AddWithValue("LockoutUntil", model.LockoutUntil != null ? ((DateTime)model.LockoutUntil - new DateTime(1970, 1, 1)).TotalSeconds : DBNull.Value);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await ReadUsers(await cmd.ExecuteReaderAsync()))?.FirstOrDefault();
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> DeleteUser(UserModel model)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
if (model.Id > 1 && !string.Equals("Admin", model.Username, StringComparison.OrdinalIgnoreCase))
|
||||
{
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"DELETE FROM `users` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (await cmd.ExecuteNonQueryAsync()) > 0;
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> ChangePassword(UserModel model)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
var cmd = new SqliteCommand($"UPDATE `users` SET `password`=@Password WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", model.Id);
|
||||
cmd.Parameters.AddWithValue("Password", model.Password);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
result = (int)(await cmd.ExecuteScalarAsync() ?? 0) > 0;
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<int> IncrementLockoutCount(int id)
|
||||
{
|
||||
int result = 0;
|
||||
|
||||
var user = await this.GetUser(id);
|
||||
if (user != null)
|
||||
{
|
||||
user.FailedLogins++;
|
||||
result = (await this.EditUser(user))?.FailedLogins ?? 0;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> SetLockout(int id, DateTime? datetime)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
if (datetime != null)
|
||||
{
|
||||
var lockout = (DateTime)datetime;
|
||||
datetime = new DateTime(lockout.Year, lockout.Month, lockout.Day, lockout.Hour, lockout.Minute, 0, DateTimeKind.Utc);
|
||||
}
|
||||
|
||||
var cmd = new SqliteCommand($"UPDATE `users` SET `lockout_until`=@LockoutUntil WHERE `id`=@Id; SELECT `lockout_until` FROM `users` WHERE `id`=@Id;", conn);
|
||||
cmd.CommandType = CommandType.Text;
|
||||
cmd.Parameters.AddWithValue("Id", id);
|
||||
cmd.Parameters.AddWithValue("LockoutUntil", datetime != null ? ((DateTime)datetime - new DateTime(1970, 1, 1)).TotalSeconds : DBNull.Value);
|
||||
|
||||
await conn.OpenAsync();
|
||||
var tran = await conn.BeginTransactionAsync();
|
||||
|
||||
try
|
||||
{
|
||||
cmd.Transaction = (SqliteTransaction)tran;
|
||||
var reader = await cmd.ExecuteReaderAsync();
|
||||
if (reader != null && reader.HasRows)
|
||||
{
|
||||
while (reader.Read())
|
||||
{
|
||||
try
|
||||
{
|
||||
result = (!await reader.IsDBNullAsync("lockout_until") ? DateTime.UnixEpoch.AddSeconds(reader.GetInt32("lockout_until")) : null) == datetime;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to parse user lockout from database - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
await tran.CommitAsync();
|
||||
}
|
||||
catch
|
||||
{
|
||||
await tran.RollbackAsync();
|
||||
}
|
||||
|
||||
await conn.CloseAsync();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> ResetLockoutCount(int id)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
var user = await this.GetUser(id);
|
||||
if (user != null)
|
||||
{
|
||||
user.FailedLogins = 0;
|
||||
result = ((await this.EditUser(user))?.FailedLogins ?? 0) == 0;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Backups
|
||||
public async Task<bool> Import(string path)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
try
|
||||
{
|
||||
using (var backup = new SqliteConnection(path))
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
{
|
||||
await backup.OpenAsync();
|
||||
await conn.OpenAsync();
|
||||
|
||||
backup.BackupDatabase(conn);
|
||||
|
||||
await conn.CloseAsync();
|
||||
await backup.CloseAsync();
|
||||
|
||||
SqliteConnection.ClearPool(backup);
|
||||
}
|
||||
|
||||
result = true;
|
||||
}
|
||||
catch { }
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public async Task<bool> Export(string path)
|
||||
{
|
||||
bool result = false;
|
||||
|
||||
try
|
||||
{
|
||||
using (var conn = new SqliteConnection(_connString))
|
||||
using (var backup = new SqliteConnection(path))
|
||||
{
|
||||
await conn.OpenAsync();
|
||||
await backup.OpenAsync();
|
||||
|
||||
conn.BackupDatabase(backup);
|
||||
|
||||
await backup.CloseAsync();
|
||||
await conn.CloseAsync();
|
||||
|
||||
SqliteConnection.ClearPool(backup);
|
||||
}
|
||||
|
||||
result = true;
|
||||
}
|
||||
catch { }
|
||||
|
||||
return result;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Data Parsers
|
||||
private async Task<List<GalleryModel>> ReadGalleries(SqliteDataReader? reader)
|
||||
{
|
||||
var items = new List<GalleryModel>();
|
||||
|
||||
if (reader != null && reader.HasRows)
|
||||
{
|
||||
while (reader.Read())
|
||||
{
|
||||
try
|
||||
{
|
||||
var id = !await reader.IsDBNullAsync("id") ? reader.GetInt32("id") : 0;
|
||||
if (id > 0)
|
||||
{
|
||||
items.Add(new GalleryModel()
|
||||
{
|
||||
Id = id,
|
||||
Name = !await reader.IsDBNullAsync("name") ? reader.GetString("name") : null,
|
||||
SecretKey = !await reader.IsDBNullAsync("secret_key") ? reader.GetString("secret_key") : null,
|
||||
TotalItems = !await reader.IsDBNullAsync("total") ? reader.GetInt32("total") : 0,
|
||||
ApprovedItems = !await reader.IsDBNullAsync("approved") ? reader.GetInt32("approved") : 0,
|
||||
PendingItems = !await reader.IsDBNullAsync("pending") ? reader.GetInt32("pending") : 0,
|
||||
});
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to parse gallery model from database - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return items;
|
||||
}
|
||||
|
||||
private async Task<List<GalleryItemModel>> ReadGalleryItems(SqliteDataReader? reader)
|
||||
{
|
||||
var items = new List<GalleryItemModel>();
|
||||
|
||||
if (reader != null && reader.HasRows)
|
||||
{
|
||||
while (reader.Read())
|
||||
{
|
||||
try
|
||||
{
|
||||
var id = !await reader.IsDBNullAsync("id") ? reader.GetInt32("id") : 0;
|
||||
if (id > 0)
|
||||
{
|
||||
items.Add(new GalleryItemModel()
|
||||
{
|
||||
Id = id,
|
||||
GalleryId = !await reader.IsDBNullAsync("gallery_id") ? reader.GetInt32("gallery_id") : 0,
|
||||
Title = !await reader.IsDBNullAsync("title") ? reader.GetString("title") : string.Empty,
|
||||
UploadedBy = !await reader.IsDBNullAsync("uploaded_by") ? reader.GetString("uploaded_by") : null,
|
||||
State = !await reader.IsDBNullAsync("state") ? (GalleryItemState)reader.GetInt32("state") : GalleryItemState.Pending
|
||||
});
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to parse gallery item model from database - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return items;
|
||||
}
|
||||
|
||||
private async Task<List<PendingGalleryItemModel>> ReadPendingGalleryItems(SqliteDataReader? reader)
|
||||
{
|
||||
var items = new List<PendingGalleryItemModel>();
|
||||
|
||||
if (reader != null && reader.HasRows)
|
||||
{
|
||||
while (reader.Read())
|
||||
{
|
||||
try
|
||||
{
|
||||
|
||||
var id = !await reader.IsDBNullAsync("id") ? reader.GetInt32("id") : 0;
|
||||
if (id > 0)
|
||||
{
|
||||
items.Add(new PendingGalleryItemModel()
|
||||
{
|
||||
Id = id,
|
||||
GalleryId = !await reader.IsDBNullAsync("gallery_id") ? reader.GetInt32("gallery_id") : 0,
|
||||
GalleryName = !await reader.IsDBNullAsync("gallery_name") ? reader.GetString("gallery_name") : "default",
|
||||
Title = !await reader.IsDBNullAsync("title") ? reader.GetString("title") : string.Empty,
|
||||
UploadedBy = !await reader.IsDBNullAsync("uploaded_by") ? reader.GetString("uploaded_by") : null,
|
||||
State = !await reader.IsDBNullAsync("state") ? (GalleryItemState)reader.GetInt32("state") : GalleryItemState.Pending
|
||||
});
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to parse pending gallery item model from database - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return items;
|
||||
}
|
||||
|
||||
private async Task<List<UserModel>> ReadUsers(SqliteDataReader? reader)
|
||||
{
|
||||
var items = new List<UserModel>();
|
||||
|
||||
if (reader != null && reader.HasRows)
|
||||
{
|
||||
while (reader.Read())
|
||||
{
|
||||
try
|
||||
{
|
||||
var id = !await reader.IsDBNullAsync("id") ? reader.GetInt32("id") : 0;
|
||||
if (id > 0)
|
||||
{
|
||||
items.Add(new UserModel()
|
||||
{
|
||||
Id = id,
|
||||
Username = !await reader.IsDBNullAsync("failed_logins") ? reader.GetString("username").ToLower() : string.Empty,
|
||||
Email = !await reader.IsDBNullAsync("email") ? reader.GetString("email") : null,
|
||||
Password = null,
|
||||
FailedLogins = !await reader.IsDBNullAsync("failed_logins") ? reader.GetInt32("failed_logins") : 0,
|
||||
LockoutUntil = !await reader.IsDBNullAsync("lockout_until") ? DateTime.UnixEpoch.AddSeconds(reader.GetInt32("lockout_until")) : null
|
||||
});
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to parse user model from database - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return items;
|
||||
}
|
||||
#endregion
|
||||
}
|
||||
}
|
||||
78
WeddingShare/Helpers/Dbup/DbupHelper.cs
Normal file
78
WeddingShare/Helpers/Dbup/DbupHelper.cs
Normal file
@@ -0,0 +1,78 @@
|
||||
using System.Reflection;
|
||||
using DbUp;
|
||||
using DbUp.Engine;
|
||||
using WeddingShare.Enums;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.Helpers.Dbup
|
||||
{
|
||||
public sealed class DbupMigrator(IEnvironmentWrapper environment, IConfiguration configuration, IDatabaseHelper database, IFileHelper fileHelper, ILoggerFactory loggerFactory) : BackgroundService
|
||||
{
|
||||
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
|
||||
{
|
||||
await Task.Run(() =>
|
||||
{
|
||||
var logger = loggerFactory.CreateLogger<DbupMigrator>();
|
||||
|
||||
fileHelper.CreateDirectoryIfNotExists("config");
|
||||
|
||||
var config = new ConfigHelper(environment, configuration, loggerFactory.CreateLogger<ConfigHelper>());
|
||||
var connString = config.GetOrDefault("Database", "Connection_String", "Data Source=./config/wedding-share.db");
|
||||
if (!string.IsNullOrWhiteSpace(connString))
|
||||
{
|
||||
DatabaseUpgradeResult? dbupResult;
|
||||
|
||||
var dbType = config.GetOrDefault("Database", "Database_Type", "sqlite")?.ToLower();
|
||||
switch (dbType)
|
||||
{
|
||||
case "sqlite":
|
||||
dbupResult = new DbupSqliteHelper().Migrate(connString);
|
||||
break;
|
||||
default:
|
||||
var error = $"Database type '{dbType}' is not yet supported by this application";
|
||||
logger.LogWarning(error);
|
||||
throw new NotImplementedException(error);
|
||||
}
|
||||
|
||||
if (dbupResult != null && !dbupResult.Successful)
|
||||
{
|
||||
logger.LogWarning($"DBUP failed with error: '{dbupResult?.Error?.Message}' - '{dbupResult?.Error?.ToString()}'");
|
||||
}
|
||||
|
||||
var adminAccount = new UserModel() { Username = config.GetOrDefault("Settings", "Admin", "Username", "admin"), Password = config.GetOrDefault("Settings", "Admin", "Password", "admin") };
|
||||
database.InitAdminAccount(adminAccount);
|
||||
logger.LogInformation($"Password: {adminAccount.Password}");
|
||||
}
|
||||
else
|
||||
{
|
||||
logger.LogError($"DBUP failed with error: 'Connection string was null or empty'");
|
||||
throw new ArgumentNullException("Please specify a valid database connection string");
|
||||
}
|
||||
}, stoppingToken);
|
||||
}
|
||||
}
|
||||
|
||||
public class DbupSqliteHelper
|
||||
{
|
||||
public DatabaseUpgradeResult Migrate(string connectionString)
|
||||
{
|
||||
try
|
||||
{
|
||||
var dbupBuilder = DeployChanges.To
|
||||
.SQLiteDatabase(connectionString)
|
||||
.WithScriptsEmbeddedInAssembly(Assembly.GetExecutingAssembly())
|
||||
.WithScriptNameComparer(new DbupScriptComparer())
|
||||
.WithFilter(new DbupScriptFilter(DatabaseType.SQLite))
|
||||
.LogToConsole();
|
||||
dbupBuilder.Configure(c => c.Journal = new DbupTableJournal(() => c.ConnectionManager, () => c.Log, "schemaversions"));
|
||||
|
||||
return dbupBuilder.Build().PerformUpgrade();
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
return new DatabaseUpgradeResult(null, false, ex, null);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
31
WeddingShare/Helpers/Dbup/DbupScriptComparer.cs
Normal file
31
WeddingShare/Helpers/Dbup/DbupScriptComparer.cs
Normal file
@@ -0,0 +1,31 @@
|
||||
using System.Diagnostics.CodeAnalysis;
|
||||
|
||||
namespace WeddingShare.Helpers.Dbup
|
||||
{
|
||||
public class DbupScriptComparer : IComparer<string>
|
||||
{
|
||||
public DbupScriptComparer()
|
||||
{
|
||||
}
|
||||
|
||||
public int Compare([AllowNull] string x, [AllowNull] string y)
|
||||
{
|
||||
return GetFileName(x).ToLower().CompareTo(GetFileName(y).ToLower());
|
||||
}
|
||||
|
||||
private string GetFileName(string name)
|
||||
{
|
||||
try
|
||||
{
|
||||
var parts = name.Split('.', StringSplitOptions.RemoveEmptyEntries);
|
||||
if (parts != null && parts.Length >= 2)
|
||||
{
|
||||
return string.Join(".", parts.Skip(parts.Length - 2).Take(2));
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return name;
|
||||
}
|
||||
}
|
||||
}
|
||||
28
WeddingShare/Helpers/Dbup/DbupScriptFilter.cs
Normal file
28
WeddingShare/Helpers/Dbup/DbupScriptFilter.cs
Normal file
@@ -0,0 +1,28 @@
|
||||
using DbUp.Engine;
|
||||
using DbUp.Support;
|
||||
using WeddingShare.Enums;
|
||||
|
||||
namespace WeddingShare.Helpers.Dbup
|
||||
{
|
||||
public class DbupScriptFilter : IScriptFilter
|
||||
{
|
||||
private readonly DatabaseType _dbType;
|
||||
|
||||
public DbupScriptFilter(DatabaseType dbType)
|
||||
{
|
||||
_dbType = dbType;
|
||||
}
|
||||
|
||||
public IEnumerable<SqlScript> Filter(IEnumerable<SqlScript> sorted, HashSet<string> executedScriptNames, ScriptNameComparer comparer)
|
||||
{
|
||||
var scripts = sorted.Where(s => s.SqlScriptOptions.ScriptType == ScriptType.RunAlways || !executedScriptNames.Contains(s.Name, comparer));
|
||||
switch (_dbType)
|
||||
{
|
||||
case DatabaseType.SQLite:
|
||||
return scripts.Where(s => s.Name.ToLower().Contains(".sqlscripts.sqlite."));
|
||||
default:
|
||||
return new List<SqlScript>();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
33
WeddingShare/Helpers/Dbup/DbupTableJournal.cs
Normal file
33
WeddingShare/Helpers/Dbup/DbupTableJournal.cs
Normal file
@@ -0,0 +1,33 @@
|
||||
using System.Data;
|
||||
using DbUp.Engine;
|
||||
using DbUp.Engine.Output;
|
||||
using DbUp.Engine.Transactions;
|
||||
using DbUp.SQLite;
|
||||
|
||||
namespace WeddingShare.Helpers.Dbup
|
||||
{
|
||||
public class DbupTableJournal : SQLiteTableJournal
|
||||
{
|
||||
public DbupTableJournal(Func<IConnectionManager> connectionManager, Func<IUpgradeLog> logger, string table)
|
||||
: base(connectionManager, logger, table)
|
||||
{
|
||||
}
|
||||
|
||||
public override void StoreExecutedScript(SqlScript script, Func<IDbCommand> dbCommandFactory)
|
||||
{
|
||||
var scriptName = script.Name;
|
||||
|
||||
try
|
||||
{
|
||||
var parts = script.Name.Split('.', StringSplitOptions.RemoveEmptyEntries);
|
||||
if (parts != null && parts.Length >= 2)
|
||||
{
|
||||
scriptName = string.Join(".", parts.Skip(parts.Length - 2).Take(2));
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
base.StoreExecutedScript(new SqlScript(scriptName, script.Contents), dbCommandFactory);
|
||||
}
|
||||
}
|
||||
}
|
||||
36
WeddingShare/Helpers/DeviceDetector.cs
Normal file
36
WeddingShare/Helpers/DeviceDetector.cs
Normal file
@@ -0,0 +1,36 @@
|
||||
using System.Text.RegularExpressions;
|
||||
using WeddingShare.Enums;
|
||||
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public interface IDeviceDetector
|
||||
{
|
||||
Task<DeviceType> ParseDeviceType(string userAgent);
|
||||
}
|
||||
|
||||
public class DeviceDetector : IDeviceDetector
|
||||
{
|
||||
public async Task<DeviceType> ParseDeviceType(string userAgent)
|
||||
{
|
||||
return await Task.Run(() =>
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(userAgent))
|
||||
{
|
||||
return DeviceType.Unknown;
|
||||
}
|
||||
|
||||
if (Regex.IsMatch(userAgent, "(tablet|ipad|playbook|silk)|(android(?!.*mobile))", RegexOptions.IgnoreCase))
|
||||
{
|
||||
return DeviceType.Tablet;
|
||||
}
|
||||
|
||||
if (Regex.IsMatch(userAgent, "blackberry|iphone|mobile|windows ce|opera mini|htc|sony|palm|symbianos|ipad|ipod|blackberry|bada|kindle|symbian|sonyericsson|android|samsung|nokia|wap|motor", RegexOptions.IgnoreCase))
|
||||
{
|
||||
return DeviceType.Mobile;
|
||||
}
|
||||
|
||||
return DeviceType.Desktop;
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
15
WeddingShare/Helpers/EnvironmentWrapper.cs
Normal file
15
WeddingShare/Helpers/EnvironmentWrapper.cs
Normal file
@@ -0,0 +1,15 @@
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public interface IEnvironmentWrapper
|
||||
{
|
||||
string? GetEnvironmentVariable(string variable);
|
||||
}
|
||||
|
||||
public class EnvironmentWrapper : IEnvironmentWrapper
|
||||
{
|
||||
public string? GetEnvironmentVariable(string variable)
|
||||
{
|
||||
return Environment.GetEnvironmentVariable(variable);
|
||||
}
|
||||
}
|
||||
}
|
||||
10
WeddingShare/Helpers/ErrorHelper.cs
Normal file
10
WeddingShare/Helpers/ErrorHelper.cs
Normal file
@@ -0,0 +1,10 @@
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public static class ErrorCode
|
||||
{
|
||||
public const int UnexpectedError = 400;
|
||||
public const int Unauthorized = 401;
|
||||
public const int InvalidSecretKey = 402;
|
||||
public const int GalleryCreationNotAllowed = 403;
|
||||
}
|
||||
}
|
||||
106
WeddingShare/Helpers/FileHelper.cs
Normal file
106
WeddingShare/Helpers/FileHelper.cs
Normal file
@@ -0,0 +1,106 @@
|
||||
using WeddingShare.Helpers.Database;
|
||||
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public interface IFileHelper
|
||||
{
|
||||
bool DirectoryExists(string path);
|
||||
bool CreateDirectoryIfNotExists(string path);
|
||||
bool DeleteDirectoryIfExists(string path, bool recursive = true);
|
||||
string[] GetDirectories(string path, string pattern = "*", SearchOption searchOption = SearchOption.AllDirectories);
|
||||
string[] GetFiles(string path, string pattern = "*.*", SearchOption searchOption = SearchOption.AllDirectories);
|
||||
bool FileExists(string path);
|
||||
bool DeleteFileIfExists(string path);
|
||||
bool MoveFileIfExists(string source, string destination);
|
||||
Task<byte[]> ReadAllBytes(string path);
|
||||
Task SaveFile(IFormFile file, string path, FileMode mode);
|
||||
}
|
||||
|
||||
public class FileHelper : IFileHelper
|
||||
{
|
||||
public FileHelper()
|
||||
{
|
||||
}
|
||||
|
||||
public bool DirectoryExists(string path)
|
||||
{
|
||||
return Directory.Exists(path);
|
||||
}
|
||||
|
||||
public bool CreateDirectoryIfNotExists(string path)
|
||||
{
|
||||
if (!DirectoryExists(path))
|
||||
{
|
||||
Directory.CreateDirectory(path);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public bool DeleteDirectoryIfExists(string path, bool recursive = true)
|
||||
{
|
||||
if (DirectoryExists(path))
|
||||
{
|
||||
Directory.Delete(path, recursive);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public string[] GetDirectories(string path, string pattern = "*", SearchOption searchOption = SearchOption.AllDirectories)
|
||||
{
|
||||
return Directory.GetDirectories(path, pattern, searchOption);
|
||||
}
|
||||
|
||||
public string[] GetFiles(string path, string pattern = "*", SearchOption searchOption = SearchOption.AllDirectories)
|
||||
{
|
||||
return Directory.GetFiles(path, pattern, searchOption);
|
||||
}
|
||||
|
||||
public bool FileExists(string path)
|
||||
{
|
||||
return File.Exists(path);
|
||||
}
|
||||
|
||||
public bool DeleteFileIfExists(string path)
|
||||
{
|
||||
if (FileExists(path))
|
||||
{
|
||||
File.Delete(path);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public bool MoveFileIfExists(string source, string destination)
|
||||
{
|
||||
if (FileExists(source))
|
||||
{
|
||||
File.Move(source, destination);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public async Task<byte[]> ReadAllBytes(string path)
|
||||
{
|
||||
return await File.ReadAllBytesAsync(path);
|
||||
}
|
||||
|
||||
public async Task SaveFile(IFormFile file, string path, FileMode mode)
|
||||
{
|
||||
using (var fs = new FileStream(path, mode))
|
||||
{
|
||||
await file.CopyToAsync(fs);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
165
WeddingShare/Helpers/GalleryHelper.cs
Normal file
165
WeddingShare/Helpers/GalleryHelper.cs
Normal file
@@ -0,0 +1,165 @@
|
||||
using Microsoft.EntityFrameworkCore.Metadata.Internal;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using static System.Collections.Specialized.BitVector32;
|
||||
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public interface IGalleryHelper
|
||||
{
|
||||
string GetConfig(string? galleryId, string key, string defaultValue);
|
||||
int GetConfig(string? galleryId, string key, int defaultValue);
|
||||
long GetConfig(string? galleryId, string key, long defaultValue);
|
||||
decimal GetConfig(string? galleryId, string key, decimal defaultValue);
|
||||
double GetConfig(string? galleryId, string key, double defaultValue);
|
||||
bool GetConfig(string? galleryId, string key, bool defaultValue);
|
||||
DateTime? GetConfig(string section, string key, DateTime? defaultValue);
|
||||
Task<string?> GetSecretKey(string galleryId);
|
||||
}
|
||||
|
||||
public class GalleryHelper : IGalleryHelper
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly IDatabaseHelper _database;
|
||||
|
||||
public GalleryHelper(IConfigHelper config, IDatabaseHelper database)
|
||||
{
|
||||
_config = config;
|
||||
_database = database;
|
||||
}
|
||||
|
||||
public string GetConfig(string? galleryId, string key, string defaultValue = null)
|
||||
{
|
||||
string? value = null;
|
||||
|
||||
try
|
||||
{
|
||||
value = _config.Get("Settings", $"{key}_{galleryId ?? "default"}");
|
||||
if (string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
value = _config.Get("Settings", key);
|
||||
if (string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
value = defaultValue;
|
||||
}
|
||||
}
|
||||
}
|
||||
catch
|
||||
{
|
||||
}
|
||||
|
||||
return !string.IsNullOrWhiteSpace(value) ? value : defaultValue;
|
||||
}
|
||||
|
||||
public int GetConfig(string? galleryId, string key, int defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetConfig(galleryId, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToInt32(value);
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
public long GetConfig(string? galleryId, string key, long defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetConfig(galleryId, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToInt64(value);
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
public decimal GetConfig(string? galleryId, string key, decimal defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetConfig(galleryId, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToDecimal(value);
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
public double GetConfig(string? galleryId, string key, double defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetConfig(galleryId, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToDouble(value);
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
public bool GetConfig(string? galleryId, string key, bool defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetConfig(galleryId, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToBoolean(value);
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
public DateTime? GetConfig(string? galleryId, string key, DateTime? defaultValue)
|
||||
{
|
||||
try
|
||||
{
|
||||
var value = this.GetConfig(galleryId, key, string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(value))
|
||||
{
|
||||
return Convert.ToDateTime(value);
|
||||
}
|
||||
}
|
||||
catch { }
|
||||
|
||||
return defaultValue;
|
||||
}
|
||||
|
||||
public async Task<string?> GetSecretKey(string galleryId)
|
||||
{
|
||||
try
|
||||
{
|
||||
var secretKey = _config.Get("Settings", $"Secret_Key_{galleryId}");
|
||||
if (string.IsNullOrWhiteSpace(secretKey))
|
||||
{
|
||||
secretKey = (await _database.GetGallery(galleryId))?.SecretKey;
|
||||
if (string.IsNullOrWhiteSpace(secretKey))
|
||||
{
|
||||
secretKey = _config.Get("Settings", "Secret_Key");
|
||||
}
|
||||
}
|
||||
|
||||
return secretKey;
|
||||
}
|
||||
catch
|
||||
{
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
96
WeddingShare/Helpers/ImageHelper.cs
Normal file
96
WeddingShare/Helpers/ImageHelper.cs
Normal file
@@ -0,0 +1,96 @@
|
||||
using SixLabors.ImageSharp;
|
||||
using SixLabors.ImageSharp.Processing;
|
||||
using WeddingShare.Enums;
|
||||
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public interface IImageHelper
|
||||
{
|
||||
Task<bool> GenerateThumbnail(string imagePath, string savePath, int size = 720);
|
||||
ImageOrientation GetOrientation(Image img);
|
||||
}
|
||||
|
||||
public class ImageHelper : IImageHelper
|
||||
{
|
||||
private readonly IFileHelper _fileHelper;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public ImageHelper(IFileHelper fileHelper, ILogger<ImageHelper> logger)
|
||||
{
|
||||
_fileHelper = fileHelper;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
public async Task<bool> GenerateThumbnail(string imagePath, string savePath, int size = 720)
|
||||
{
|
||||
if (_fileHelper.FileExists(imagePath))
|
||||
{
|
||||
try
|
||||
{
|
||||
var filename = Path.GetFileName(imagePath);
|
||||
using (var img = await Image.LoadAsync(imagePath))
|
||||
{
|
||||
var width = 0;
|
||||
var height = 0;
|
||||
|
||||
var orientation = this.GetOrientation(img);
|
||||
if (orientation == ImageOrientation.Square)
|
||||
{
|
||||
width = size;
|
||||
height = size;
|
||||
}
|
||||
else if (orientation == ImageOrientation.Landscape)
|
||||
{
|
||||
var scale = (decimal)size / (decimal)img.Width;
|
||||
width = (int)((decimal)img.Width * scale);
|
||||
height = (int)((decimal)img.Height * scale);
|
||||
}
|
||||
else if (orientation == ImageOrientation.Portrait)
|
||||
{
|
||||
var scale = (decimal)size / (decimal)img.Height;
|
||||
width = (int)((decimal)img.Width * scale);
|
||||
height = (int)((decimal)img.Height * scale);
|
||||
}
|
||||
|
||||
img.Mutate(x =>
|
||||
{
|
||||
x.Resize(width, height);
|
||||
x.AutoOrient();
|
||||
});
|
||||
|
||||
await img.SaveAsWebpAsync(savePath);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to generate thumbnail - '{imagePath}'");
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public ImageOrientation GetOrientation(Image img)
|
||||
{
|
||||
if (img != null)
|
||||
{
|
||||
if (img.Width > img.Height)
|
||||
{
|
||||
return ImageOrientation.Landscape;
|
||||
}
|
||||
else if (img.Width < img.Height)
|
||||
{
|
||||
return ImageOrientation.Portrait;
|
||||
}
|
||||
else if (img.Width == img.Height)
|
||||
{
|
||||
return ImageOrientation.Square;
|
||||
}
|
||||
}
|
||||
|
||||
return ImageOrientation.None;
|
||||
}
|
||||
}
|
||||
}
|
||||
103
WeddingShare/Helpers/Notifications/EmailHelper.cs
Normal file
103
WeddingShare/Helpers/Notifications/EmailHelper.cs
Normal file
@@ -0,0 +1,103 @@
|
||||
using System.Net;
|
||||
using System.Net.Mail;
|
||||
using System.Text;
|
||||
|
||||
namespace WeddingShare.Helpers.Notifications
|
||||
{
|
||||
public class EmailHelper : INotificationHelper
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly ISmtpClientWrapper _client;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public EmailHelper(IConfigHelper config, ISmtpClientWrapper client, ILogger<EmailHelper> logger)
|
||||
{
|
||||
_config = config;
|
||||
_client = client;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
public async Task<bool> Send(string title, string message, string? actionLink = null)
|
||||
{
|
||||
if (_config.GetOrDefault("Notifications", "Smtp", "Enabled", false))
|
||||
{
|
||||
try
|
||||
{
|
||||
var recipients = _config.GetOrDefault("Notifications", "Smtp", "Recipient", string.Empty)?.Split(new char[] { ';', ',' }, StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries)?.Select(x => new MailAddress(x));
|
||||
if (recipients != null && recipients.Any())
|
||||
{
|
||||
var host = _config.GetOrDefault("Notifications", "Smtp", "Host", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(host))
|
||||
{
|
||||
var port = _config.GetOrDefault("Notifications", "Smtp", "Port", 587);
|
||||
if (port > 0)
|
||||
{
|
||||
var from = _config.GetOrDefault("Notifications", "Smtp", "From", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(from))
|
||||
{
|
||||
var sentToAll = true;
|
||||
using (var smtp = new SmtpClient(host, port))
|
||||
{
|
||||
var username = _config.GetOrDefault("Notifications", "Smtp", "Username", string.Empty);
|
||||
var password = _config.GetOrDefault("Notifications", "Smtp", "Password", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(username) && !string.IsNullOrWhiteSpace(password))
|
||||
{
|
||||
smtp.UseDefaultCredentials = false;
|
||||
smtp.Credentials = new NetworkCredential(username, password);
|
||||
}
|
||||
|
||||
smtp.EnableSsl = _config.GetOrDefault("Notifications", "Smtp", "Use_SSL", false);
|
||||
|
||||
var sender = new MailAddress(from, _config.GetOrDefault("Notifications", "Smtp", "DisplayName", "WeddingShare"));
|
||||
foreach (var to in recipients)
|
||||
{
|
||||
try
|
||||
{
|
||||
await _client.SendMailAsync(smtp, new MailMessage(new MailAddress(from, _config.GetOrDefault("Notifications", "Smtp", "DisplayName", "WeddingShare")), to)
|
||||
{
|
||||
Sender = sender,
|
||||
Subject = title,
|
||||
Body = !string.IsNullOrWhiteSpace(actionLink) ? $"{message}<br/><br/>Visit - {actionLink}" : message,
|
||||
IsBodyHtml = true,
|
||||
});
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogWarning(ex, $"Failed to send email to '{to}' - {ex.Message}");
|
||||
sentToAll = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return sentToAll;
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid SMTP sender specified");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid SMTP port specified");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid SMTP host specified");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid SMTP recipient specified");
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"Failed to send email with title '{title}' - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
75
WeddingShare/Helpers/Notifications/GotifyHelper.cs
Normal file
75
WeddingShare/Helpers/Notifications/GotifyHelper.cs
Normal file
@@ -0,0 +1,75 @@
|
||||
namespace WeddingShare.Helpers.Notifications
|
||||
{
|
||||
public class GotifyHelper : INotificationHelper
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly IHttpClientFactory _clientFactory;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public GotifyHelper(IConfigHelper config, IHttpClientFactory clientFactory, ILogger<GotifyHelper> logger)
|
||||
{
|
||||
_config = config;
|
||||
_clientFactory = clientFactory;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
public async Task<bool> Send(string title, string message, string? actionLink = null)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(_config.GetOrDefault("Notifications", "Gotify", "Endpoint", string.Empty)))
|
||||
{
|
||||
_logger.LogWarning($"Invalid Gotify endpoint specified");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (string.IsNullOrWhiteSpace(_config.GetOrDefault("Notifications", "Gotify", "Token", string.Empty)))
|
||||
{
|
||||
_logger.LogWarning($"Invalid Gotify token specified");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Gotify", "Enabled", false))
|
||||
{
|
||||
try
|
||||
{
|
||||
var token = _config.GetOrDefault("Notifications", "Gotify", "Token", string.Empty);
|
||||
if (!string.IsNullOrWhiteSpace(token))
|
||||
{
|
||||
var priority = _config.GetOrDefault("Notifications", "Gotify", "Priority", 4);
|
||||
if (priority > 0)
|
||||
{
|
||||
message = !string.IsNullOrWhiteSpace(actionLink) ? $"{message} - Visit - {actionLink}" : message;
|
||||
|
||||
var client = _clientFactory.CreateClient("GotifyClient");
|
||||
using (var response = await client.PostAsJsonAsync($"/message?token={token}", new { title, message, priority }))
|
||||
{
|
||||
if (response.IsSuccessStatusCode)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
else
|
||||
{
|
||||
var error = await response.Content.ReadAsStringAsync();
|
||||
_logger.LogError($"Failed to send Gotify message with title '{title}' - {error}");
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid Gotify priority specified");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid Gotify token specified");
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"Failed to send Gotify message with title '{title}' - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,7 @@
|
||||
namespace WeddingShare.Helpers.Notifications
|
||||
{
|
||||
public interface INotificationHelper
|
||||
{
|
||||
Task<bool> Send(string title, string message, string? actionLink = null);
|
||||
}
|
||||
}
|
||||
42
WeddingShare/Helpers/Notifications/NotificationBroker.cs
Normal file
42
WeddingShare/Helpers/Notifications/NotificationBroker.cs
Normal file
@@ -0,0 +1,42 @@
|
||||
namespace WeddingShare.Helpers.Notifications
|
||||
{
|
||||
public class NotificationBroker : INotificationHelper
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly ISmtpClientWrapper _smtp;
|
||||
private readonly IHttpClientFactory _clientFactory;
|
||||
private readonly ILoggerFactory _logger;
|
||||
|
||||
public NotificationBroker(IConfigHelper config, ISmtpClientWrapper smtp, IHttpClientFactory clientFactory, ILoggerFactory logger)
|
||||
{
|
||||
_config = config;
|
||||
_smtp = smtp;
|
||||
_clientFactory = clientFactory;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
public async Task<bool> Send(string title, string message, string? actionLink = null)
|
||||
{
|
||||
var emailSent = true;
|
||||
var ntfySent = true;
|
||||
var gotifySent = true;
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Smtp", "Enabled", false))
|
||||
{
|
||||
emailSent = await new EmailHelper(_config, _smtp, _logger.CreateLogger<EmailHelper>()).Send(title, message, actionLink);
|
||||
}
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Ntfy", "Enabled", false))
|
||||
{
|
||||
ntfySent = await new NtfyHelper(_config, _clientFactory, _logger.CreateLogger<NtfyHelper>()).Send(title, message, actionLink);
|
||||
}
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Gotify", "Enabled", false))
|
||||
{
|
||||
gotifySent = await new GotifyHelper(_config, _clientFactory, _logger.CreateLogger<GotifyHelper>()).Send(title, message, actionLink);
|
||||
}
|
||||
|
||||
return emailSent && ntfySent && gotifySent;
|
||||
}
|
||||
}
|
||||
}
|
||||
77
WeddingShare/Helpers/Notifications/NtfyHelper.cs
Normal file
77
WeddingShare/Helpers/Notifications/NtfyHelper.cs
Normal file
@@ -0,0 +1,77 @@
|
||||
namespace WeddingShare.Helpers.Notifications
|
||||
{
|
||||
public class NtfyHelper : INotificationHelper
|
||||
{
|
||||
private readonly IConfigHelper _config;
|
||||
private readonly IHttpClientFactory _clientFactory;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public NtfyHelper(IConfigHelper config, IHttpClientFactory clientFactory, ILogger<NtfyHelper> logger)
|
||||
{
|
||||
_config = config;
|
||||
_clientFactory = clientFactory;
|
||||
_logger = logger;
|
||||
}
|
||||
|
||||
public async Task<bool> Send(string title, string message, string? actionLink = null)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(_config.GetOrDefault("Notifications", "Ntfy", "Endpoint", string.Empty)))
|
||||
{
|
||||
_logger.LogWarning($"Invalid Ntfy endpoint specified");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (string.IsNullOrWhiteSpace(_config.GetOrDefault("Notifications", "Ntfy", "Token", string.Empty)))
|
||||
{
|
||||
_logger.LogWarning($"Invalid Ntfy token specified");
|
||||
return false;
|
||||
}
|
||||
|
||||
if (_config.GetOrDefault("Notifications", "Ntfy", "Enabled", false))
|
||||
{
|
||||
try
|
||||
{
|
||||
var topic = _config.GetOrDefault("Notifications", "Ntfy", "Topic", "WeddingShare");
|
||||
if (!string.IsNullOrWhiteSpace(topic))
|
||||
{
|
||||
var priority = _config.GetOrDefault("Notifications", "Ntfy", "Priority", 4);
|
||||
if (priority > 0)
|
||||
{
|
||||
var defaultIcon = "https://github.com/Cirx08/WeddingShare/blob/main/WeddingShare/wwwroot/images/logo.png?raw=true";
|
||||
var icon = _config.GetOrDefault("Settings", "Logo", defaultIcon);
|
||||
icon = !icon.StartsWith('.') && !icon.StartsWith('/') ? icon : defaultIcon;
|
||||
|
||||
var client = _clientFactory.CreateClient("NtfyClient");
|
||||
using (var response = await client.PostAsJsonAsync("/", new { icon, topic, title, message, priority, click = actionLink }))
|
||||
{
|
||||
if (response.IsSuccessStatusCode)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
else
|
||||
{
|
||||
var error = await response.Content.ReadAsStringAsync();
|
||||
_logger.LogError($"Failed to send Ntfy message with title '{title}' - {error}");
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid Ntfy priority specified");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.LogWarning($"Invalid Ntfy topic specified");
|
||||
}
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_logger.LogError(ex, $"Failed to send Ntfy message with title '{title}' - {ex?.Message}");
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
17
WeddingShare/Helpers/SmtpClientWrapper.cs
Normal file
17
WeddingShare/Helpers/SmtpClientWrapper.cs
Normal file
@@ -0,0 +1,17 @@
|
||||
using System.Net.Mail;
|
||||
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public interface ISmtpClientWrapper
|
||||
{
|
||||
Task SendMailAsync(SmtpClient client, MailMessage message);
|
||||
}
|
||||
|
||||
public class SmtpClientWrapper : ISmtpClientWrapper
|
||||
{
|
||||
public async Task SendMailAsync(SmtpClient client, MailMessage message)
|
||||
{
|
||||
await client.SendMailAsync(message);
|
||||
}
|
||||
}
|
||||
}
|
||||
25
WeddingShare/Helpers/UrlHelper.cs
Normal file
25
WeddingShare/Helpers/UrlHelper.cs
Normal file
@@ -0,0 +1,25 @@
|
||||
using System.Text.RegularExpressions;
|
||||
|
||||
namespace WeddingShare.Helpers
|
||||
{
|
||||
public class UrlHelper
|
||||
{
|
||||
public static string Generate(HttpContext? ctx, IConfigHelper config, string? append)
|
||||
{
|
||||
return Generate(ctx?.Request, config, append);
|
||||
}
|
||||
|
||||
public static string Generate(HttpRequest? ctx, IConfigHelper config, string? append)
|
||||
{
|
||||
if (ctx != null)
|
||||
{
|
||||
var scheme = config.GetOrDefault("Settings", "Force_Https", false) ? "https" : ctx.Scheme;
|
||||
var host = Regex.Replace(config.GetOrDefault("Settings", "Base_Url", ctx.Host.Value), "http[s]*\\:\\/\\/", string.Empty).TrimEnd('/');
|
||||
|
||||
return $"{scheme}://{host}/{append?.TrimStart('/')}";
|
||||
}
|
||||
|
||||
return string.Empty;
|
||||
}
|
||||
}
|
||||
}
|
||||
27
WeddingShare/Models/Database/GalleryItemModel.cs
Normal file
27
WeddingShare/Models/Database/GalleryItemModel.cs
Normal file
@@ -0,0 +1,27 @@
|
||||
using WeddingShare.Enums;
|
||||
|
||||
namespace WeddingShare.Models.Database
|
||||
{
|
||||
public class GalleryItemModel
|
||||
{
|
||||
public GalleryItemModel()
|
||||
: this(0, 0, string.Empty, null, GalleryItemState.Pending)
|
||||
{
|
||||
}
|
||||
|
||||
public GalleryItemModel(int id, int galleryId, string title, string? uploadedBy, GalleryItemState state)
|
||||
{
|
||||
Id = id;
|
||||
GalleryId = galleryId;
|
||||
Title = title;
|
||||
UploadedBy = uploadedBy;
|
||||
State = state;
|
||||
}
|
||||
|
||||
public int Id { get; set; }
|
||||
public int GalleryId { get; set; }
|
||||
public string Title { get; set; }
|
||||
public string? UploadedBy { get; set; }
|
||||
public GalleryItemState State { get; set; }
|
||||
}
|
||||
}
|
||||
12
WeddingShare/Models/Database/GalleryModel.cs
Normal file
12
WeddingShare/Models/Database/GalleryModel.cs
Normal file
@@ -0,0 +1,12 @@
|
||||
namespace WeddingShare.Models.Database
|
||||
{
|
||||
public class GalleryModel
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public string Name { get; set; }
|
||||
public string? SecretKey { get; set; }
|
||||
public int TotalItems { get; set; }
|
||||
public int ApprovedItems { get; set; }
|
||||
public int PendingItems { get; set; }
|
||||
}
|
||||
}
|
||||
7
WeddingShare/Models/Database/PendingGalleryItemModel.cs
Normal file
7
WeddingShare/Models/Database/PendingGalleryItemModel.cs
Normal file
@@ -0,0 +1,7 @@
|
||||
namespace WeddingShare.Models.Database
|
||||
{
|
||||
public class PendingGalleryItemModel : GalleryItemModel
|
||||
{
|
||||
public string GalleryName { get; set; }
|
||||
}
|
||||
}
|
||||
20
WeddingShare/Models/Database/UserModel.cs
Normal file
20
WeddingShare/Models/Database/UserModel.cs
Normal file
@@ -0,0 +1,20 @@
|
||||
namespace WeddingShare.Models.Database
|
||||
{
|
||||
public class UserModel
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public string Username { get; set; }
|
||||
public string? Email { get; set; }
|
||||
public string? Password { get; set; }
|
||||
public int FailedLogins { get; set; }
|
||||
public DateTime? LockoutUntil { get; set; }
|
||||
|
||||
public bool IsLockedOut
|
||||
{
|
||||
get
|
||||
{
|
||||
return this.LockoutUntil != null && this.LockoutUntil >= DateTime.UtcNow;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -4,6 +4,6 @@ namespace WeddingShare.Models
|
||||
{
|
||||
public string? RequestId { get; set; }
|
||||
|
||||
public bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
|
||||
public bool ShowRequestId => !string.IsNullOrWhiteSpace(RequestId);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2,13 +2,15 @@
|
||||
{
|
||||
public class FileUploader
|
||||
{
|
||||
public FileUploader(string id, string url)
|
||||
public FileUploader(string id, string? key, string url)
|
||||
{
|
||||
this.GalleryId = id;
|
||||
this.SecretKey = key;
|
||||
this.UploadUrl = url;
|
||||
}
|
||||
|
||||
public string? GalleryId { get; set; }
|
||||
public string? SecretKey { get; set; }
|
||||
public string? UploadUrl { get; set; }
|
||||
}
|
||||
}
|
||||
@@ -4,8 +4,11 @@
|
||||
{
|
||||
public LoginModel()
|
||||
{
|
||||
this.Username = string.Empty;
|
||||
this.Password = string.Empty;
|
||||
}
|
||||
|
||||
public string? Password { get; set; }
|
||||
public string Username { get; set; }
|
||||
public string Password { get; set; }
|
||||
}
|
||||
}
|
||||
@@ -1,30 +1,34 @@
|
||||
namespace WeddingShare.Models
|
||||
using WeddingShare.Enums;
|
||||
|
||||
namespace WeddingShare.Models
|
||||
{
|
||||
public class PhotoGallery
|
||||
{
|
||||
public PhotoGallery()
|
||||
: this(3)
|
||||
: this(ViewMode.Default)
|
||||
{
|
||||
}
|
||||
|
||||
public PhotoGallery(int columnCount)
|
||||
: this("default", columnCount, string.Empty, new List<string>())
|
||||
public PhotoGallery(ViewMode viewMode)
|
||||
: this("default", string.Empty, string.Empty, string.Empty, viewMode, new List<PhotoGalleryImage>())
|
||||
{
|
||||
}
|
||||
|
||||
public PhotoGallery(string id, int columnCount, string path, List<string> images)
|
||||
public PhotoGallery(string id, string secretKey, string galleryPath, string thumbnailPath, ViewMode viewMode, List<PhotoGalleryImage> images)
|
||||
{
|
||||
this.GalleryId = id;
|
||||
this.GalleryPath = path;
|
||||
this.ColumnCount = columnCount;
|
||||
this.GalleryPath = galleryPath;
|
||||
this.ThumbnailsPath = thumbnailPath;
|
||||
this.ViewMode = viewMode;
|
||||
this.PendingCount = 0;
|
||||
this.Images = images;
|
||||
this.FileUploader = new FileUploader(id, "/Gallery/UploadImage");
|
||||
this.FileUploader = new FileUploader(id, secretKey, "/Gallery/UploadImage");
|
||||
}
|
||||
|
||||
public string? GalleryId { get; set; }
|
||||
public string? GalleryPath { get; set; }
|
||||
public int ColumnCount { get; set; }
|
||||
public string? ThumbnailsPath { get; set; }
|
||||
public ViewMode ViewMode { get; set; }
|
||||
public int PendingCount { get; set; }
|
||||
public int ApprovedCount
|
||||
{
|
||||
@@ -40,7 +44,19 @@
|
||||
return this.ApprovedCount + this.PendingCount;
|
||||
}
|
||||
}
|
||||
public List<string>? Images { get; set; }
|
||||
public List<PhotoGalleryImage>? Images { get; set; }
|
||||
public FileUploader? FileUploader { get; set; }
|
||||
}
|
||||
|
||||
public class PhotoGalleryImage
|
||||
{
|
||||
public PhotoGalleryImage()
|
||||
{
|
||||
}
|
||||
|
||||
public int Id { get; set; }
|
||||
public string? Path { get; set; }
|
||||
public string? Name { get; set; }
|
||||
public string? UploadedBy { get; set; }
|
||||
}
|
||||
}
|
||||
@@ -7,7 +7,7 @@
|
||||
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||
},
|
||||
"dotnetRunMessages": true,
|
||||
"applicationUrl": "http://localhost:5051"
|
||||
"applicationUrl": "http://localhost:5000"
|
||||
},
|
||||
"Docker": {
|
||||
"commandName": "Docker",
|
||||
|
||||
@@ -117,15 +117,45 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Failed_Add_Gallery" xml:space="preserve">
|
||||
<value>Failed to create gallery</value>
|
||||
</data>
|
||||
<data name="Failed_Delete_Gallery" xml:space="preserve">
|
||||
<value>Failed to delete gallery</value>
|
||||
</data>
|
||||
<data name="Failed_Download_Gallery" xml:space="preserve">
|
||||
<value>Failed to download gallery</value>
|
||||
</data>
|
||||
<data name="Failed_Edit_Gallery" xml:space="preserve">
|
||||
<value>Failed to update gallery</value>
|
||||
</data>
|
||||
<data name="Failed_Export" xml:space="preserve">
|
||||
<value>Failed to export data</value>
|
||||
</data>
|
||||
<data name="Failed_Finding_File" xml:space="preserve">
|
||||
<value>Failed to find file</value>
|
||||
</data>
|
||||
<data name="Failed_Import" xml:space="preserve">
|
||||
<value>Failed to import data</value>
|
||||
</data>
|
||||
<data name="Failed_Reviewing_Photo" xml:space="preserve">
|
||||
<value>Failed to review photo</value>
|
||||
</data>
|
||||
<data name="Failed_Wipe_Galleries" xml:space="preserve">
|
||||
<value>Failed to wipe galleries</value>
|
||||
</data>
|
||||
<data name="Failed_Wipe_Gallery" xml:space="preserve">
|
||||
<value>Failed to wipe gallery</value>
|
||||
</data>
|
||||
<data name="Gallery_Name_Already_Exists" xml:space="preserve">
|
||||
<value>Gallery name already exists</value>
|
||||
</data>
|
||||
<data name="Login_Failed" xml:space="preserve">
|
||||
<value>Faied to log user in</value>
|
||||
</data>
|
||||
<data name="Name_Cannot_Be_Blank" xml:space="preserve">
|
||||
<value>Gallery name cannot be empty</value>
|
||||
</data>
|
||||
<data name="Pending_Uploads_Failed" xml:space="preserve">
|
||||
<value>Failed to get pending uploads</value>
|
||||
</data>
|
||||
|
||||
@@ -117,15 +117,45 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Failed_Add_Gallery" xml:space="preserve">
|
||||
<value>Impossible de créer la galerie</value>
|
||||
</data>
|
||||
<data name="Failed_Delete_Gallery" xml:space="preserve">
|
||||
<value>Impossible de supprimer la galerie</value>
|
||||
</data>
|
||||
<data name="Failed_Download_Gallery" xml:space="preserve">
|
||||
<value>Impossible de télécharger la galerie</value>
|
||||
</data>
|
||||
<data name="Failed_Edit_Gallery" xml:space="preserve">
|
||||
<value>Échec de la mise à jour de la galerie</value>
|
||||
</data>
|
||||
<data name="Failed_Export" xml:space="preserve">
|
||||
<value>Échec de l'exportation des données</value>
|
||||
</data>
|
||||
<data name="Failed_Finding_File" xml:space="preserve">
|
||||
<value>Impossible de trouver le fichier</value>
|
||||
</data>
|
||||
<data name="Failed_Import" xml:space="preserve">
|
||||
<value>Échec de l'importation des données</value>
|
||||
</data>
|
||||
<data name="Failed_Reviewing_Photo" xml:space="preserve">
|
||||
<value>Impossible de vérifier la photo</value>
|
||||
</data>
|
||||
<data name="Failed_Wipe_Galleries" xml:space="preserve">
|
||||
<value>Impossible d'effacer la galerie</value>
|
||||
</data>
|
||||
<data name="Failed_Wipe_Gallery" xml:space="preserve">
|
||||
<value>Impossible d'effacer la galerie</value>
|
||||
</data>
|
||||
<data name="Gallery_Name_Already_Exists" xml:space="preserve">
|
||||
<value>Le nom de la galerie existe déjà</value>
|
||||
</data>
|
||||
<data name="Login_Failed" xml:space="preserve">
|
||||
<value>Impossible de connecter l'utilisateur</value>
|
||||
</data>
|
||||
<data name="Name_Cannot_Be_Blank" xml:space="preserve">
|
||||
<value>Le nom de la galerie ne peut pas être vide</value>
|
||||
</data>
|
||||
<data name="Pending_Uploads_Failed" xml:space="preserve">
|
||||
<value>Impossible d'obtenir les téléchargements en attente</value>
|
||||
</data>
|
||||
|
||||
@@ -120,6 +120,9 @@
|
||||
<data name="File_Upload_Failed" xml:space="preserve">
|
||||
<value>Failed to upload file</value>
|
||||
</data>
|
||||
<data name="Gallery_Does_Not_Exist" xml:space="preserve">
|
||||
<value>The requested gallery was not found or does not exist</value>
|
||||
</data>
|
||||
<data name="Image_Upload_Failed" xml:space="preserve">
|
||||
<value>Failed to upload images</value>
|
||||
</data>
|
||||
@@ -135,12 +138,15 @@
|
||||
<data name="Invalid_Gallery_Key" xml:space="preserve">
|
||||
<value>Invalid gallery key</value>
|
||||
</data>
|
||||
<data name="Invalid_Security_Key_Warning" xml:space="preserve">
|
||||
<value>A request was made using an invalid security key</value>
|
||||
<data name="Invalid_Secret_Key_Warning" xml:space="preserve">
|
||||
<value>A request was made using an invalid secret key</value>
|
||||
</data>
|
||||
<data name="Max_File_Size" xml:space="preserve">
|
||||
<value>Max file size is</value>
|
||||
</data>
|
||||
<data name="No_Files_For_Upload" xml:space="preserve">
|
||||
<value>No files were detected for upload</value>
|
||||
</data>
|
||||
<data name="Save_To_Gallery_Failed" xml:space="preserve">
|
||||
<value>Failed to save image to gallery</value>
|
||||
</data>
|
||||
|
||||
@@ -120,6 +120,9 @@
|
||||
<data name="File_Upload_Failed" xml:space="preserve">
|
||||
<value>Impossible de télécharger le fichier</value>
|
||||
</data>
|
||||
<data name="Gallery_Does_Not_Exist" xml:space="preserve">
|
||||
<value>La galerie demandée n'a pas été trouvée ou n'existe pas</value>
|
||||
</data>
|
||||
<data name="Image_Upload_Failed" xml:space="preserve">
|
||||
<value>Impossible de télécharger les images</value>
|
||||
</data>
|
||||
@@ -135,12 +138,15 @@
|
||||
<data name="Invalid_Gallery_Key" xml:space="preserve">
|
||||
<value>Clé de galerie non valide</value>
|
||||
</data>
|
||||
<data name="Invalid_Security_Key_Warning" xml:space="preserve">
|
||||
<data name="Invalid_Secret_Key_Warning" xml:space="preserve">
|
||||
<value>Une demande a été effectuée à l'aide d'une clé de sécurité non valide</value>
|
||||
</data>
|
||||
<data name="Max_File_Size" xml:space="preserve">
|
||||
<value>La taille maximale du fichier est</value>
|
||||
</data>
|
||||
<data name="No_Files_For_Upload" xml:space="preserve">
|
||||
<value>Aucun fichier n'a été détecté pour le téléchargement</value>
|
||||
</data>
|
||||
<data name="Save_To_Gallery_Failed" xml:space="preserve">
|
||||
<value>Impossible d'enregistrer l'image dans la galerie</value>
|
||||
</data>
|
||||
|
||||
@@ -117,15 +117,33 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Actions" xml:space="preserve">
|
||||
<value>Actions</value>
|
||||
</data>
|
||||
<data name="Approve" xml:space="preserve">
|
||||
<value>Approve</value>
|
||||
</data>
|
||||
<data name="Approved" xml:space="preserve">
|
||||
<value>Approved</value>
|
||||
</data>
|
||||
<data name="Available_Galleries" xml:space="preserve">
|
||||
<value>Available Galleries</value>
|
||||
</data>
|
||||
<data name="Create" xml:space="preserve">
|
||||
<value>Create</value>
|
||||
</data>
|
||||
<data name="Delete" xml:space="preserve">
|
||||
<value>Delete</value>
|
||||
</data>
|
||||
<data name="Key" xml:space="preserve">
|
||||
<value>Key</value>
|
||||
</data>
|
||||
<data name="Link" xml:space="preserve">
|
||||
<value>Link</value>
|
||||
</data>
|
||||
<data name="Login" xml:space="preserve">
|
||||
<value>Login</value>
|
||||
</data>
|
||||
<data name="Name" xml:space="preserve">
|
||||
<value>Name</value>
|
||||
</data>
|
||||
@@ -135,10 +153,19 @@
|
||||
<data name="No_Pending_Uploads" xml:space="preserve">
|
||||
<value>No uploads pending review</value>
|
||||
</data>
|
||||
<data name="Pending" xml:space="preserve">
|
||||
<value>Pending</value>
|
||||
</data>
|
||||
<data name="Pending_Uploads" xml:space="preserve">
|
||||
<value>Pending Uploads</value>
|
||||
</data>
|
||||
<data name="Reject" xml:space="preserve">
|
||||
<value>Reject</value>
|
||||
</data>
|
||||
<data name="Rename" xml:space="preserve">
|
||||
<value>Rename</value>
|
||||
</data>
|
||||
<data name="Total" xml:space="preserve">
|
||||
<value>Total</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,15 +117,33 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Actions" xml:space="preserve">
|
||||
<value>Actes</value>
|
||||
</data>
|
||||
<data name="Approve" xml:space="preserve">
|
||||
<value>Approuver</value>
|
||||
</data>
|
||||
<data name="Approved" xml:space="preserve">
|
||||
<value>Approuvée</value>
|
||||
</data>
|
||||
<data name="Available_Galleries" xml:space="preserve">
|
||||
<value>Galeries disponibles</value>
|
||||
</data>
|
||||
<data name="Create" xml:space="preserve">
|
||||
<value>Créer</value>
|
||||
</data>
|
||||
<data name="Delete" xml:space="preserve">
|
||||
<value>Supprimer</value>
|
||||
</data>
|
||||
<data name="Key" xml:space="preserve">
|
||||
<value>Clé</value>
|
||||
</data>
|
||||
<data name="Link" xml:space="preserve">
|
||||
<value>Lien</value>
|
||||
</data>
|
||||
<data name="Login" xml:space="preserve">
|
||||
<value>Se connecter</value>
|
||||
</data>
|
||||
<data name="Name" xml:space="preserve">
|
||||
<value>Nom</value>
|
||||
</data>
|
||||
@@ -135,10 +153,19 @@
|
||||
<data name="No_Pending_Uploads" xml:space="preserve">
|
||||
<value>Aucun téléchargement en attente de révision</value>
|
||||
</data>
|
||||
<data name="Pending" xml:space="preserve">
|
||||
<value>En attente</value>
|
||||
</data>
|
||||
<data name="Pending_Uploads" xml:space="preserve">
|
||||
<value>Téléchargements en attente</value>
|
||||
</data>
|
||||
<data name="Reject" xml:space="preserve">
|
||||
<value>Rejeter</value>
|
||||
</data>
|
||||
<data name="Rename" xml:space="preserve">
|
||||
<value>Rebaptiser</value>
|
||||
</data>
|
||||
<data name="Total" xml:space="preserve">
|
||||
<value>Totale</value>
|
||||
</data>
|
||||
</root>
|
||||
189
WeddingShare/Resources/Views/Admin/LoginViewModel.en-GB.resx
Normal file
189
WeddingShare/Resources/Views/Admin/LoginViewModel.en-GB.resx
Normal file
@@ -0,0 +1,189 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<root>
|
||||
<!--
|
||||
Microsoft ResX Schema
|
||||
|
||||
Version 2.0
|
||||
|
||||
The primary goals of this format is to allow a simple XML format
|
||||
that is mostly human readable. The generation and parsing of the
|
||||
various data types are done through the TypeConverter classes
|
||||
associated with the data types.
|
||||
|
||||
Example:
|
||||
|
||||
... ado.net/XML headers & schema ...
|
||||
<resheader name="resmimetype">text/microsoft-resx</resheader>
|
||||
<resheader name="version">2.0</resheader>
|
||||
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
|
||||
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
|
||||
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
|
||||
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
|
||||
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
|
||||
<value>[base64 mime encoded serialized .NET Framework object]</value>
|
||||
</data>
|
||||
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
|
||||
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
|
||||
<comment>This is a comment</comment>
|
||||
</data>
|
||||
|
||||
There are any number of "resheader" rows that contain simple
|
||||
name/value pairs.
|
||||
|
||||
Each data row contains a name, and value. The row also contains a
|
||||
type or mimetype. Type corresponds to a .NET class that support
|
||||
text/value conversion through the TypeConverter architecture.
|
||||
Classes that don't support this are serialized and stored with the
|
||||
mimetype set.
|
||||
|
||||
The mimetype is used for serialized objects, and tells the
|
||||
ResXResourceReader how to depersist the object. This is currently not
|
||||
extensible. For a given mimetype the value must be set accordingly:
|
||||
|
||||
Note - application/x-microsoft.net.object.binary.base64 is the format
|
||||
that the ResXResourceWriter will generate, however the reader can
|
||||
read any of the formats listed below.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.binary.base64
|
||||
value : The object must be serialized with
|
||||
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
|
||||
: and then encoded with base64 encoding.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.soap.base64
|
||||
value : The object must be serialized with
|
||||
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter
|
||||
: and then encoded with base64 encoding.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.bytearray.base64
|
||||
value : The object must be serialized into a byte array
|
||||
: using a System.ComponentModel.TypeConverter
|
||||
: and then encoded with base64 encoding.
|
||||
-->
|
||||
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
|
||||
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
|
||||
<xsd:element name="root" msdata:IsDataSet="true">
|
||||
<xsd:complexType>
|
||||
<xsd:choice maxOccurs="unbounded">
|
||||
<xsd:element name="metadata">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" use="required" type="xsd:string" />
|
||||
<xsd:attribute name="type" type="xsd:string" />
|
||||
<xsd:attribute name="mimetype" type="xsd:string" />
|
||||
<xsd:attribute ref="xml:space" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="assembly">
|
||||
<xsd:complexType>
|
||||
<xsd:attribute name="alias" type="xsd:string" />
|
||||
<xsd:attribute name="name" type="xsd:string" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="data">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
||||
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
|
||||
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
|
||||
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
|
||||
<xsd:attribute ref="xml:space" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="resheader">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" use="required" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
</xsd:choice>
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
</xsd:schema>
|
||||
<resheader name="resmimetype">
|
||||
<value>text/microsoft-resx</value>
|
||||
</resheader>
|
||||
<resheader name="version">
|
||||
<value>2.0</value>
|
||||
</resheader>
|
||||
<resheader name="reader">
|
||||
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Actions" xml:space="preserve">
|
||||
<value>Actions</value>
|
||||
</data>
|
||||
<data name="Username" xml:space="preserve">
|
||||
<value>Username</value>
|
||||
</data>
|
||||
<data name="Login" xml:space="preserve">
|
||||
<value>Login</value>
|
||||
</data>
|
||||
<data name="Login_Username_Help" xml:space="preserve">
|
||||
<value>Please enter account username.</value>
|
||||
</data>
|
||||
<data name="Login_Username_Placeholder" xml:space="preserve">
|
||||
<value>admin</value>
|
||||
</data>
|
||||
<data name="Login_Password_Help" xml:space="preserve">
|
||||
<value>Please enter account password.</value>
|
||||
</data>
|
||||
<data name="Login_Password_Placeholder" xml:space="preserve">
|
||||
<value>Password</value>
|
||||
</data>
|
||||
<data name="Approve" xml:space="preserve">
|
||||
<value>Approve</value>
|
||||
</data>
|
||||
<data name="Approved" xml:space="preserve">
|
||||
<value>Approved</value>
|
||||
</data>
|
||||
<data name="Available_Galleries" xml:space="preserve">
|
||||
<value>Available Galleries</value>
|
||||
</data>
|
||||
<data name="Create" xml:space="preserve">
|
||||
<value>Create</value>
|
||||
</data>
|
||||
<data name="Delete" xml:space="preserve">
|
||||
<value>Delete</value>
|
||||
</data>
|
||||
<data name="Key" xml:space="preserve">
|
||||
<value>Key</value>
|
||||
</data>
|
||||
<data name="Link" xml:space="preserve">
|
||||
<value>Link</value>
|
||||
</data>
|
||||
<data name="Login" xml:space="preserve">
|
||||
<value>Login</value>
|
||||
</data>
|
||||
<data name="Name" xml:space="preserve">
|
||||
<value>Name</value>
|
||||
</data>
|
||||
<data name="No_Galleries" xml:space="preserve">
|
||||
<value>No galleries created yet</value>
|
||||
</data>
|
||||
<data name="No_Pending_Uploads" xml:space="preserve">
|
||||
<value>No uploads pending review</value>
|
||||
</data>
|
||||
<data name="Pending" xml:space="preserve">
|
||||
<value>Pending</value>
|
||||
</data>
|
||||
<data name="Pending_Uploads" xml:space="preserve">
|
||||
<value>Pending Uploads</value>
|
||||
</data>
|
||||
<data name="Reject" xml:space="preserve">
|
||||
<value>Reject</value>
|
||||
</data>
|
||||
<data name="Rename" xml:space="preserve">
|
||||
<value>Rename</value>
|
||||
</data>
|
||||
<data name="Total" xml:space="preserve">
|
||||
<value>Total</value>
|
||||
</data>
|
||||
</root>
|
||||
186
WeddingShare/Resources/Views/Admin/LoginViewModel.fr-FR.resx
Normal file
186
WeddingShare/Resources/Views/Admin/LoginViewModel.fr-FR.resx
Normal file
@@ -0,0 +1,186 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<root>
|
||||
<!--
|
||||
Microsoft ResX Schema
|
||||
|
||||
Version 2.0
|
||||
|
||||
The primary goals of this format is to allow a simple XML format
|
||||
that is mostly human readable. The generation and parsing of the
|
||||
various data types are done through the TypeConverter classes
|
||||
associated with the data types.
|
||||
|
||||
Example:
|
||||
|
||||
... ado.net/XML headers & schema ...
|
||||
<resheader name="resmimetype">text/microsoft-resx</resheader>
|
||||
<resheader name="version">2.0</resheader>
|
||||
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
|
||||
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
|
||||
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
|
||||
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
|
||||
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
|
||||
<value>[base64 mime encoded serialized .NET Framework object]</value>
|
||||
</data>
|
||||
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
|
||||
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
|
||||
<comment>This is a comment</comment>
|
||||
</data>
|
||||
|
||||
There are any number of "resheader" rows that contain simple
|
||||
name/value pairs.
|
||||
|
||||
Each data row contains a name, and value. The row also contains a
|
||||
type or mimetype. Type corresponds to a .NET class that support
|
||||
text/value conversion through the TypeConverter architecture.
|
||||
Classes that don't support this are serialized and stored with the
|
||||
mimetype set.
|
||||
|
||||
The mimetype is used for serialized objects, and tells the
|
||||
ResXResourceReader how to depersist the object. This is currently not
|
||||
extensible. For a given mimetype the value must be set accordingly:
|
||||
|
||||
Note - application/x-microsoft.net.object.binary.base64 is the format
|
||||
that the ResXResourceWriter will generate, however the reader can
|
||||
read any of the formats listed below.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.binary.base64
|
||||
value : The object must be serialized with
|
||||
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
|
||||
: and then encoded with base64 encoding.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.soap.base64
|
||||
value : The object must be serialized with
|
||||
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter
|
||||
: and then encoded with base64 encoding.
|
||||
|
||||
mimetype: application/x-microsoft.net.object.bytearray.base64
|
||||
value : The object must be serialized into a byte array
|
||||
: using a System.ComponentModel.TypeConverter
|
||||
: and then encoded with base64 encoding.
|
||||
-->
|
||||
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
|
||||
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
|
||||
<xsd:element name="root" msdata:IsDataSet="true">
|
||||
<xsd:complexType>
|
||||
<xsd:choice maxOccurs="unbounded">
|
||||
<xsd:element name="metadata">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" use="required" type="xsd:string" />
|
||||
<xsd:attribute name="type" type="xsd:string" />
|
||||
<xsd:attribute name="mimetype" type="xsd:string" />
|
||||
<xsd:attribute ref="xml:space" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="assembly">
|
||||
<xsd:complexType>
|
||||
<xsd:attribute name="alias" type="xsd:string" />
|
||||
<xsd:attribute name="name" type="xsd:string" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="data">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
||||
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
|
||||
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
|
||||
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
|
||||
<xsd:attribute ref="xml:space" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
<xsd:element name="resheader">
|
||||
<xsd:complexType>
|
||||
<xsd:sequence>
|
||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
||||
</xsd:sequence>
|
||||
<xsd:attribute name="name" type="xsd:string" use="required" />
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
</xsd:choice>
|
||||
</xsd:complexType>
|
||||
</xsd:element>
|
||||
</xsd:schema>
|
||||
<resheader name="resmimetype">
|
||||
<value>text/microsoft-resx</value>
|
||||
</resheader>
|
||||
<resheader name="version">
|
||||
<value>2.0</value>
|
||||
</resheader>
|
||||
<resheader name="reader">
|
||||
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Actions" xml:space="preserve">
|
||||
<value>Actes</value>
|
||||
</data>
|
||||
<data name="Login" xml:space="preserve">
|
||||
<value>Se connecter</value>
|
||||
</data>
|
||||
<data name="Login_Username_Help" xml:space="preserve">
|
||||
<value>Veuillez entrer le nom d'utilisateur du compte.</value>
|
||||
</data>
|
||||
<data name="Login_Username_Placeholder" xml:space="preserve">
|
||||
<value>admin</value>
|
||||
</data>
|
||||
<data name="Login_Password_Help" xml:space="preserve">
|
||||
<value>Veuillez entrer le mot de passe du compte.</value>
|
||||
</data>
|
||||
<data name="Login_Password_Placeholder" xml:space="preserve">
|
||||
<value>Mot de passe</value>
|
||||
</data>
|
||||
<data name="Approve" xml:space="preserve">
|
||||
<value>Approuver</value>
|
||||
</data>
|
||||
<data name="Approved" xml:space="preserve">
|
||||
<value>Approuvée</value>
|
||||
</data>
|
||||
<data name="Available_Galleries" xml:space="preserve">
|
||||
<value>Galeries disponibles</value>
|
||||
</data>
|
||||
<data name="Create" xml:space="preserve">
|
||||
<value>Créer</value>
|
||||
</data>
|
||||
<data name="Delete" xml:space="preserve">
|
||||
<value>Supprimer</value>
|
||||
</data>
|
||||
<data name="Key" xml:space="preserve">
|
||||
<value>Clé</value>
|
||||
</data>
|
||||
<data name="Link" xml:space="preserve">
|
||||
<value>Lien</value>
|
||||
</data>
|
||||
<data name="Login" xml:space="preserve">
|
||||
<value>Se connecter</value>
|
||||
</data>
|
||||
<data name="Name" xml:space="preserve">
|
||||
<value>Nom</value>
|
||||
</data>
|
||||
<data name="No_Galleries" xml:space="preserve">
|
||||
<value>Aucune galerie n'a encore été créée</value>
|
||||
</data>
|
||||
<data name="No_Pending_Uploads" xml:space="preserve">
|
||||
<value>Aucun téléchargement en attente de révision</value>
|
||||
</data>
|
||||
<data name="Pending" xml:space="preserve">
|
||||
<value>En attente</value>
|
||||
</data>
|
||||
<data name="Pending_Uploads" xml:space="preserve">
|
||||
<value>Téléchargements en attente</value>
|
||||
</data>
|
||||
<data name="Reject" xml:space="preserve">
|
||||
<value>Rejeter</value>
|
||||
</data>
|
||||
<data name="Rename" xml:space="preserve">
|
||||
<value>Rebaptiser</value>
|
||||
</data>
|
||||
<data name="Total" xml:space="preserve">
|
||||
<value>Totale</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,10 +117,28 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Unexpected_Error" xml:space="preserve">
|
||||
<value>Opps, it looks like an unexpected error occurred...</value>
|
||||
</data>
|
||||
<data name="Unexpected_Error_Message" xml:space="preserve">
|
||||
<data name="400_Error_Message" xml:space="preserve">
|
||||
<value>If this issue persists please contact the web administrator</value>
|
||||
</data>
|
||||
<data name="400_Error_Title" xml:space="preserve">
|
||||
<value>Opps, it looks like an unexpected error occurred...</value>
|
||||
</data>
|
||||
<data name="401_Error_Message" xml:space="preserve">
|
||||
<value>It looks like you are not authorized to view that page...</value>
|
||||
</data>
|
||||
<data name="401_Error_Title" xml:space="preserve">
|
||||
<value>Access Denied</value>
|
||||
</data>
|
||||
<data name="402_Error_Message" xml:space="preserve">
|
||||
<value>The provided secret key was incorrect</value>
|
||||
</data>
|
||||
<data name="402_Error_Title" xml:space="preserve">
|
||||
<value>Invalid Secret Key</value>
|
||||
</data>
|
||||
<data name="403_Error_Message" xml:space="preserve">
|
||||
<value>The specified gellery does not exist and you do not have permissions to create it</value>
|
||||
</data>
|
||||
<data name="403_Error_Title" xml:space="preserve">
|
||||
<value>Access Denied</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,10 +117,28 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Unexpected_Error" xml:space="preserve">
|
||||
<value>Oups, il semble qu'une erreur inattendue se soit produite...</value>
|
||||
</data>
|
||||
<data name="Unexpected_Error_Message" xml:space="preserve">
|
||||
<data name="400_Error_Message" xml:space="preserve">
|
||||
<value>Si ce problème persiste, veuillez contacter l'administrateur Web</value>
|
||||
</data>
|
||||
<data name="400_Error_Title" xml:space="preserve">
|
||||
<value>Oups, il semble qu'une erreur inattendue se soit produite...</value>
|
||||
</data>
|
||||
<data name="401_Error_Message" xml:space="preserve">
|
||||
<value>Il semble que vous n'êtes pas autorisé à voir cette page...</value>
|
||||
</data>
|
||||
<data name="401_Error_Title" xml:space="preserve">
|
||||
<value>Accès refusé</value>
|
||||
</data>
|
||||
<data name="402_Error_Message" xml:space="preserve">
|
||||
<value>La clé secrète fournie était incorrecte</value>
|
||||
</data>
|
||||
<data name="402_Error_Title" xml:space="preserve">
|
||||
<value>Clé secrète invalide</value>
|
||||
</data>
|
||||
<data name="403_Error_Message" xml:space="preserve">
|
||||
<value>La galerie spécifiée n'existe pas et vous n'avez pas les autorisations pour la créer</value>
|
||||
</data>
|
||||
<data name="403_Error_Title" xml:space="preserve">
|
||||
<value>Accès refusé</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,10 +117,13 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Access_Denied" xml:space="preserve">
|
||||
<value>Access Denied</value>
|
||||
<data name="Share_Code" xml:space="preserve">
|
||||
<value>Share Code</value>
|
||||
</data>
|
||||
<data name="Access_Denied_Message" xml:space="preserve">
|
||||
<value>It looks like you are not authorized to view that page...</value>
|
||||
<data name="View" xml:space="preserve">
|
||||
<value>View</value>
|
||||
</data>
|
||||
<data name="Sort" xml:space="preserve">
|
||||
<value>Sort</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,10 +117,13 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Access_Denied" xml:space="preserve">
|
||||
<value>Accès refusé</value>
|
||||
<data name="Share_Code" xml:space="preserve">
|
||||
<value>Partager le code</value>
|
||||
</data>
|
||||
<data name="Access_Denied_Message" xml:space="preserve">
|
||||
<value>Il semble que vous n'êtes pas autorisé à voir cette page...</value>
|
||||
<data name="View" xml:space="preserve">
|
||||
<value>Voir</value>
|
||||
</data>
|
||||
<data name="Sort" xml:space="preserve">
|
||||
<value>Trier</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,7 +117,19 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Gallery_Empty" xml:space="preserve">
|
||||
<data name="Gallery_Empty_Upload" xml:space="preserve">
|
||||
<value>There are currently no memories shared, why don't you upload yours above!</value>
|
||||
</data>
|
||||
<data name="Gallery_Empty" xml:space="preserve">
|
||||
<value>There are currently no memories shared</value>
|
||||
</data>
|
||||
<data name="Total" xml:space="preserve">
|
||||
<value>Total</value>
|
||||
</data>
|
||||
<data name="Approved" xml:space="preserve">
|
||||
<value>Approved</value>
|
||||
</data>
|
||||
<data name="Pending" xml:space="preserve">
|
||||
<value>Pending</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,7 +117,19 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Gallery_Empty_Upload" xml:space="preserve">
|
||||
<value>Il n'y a actuellement aucun souvenir partagé</value>
|
||||
</data>
|
||||
<data name="Gallery_Empty" xml:space="preserve">
|
||||
<value>Il n'y a actuellement aucun souvenir partagé, pourquoi ne pas télécharger le vôtre ci-dessus!</value>
|
||||
</data>
|
||||
<data name="Total" xml:space="preserve">
|
||||
<value>Totale</value>
|
||||
</data>
|
||||
<data name="Approved" xml:space="preserve">
|
||||
<value>Approuvée</value>
|
||||
</data>
|
||||
<data name="Pending" xml:space="preserve">
|
||||
<value>En attente</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,7 +117,4 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Share_Code" xml:space="preserve">
|
||||
<value>Share Code</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,7 +117,4 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Share_Code" xml:space="preserve">
|
||||
<value>Partager le code</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,15 +117,6 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Admin_Login" xml:space="preserve">
|
||||
<value>Admin Login</value>
|
||||
</data>
|
||||
<data name="Admin_Login_Help" xml:space="preserve">
|
||||
<value>Please enter the admin area password.</value>
|
||||
</data>
|
||||
<data name="Admin_Login_Placeholder" xml:space="preserve">
|
||||
<value>Password</value>
|
||||
</data>
|
||||
<data name="Gallery_Key" xml:space="preserve">
|
||||
<value>Gallery Key</value>
|
||||
</data>
|
||||
|
||||
@@ -117,15 +117,6 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Admin_Login" xml:space="preserve">
|
||||
<value>Connexion administrateur</value>
|
||||
</data>
|
||||
<data name="Admin_Login_Help" xml:space="preserve">
|
||||
<value>Veuillez entrer le mot de passe de la zone d'administration.</value>
|
||||
</data>
|
||||
<data name="Admin_Login_Placeholder" xml:space="preserve">
|
||||
<value>Mot de passe</value>
|
||||
</data>
|
||||
<data name="Gallery_Key" xml:space="preserve">
|
||||
<value>Clé de la galerie</value>
|
||||
</data>
|
||||
|
||||
@@ -117,10 +117,43 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Clean" xml:space="preserve">
|
||||
<value>Clean</value>
|
||||
</data>
|
||||
<data name="Clean_Confirmation" xml:space="preserve">
|
||||
<value>Are you sure you want to clean this gallery?</value>
|
||||
</data>
|
||||
<data name="Wipe" xml:space="preserve">
|
||||
<value>Wipe</value>
|
||||
</data>
|
||||
<data name="Wipe_Confirmation" xml:space="preserve">
|
||||
<value>Are you sure you want to wipe this gallery?</value>
|
||||
</data>
|
||||
<data name="Close" xml:space="preserve">
|
||||
<value>Close</value>
|
||||
</data>
|
||||
<data name="Default" xml:space="preserve">
|
||||
<value>Default</value>
|
||||
</data>
|
||||
<data name="Delete_Confirmation" xml:space="preserve">
|
||||
<value>Are you sure you want to delete this gallery?</value>
|
||||
</data>
|
||||
<data name="Gallery_Key" xml:space="preserve">
|
||||
<value>Secret Key</value>
|
||||
</data>
|
||||
<data name="Gallery_Key_Help" xml:space="preserve">
|
||||
<value>Please enter a new secret key for the gallery</value>
|
||||
</data>
|
||||
<data name="Gallery_Name" xml:space="preserve">
|
||||
<value>Gallery Name</value>
|
||||
</data>
|
||||
<data name="Gallery_Name_Help" xml:space="preserve">
|
||||
<value>Please enter a new name for the gallery</value>
|
||||
</data>
|
||||
<data name="Update" xml:space="preserve">
|
||||
<value>Update</value>
|
||||
</data>
|
||||
<data name="Wipe_All_Confirmation" xml:space="preserve">
|
||||
<value>Are you sure you want to wipe all galleries?</value>
|
||||
</data>
|
||||
</root>
|
||||
@@ -117,10 +117,43 @@
|
||||
<resheader name="writer">
|
||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
||||
</resheader>
|
||||
<data name="Clean" xml:space="preserve">
|
||||
<value>Faire le ménage</value>
|
||||
</data>
|
||||
<data name="Clean_Confirmation" xml:space="preserve">
|
||||
<value>Etes-vous sûr de vouloir nettoyer cette galerie ?</value>
|
||||
</data>
|
||||
<data name="Wipe" xml:space="preserve">
|
||||
<value>Essuyer</value>
|
||||
</data>
|
||||
<data name="Wipe_Confirmation" xml:space="preserve">
|
||||
<value>Etes-vous sûr de vouloir effacer cette galerie ?</value>
|
||||
</data>
|
||||
<data name="Close" xml:space="preserve">
|
||||
<value>Fermer</value>
|
||||
</data>
|
||||
<data name="Default" xml:space="preserve">
|
||||
<value>Défaut</value>
|
||||
</data>
|
||||
<data name="Delete_Confirmation" xml:space="preserve">
|
||||
<value>Etes-vous sûr de vouloir supprimer cette galerie ?</value>
|
||||
</data>
|
||||
<data name="Gallery_Key" xml:space="preserve">
|
||||
<value>Clé secrète</value>
|
||||
</data>
|
||||
<data name="Gallery_Key_Help" xml:space="preserve">
|
||||
<value>Veuillez saisir une nouvelle clé secrète pour la galerie</value>
|
||||
</data>
|
||||
<data name="Gallery_Name" xml:space="preserve">
|
||||
<value>Nom de la galerie</value>
|
||||
</data>
|
||||
<data name="Gallery_Name_Help" xml:space="preserve">
|
||||
<value>Veuillez saisir un nouveau nom pour la galerie</value>
|
||||
</data>
|
||||
<data name="Update" xml:space="preserve">
|
||||
<value>Mise à jour</value>
|
||||
</data>
|
||||
<data name="Wipe_All_Confirmation" xml:space="preserve">
|
||||
<value>Êtes-vous sûr de vouloir effacer toutes les galeries ?</value>
|
||||
</data>
|
||||
</root>
|
||||
39
WeddingShare/SqlScripts/SQLite/000 - Initialize Database.sql
Normal file
39
WeddingShare/SqlScripts/SQLite/000 - Initialize Database.sql
Normal file
@@ -0,0 +1,39 @@
|
||||
--
|
||||
-- Table structure for table `galleries`
|
||||
--
|
||||
DROP TABLE IF EXISTS `galleries`;
|
||||
CREATE TABLE `galleries` (
|
||||
`id` INTEGER NOT NULL PRIMARY KEY,
|
||||
`name` TEXT NOT NULL UNIQUE,
|
||||
`secret_key` TEXT NULL
|
||||
);
|
||||
|
||||
DROP TABLE IF EXISTS `gallery_items`;
|
||||
CREATE TABLE `gallery_items` (
|
||||
`id` INTEGER NOT NULL PRIMARY KEY,
|
||||
`gallery_id` INTEGER NOT NULL,
|
||||
`title` TEXT NOT NULL,
|
||||
`uploaded_by` TEXT NULL,
|
||||
`state` INTEGER NOT NULL DEFAULT `0`,
|
||||
FOREIGN KEY (`gallery_id`) REFERENCES `galleries` (`id`)
|
||||
);
|
||||
|
||||
INSERT INTO `galleries`
|
||||
(`id`, `name`, `secret_key`)
|
||||
VALUES
|
||||
(1, 'default', NULL);
|
||||
|
||||
--
|
||||
-- Table structure for table `users`
|
||||
--
|
||||
DROP TABLE IF EXISTS `users`;
|
||||
CREATE TABLE `users` (
|
||||
`id` INTEGER NOT NULL PRIMARY KEY,
|
||||
`username` TEXT NOT NULL UNIQUE,
|
||||
`email` TEXT NULL UNIQUE,
|
||||
`password` TEXT NOT NULL
|
||||
);
|
||||
|
||||
INSERT INTO `users`
|
||||
VALUES
|
||||
(1,'admin', NULL, 'admin');
|
||||
@@ -0,0 +1,5 @@
|
||||
--
|
||||
-- Table structure for table `users`
|
||||
--
|
||||
ALTER TABLE `users` ADD `failed_logins` INTEGER NOT NULL DEFAULT 0;
|
||||
ALTER TABLE `users` ADD `lockout_until` INTEGER;
|
||||
@@ -1,22 +1,40 @@
|
||||
using Microsoft.AspNetCore.Authentication.Cookies;
|
||||
using System.Globalization;
|
||||
using Microsoft.AspNetCore.Authentication.Cookies;
|
||||
using Microsoft.AspNetCore.Http.Features;
|
||||
using Microsoft.AspNetCore.Localization;
|
||||
using System.Globalization;
|
||||
using Microsoft.AspNetCore.Server.Kestrel.Core;
|
||||
using WeddingShare.BackgroundWorkers;
|
||||
using WeddingShare.Configurations;
|
||||
using WeddingShare.Helpers;
|
||||
using WeddingShare.Helpers.Database;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare
|
||||
{
|
||||
public class Startup
|
||||
{
|
||||
public Startup(IConfiguration configuration)
|
||||
private readonly ILoggerFactory _loggerFactory;
|
||||
private readonly ILogger _logger;
|
||||
|
||||
public Startup(IConfiguration configuration, ILoggerFactory loggerFactory)
|
||||
{
|
||||
Configuration = configuration;
|
||||
_loggerFactory = loggerFactory;
|
||||
_logger = loggerFactory.CreateLogger<Startup>();
|
||||
}
|
||||
|
||||
public IConfiguration Configuration { get; }
|
||||
|
||||
public void ConfigureServices(IServiceCollection services)
|
||||
{
|
||||
services.AddScoped<IConfigHelper, ConfigHelper>();
|
||||
{
|
||||
var config = new ConfigHelper(new EnvironmentWrapper(), Configuration, _loggerFactory.CreateLogger<ConfigHelper>());
|
||||
|
||||
services.AddDependencyInjectionConfiguration();
|
||||
services.AddDatabaseConfiguration(config);
|
||||
services.AddNotificationConfiguration(config);
|
||||
|
||||
services.AddHostedService<DirectoryScanner>();
|
||||
services.AddHostedService<NotificationReport>();
|
||||
|
||||
services.AddRazorPages();
|
||||
services.AddControllersWithViews().AddRazorRuntimeCompilation();
|
||||
@@ -26,6 +44,21 @@ namespace WeddingShare
|
||||
options.ResourcesPath = "Resources";
|
||||
});
|
||||
|
||||
services.Configure<KestrelServerOptions>(options =>
|
||||
{
|
||||
options.Limits.MaxRequestBodySize = int.MaxValue;
|
||||
});
|
||||
|
||||
services.Configure<FormOptions>(x =>
|
||||
{
|
||||
x.MultipartHeadersLengthLimit = Int32.MaxValue;
|
||||
x.MultipartBoundaryLengthLimit = Int32.MaxValue;
|
||||
x.MultipartBodyLengthLimit = Int64.MaxValue;
|
||||
x.ValueLengthLimit = Int32.MaxValue;
|
||||
x.BufferBodyLengthLimit = Int64.MaxValue;
|
||||
x.MemoryBufferThreshold = Int32.MaxValue;
|
||||
});
|
||||
|
||||
services.Configure<RequestLocalizationOptions>(options => {
|
||||
var supportedCultures = new[]
|
||||
{
|
||||
@@ -44,11 +77,10 @@ namespace WeddingShare
|
||||
options.Cookie.HttpOnly = false;
|
||||
options.ExpireTimeSpan = TimeSpan.FromMinutes(10);
|
||||
|
||||
options.LoginPath = "/Home";
|
||||
options.AccessDeniedPath = "/Error/AccessDenied";
|
||||
options.LoginPath = "/Admin/Login";
|
||||
options.AccessDeniedPath = $"/Error?Reason={ErrorCode.Unauthorized}";
|
||||
options.SlidingExpiration = true;
|
||||
});
|
||||
|
||||
services.AddSession();
|
||||
}
|
||||
|
||||
@@ -60,12 +92,18 @@ namespace WeddingShare
|
||||
app.UseHsts();
|
||||
}
|
||||
|
||||
app.UseHttpsRedirection();
|
||||
var config = new ConfigHelper(new EnvironmentWrapper(), Configuration, _loggerFactory.CreateLogger<ConfigHelper>());
|
||||
if (config.GetOrDefault("Settings", "Force_Https", false))
|
||||
{
|
||||
app.UseHttpsRedirection();
|
||||
}
|
||||
|
||||
app.UseStaticFiles();
|
||||
app.UseRouting();
|
||||
app.UseAuthentication();
|
||||
app.UseAuthorization();
|
||||
app.UseRequestLocalization();
|
||||
app.UseSession();
|
||||
|
||||
app.UseEndpoints(endpoints =>
|
||||
{
|
||||
|
||||
@@ -1,28 +1,81 @@
|
||||
@model IndexModel
|
||||
@using WeddingShare.Helpers
|
||||
@using WeddingShare.Views.Admin
|
||||
@inject Microsoft.Extensions.Localization.IStringLocalizer<IndexModel> _localizer
|
||||
@inject WeddingShare.Helpers.IConfigHelper _config
|
||||
@inject WeddingShare.Helpers.IGalleryHelper _gallery
|
||||
|
||||
@{
|
||||
var identityEnabled = _config.GetOrDefault("Settings", "Show_Identity_Request", true);
|
||||
}
|
||||
|
||||
<section class="py-1 pt-lg-5">
|
||||
<div class="container px-3 px-lg-1 my-3 mt-lg-1">
|
||||
@if (Model?.Galleries != null && Model.Galleries.Any())
|
||||
{
|
||||
var key = _config.GetOrDefault("Settings", "Secret_Key", string.Empty);
|
||||
var ctx = Context.Request;
|
||||
var link = $"{ctx.Scheme}://{ctx.Host}/Gallery?{(!string.IsNullOrEmpty(key) ? $"key={key}&" : string.Empty)}";
|
||||
var link = UrlHelper.Generate(Context, _config, "/Gallery");
|
||||
|
||||
<h1 class="display-6">@_localizer["Available_Galleries"].Value: </h1>
|
||||
<div class="row pb-2 pb-lg-4">
|
||||
<div class="col-12 text-center">
|
||||
@if (!_config.GetOrDefault("Settings", "Single_Gallery_Mode", false))
|
||||
{
|
||||
<div class="mx-0 mx-sm-5 px-2 px-xl-5 text-success d-inline-block">
|
||||
<i class="btnAddGallery fa-solid fa-calendar-plus fa-2x pointer" alt="Create"></i>
|
||||
<h6>Create</h6>
|
||||
</div>
|
||||
}
|
||||
<div class="mx-0 mx-sm-5 px-2 px-xl-5 text-warning d-inline-block">
|
||||
<i class="btnBulkReview fa-solid fa-thumbs-up fa-2x pointer" alt="Bulk Review"></i>
|
||||
<h6>Review</h6>
|
||||
</div>
|
||||
<div class="mx-0 mx-sm-5 px-2 px-xl-5 text-primary d-inline-block">
|
||||
<i class="btnImport fa-solid fa-upload fa-2x pointer" alt="Import"></i>
|
||||
<h6>Import</h6>
|
||||
</div>
|
||||
<div class="mx-0 mx-sm-5 px-2 px-xl-5 text-primary d-inline-block">
|
||||
<i class="btnExport fa-solid fa-download fa-2x pointer" alt="Export"></i>
|
||||
<h6>Export</h6>
|
||||
</div>
|
||||
<div class="mx-0 mx-sm-5 px-2 px-xl-5 text-danger d-inline-block">
|
||||
<i class="btnWipeAllGalleries fa-solid fa-broom fa-2x pointer" alt="Wipe"></i>
|
||||
<h6>Wipe</h6>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col-12">
|
||||
<h1 class="display-6">@_localizer["Available_Galleries"].Value: </h1>
|
||||
</div>
|
||||
</div>
|
||||
<table class="table table-bordered">
|
||||
<tr>
|
||||
<th>@_localizer["Name"].Value</th>
|
||||
<th>@_localizer["Link"].Value</th>
|
||||
<th class="col-5 col-md-4">@_localizer["Name"].Value</th>
|
||||
<th class="col-1 d-none d-md-table-cell">@_localizer["Total"].Value</th>
|
||||
<th class="col-1 d-none d-md-table-cell">@_localizer["Approved"].Value</th>
|
||||
<th class="col-1 d-none d-md-table-cell">@_localizer["Pending"].Value</th>
|
||||
<th class="col-1 d-none d-md-table-cell">@_localizer["Key"].Value</th>
|
||||
<th class="col-7 col-md-4">@_localizer["Actions"].Value</th>
|
||||
</tr>
|
||||
@foreach (var gallery in Model.Galleries)
|
||||
@foreach (var gallery in Model.Galleries.OrderBy(x => string.Equals("default", x.Name, StringComparison.OrdinalIgnoreCase) ? 0 : 1))
|
||||
{
|
||||
var galleryLink = $"{link}id={gallery.Key}";
|
||||
<tr>
|
||||
<td class="text-capitalize">@gallery.Key</td>
|
||||
<td><a href="@galleryLink">@galleryLink</a></td>
|
||||
var secretKey = await _gallery.GetSecretKey(gallery.Name);
|
||||
var galleryLink = $"{link}?id={gallery.Name}{(!string.IsNullOrWhiteSpace(secretKey) ? $"&key={secretKey}" : string.Empty)}";
|
||||
|
||||
<tr data-gallery-id="@gallery.Id" data-gallery-name="@gallery.Name" data-gallery-key="@secretKey">
|
||||
<td class="gallery-name text-capitalize">@gallery.Name</td>
|
||||
<td class="d-none d-md-table-cell text-center">@gallery.TotalItems</td>
|
||||
<td class="d-none d-md-table-cell text-center">@gallery.ApprovedItems</td>
|
||||
<td class="d-none d-md-table-cell text-center">@gallery.PendingItems</td>
|
||||
<td class="d-none d-md-table-cell text-center">
|
||||
<i class="fa-solid @(!string.IsNullOrWhiteSpace(secretKey) ? "fa-lock" : "fa-lock-open") m-0"></i>
|
||||
</td>
|
||||
<td>
|
||||
<i class="btnOpenGallery btn btn-outline-primary fa-solid fa-up-right-from-square" data-url="@galleryLink" data-target="@(_config.GetOrDefault("Settings", "Links_Open_New_Tab", true) ? "_blank" : "_self")" alt="Open"></i>
|
||||
<i class="btnDownloadGallery btn @(gallery.TotalItems > 0 ? "btn-outline-primary" : "btn-outline-disabled") fa-solid fa-download" alt="Download" @(gallery.TotalItems == 0 ? "disabled=disabled" : string.Empty)></i>
|
||||
<i class="btn @(gallery.Id > 1 ? "btnEditGallery btn-outline-success" : "btn-outline-disabled") fa-solid fa-pen-to-square" alt="Edit"></i>
|
||||
<i class="btn @(gallery.TotalItems > 0 ? "btnWipeGallery btn-outline-danger" : "btn-outline-disabled") fa-solid fa-broom" alt="Wipe" @(gallery.TotalItems == 0 ? "disabled=disabled" : string.Empty)></i>
|
||||
<i class="btn @(gallery.Id > 1 ? "btnDeleteGallery btn-outline-danger" : "btn-outline-disabled") fa-solid fa-trash-can" alt="Delete"></i>
|
||||
</td>
|
||||
</tr>
|
||||
}
|
||||
</table>
|
||||
@@ -46,24 +99,37 @@
|
||||
{
|
||||
foreach (var review in Model.PendingRequests)
|
||||
{
|
||||
<div class="col-12 mb-4 mb-lg-0 mb-1 mb-lg-5">
|
||||
@{
|
||||
var photo = review.Replace('\\', '/');
|
||||
var parts = photo.Split(new char[] { '/' }, StringSplitOptions.RemoveEmptyEntries | StringSplitOptions.TrimEntries);
|
||||
|
||||
<div class="card pending-approval">
|
||||
<div class="card-header">
|
||||
<h5 class="card-title m-0 text-capitalize">@parts[0]</h5>
|
||||
</div>
|
||||
<img src="/uploads/@photo.TrimStart('/')" loading="lazy" />
|
||||
<div class="card-body m-0 p-0">
|
||||
<div class="btn-group w-100" role="group" data-gallery-id="@parts[0]" data-photo-id="@parts[2]">
|
||||
<button type="button" class="btn btn-success btnReviewApprove">@_localizer["Approve"].Value</button>
|
||||
<button type="button" class="btn btn-danger btnReviewReject">@_localizer["Reject"].Value</button>
|
||||
</div>
|
||||
<div class="pending-approval col-12 col-lg-4 mb-4 mb-lg-0 mb-1 mb-lg-4">
|
||||
<div class="card">
|
||||
<a href="/uploads/@review.GalleryName/Pending/@review.Title" data-lightbox="@review.Id">
|
||||
<img src="/uploads/@review.GalleryName/Pending/@review.Title" class="review-tile" loading="lazy" />
|
||||
</a>
|
||||
<div class="card-header my-0 py-0">
|
||||
<div class="row card-title text-capitalize border-top py-0 pt-2">
|
||||
@if (identityEnabled)
|
||||
{
|
||||
<div class="col-12 col-lg-6 text-lg-center">
|
||||
<b>Gallery:</b> @review.GalleryName
|
||||
</div>
|
||||
<div class="col-12 col-lg-6 text-lg-center">
|
||||
<b>Uploader:</b> @(!string.IsNullOrWhiteSpace(review.UploadedBy) ? review.UploadedBy : "Anonymous")
|
||||
</div>
|
||||
}
|
||||
else
|
||||
{
|
||||
<div class="col-12">
|
||||
<b>Gallery:</b> @review.GalleryName
|
||||
</div>
|
||||
}
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
<div class="card-body m-0 p-0">
|
||||
<div class="btn-group w-100" role="group" data-id="@review.Id">
|
||||
<button type="button" class="btn btn-success btnReviewApprove">@_localizer["Approve"].Value</button>
|
||||
<button type="button" class="btn btn-danger btnReviewReject">@_localizer["Reject"].Value</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
}
|
||||
@@ -75,4 +141,6 @@
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
</section>
|
||||
|
||||
<script src="~/js/admin.js" asp-append-version="true"></script>
|
||||
@@ -1,4 +1,5 @@
|
||||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
using WeddingShare.Models.Database;
|
||||
|
||||
namespace WeddingShare.Views.Admin
|
||||
{
|
||||
@@ -8,8 +9,8 @@ namespace WeddingShare.Views.Admin
|
||||
{
|
||||
}
|
||||
|
||||
public List<KeyValuePair<string, string>>? Galleries { get; set; }
|
||||
public List<string>? PendingRequests { get; set; }
|
||||
public List<GalleryModel>? Galleries { get; set; }
|
||||
public List<PendingGalleryItemModel>? PendingRequests { get; set; }
|
||||
|
||||
public void OnGet()
|
||||
{
|
||||
|
||||
34
WeddingShare/Views/Admin/Login.cshtml
Normal file
34
WeddingShare/Views/Admin/Login.cshtml
Normal file
@@ -0,0 +1,34 @@
|
||||
@using WeddingShare.Views.Admin
|
||||
@inject Microsoft.Extensions.Localization.IStringLocalizer<LoginViewModel> _localizer
|
||||
@inject WeddingShare.Helpers.IConfigHelper _config
|
||||
|
||||
<section class="py-1 py-lg-5">
|
||||
<div class="container px-3 px-lg-1 my-3 mt-lg-1">
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<h1 class="display-6">@_localizer["Login"].Value</h1>
|
||||
</div>
|
||||
<div class="card-body m-1 m-lg-5">
|
||||
<form id="frmAdminLogin">
|
||||
<div class="row mb-4 mb-lg-5">
|
||||
<div class="col-12">
|
||||
<input type="text" class="input-username form-control" aria-describedby="admin-username-help" placeholder="@_localizer["Login_Username_Placeholder"].Value" aria-label="@_localizer["Login_Username"].Value" />
|
||||
<div class="form-text">@_localizer["Login_Username_Help"].Value</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row mb-4 mb-lg-5">
|
||||
<div class="col-12">
|
||||
<input type="password" class="input-password form-control" aria-describedby="admin-password-help" placeholder="@_localizer["Login_Password_Placeholder"].Value" aria-label="@_localizer["Login_Password"].Value" />
|
||||
<div class="form-text">@_localizer["Login_Password_Help"].Value</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row mt-4 mt-lg-5">
|
||||
<div class="col-12">
|
||||
<button id="btnAdminLogin" class="btn btn-primary w-100" type="submit">@_localizer["Login"].Value</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
@@ -1,8 +1,8 @@
|
||||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
|
||||
namespace WeddingShare.Views.Error
|
||||
namespace WeddingShare.Views.Admin
|
||||
{
|
||||
public class AccessDeniedModel : PageModel
|
||||
public class LoginViewModel : PageModel
|
||||
{
|
||||
public void OnGet()
|
||||
{
|
||||
@@ -1,10 +0,0 @@
|
||||
@model IndexModel
|
||||
@using WeddingShare.Views.Error
|
||||
@inject Microsoft.Extensions.Localization.IStringLocalizer<AccessDeniedModel> _localizer
|
||||
@inject WeddingShare.Helpers.IConfigHelper _config
|
||||
|
||||
<!--( Error Message )-->
|
||||
<div class="content border p-5">
|
||||
<p class="text-center mb-2" style="font-size: 40px;">@_localizer["Access_Denied"].Value</p>
|
||||
<p class="text-center" style="font-size: 18px;">@_localizer["Access_Denied_Message"].Value</p>
|
||||
</div>
|
||||
@@ -1,10 +1,23 @@
|
||||
@model IndexModel
|
||||
@using WeddingShare.Helpers
|
||||
@using WeddingShare.Views.Error
|
||||
@inject Microsoft.Extensions.Localization.IStringLocalizer<IndexModel> _localizer
|
||||
@inject WeddingShare.Helpers.IConfigHelper _config
|
||||
|
||||
@{
|
||||
var reasonCode = 400;
|
||||
try
|
||||
{
|
||||
reasonCode = Context.Request.Query.ContainsKey("Reason") ? Convert.ToInt32(Context.Request.Query["Reason"]) : 400;
|
||||
}
|
||||
catch { }
|
||||
|
||||
var title = _localizer[$"{reasonCode}_Error_Title"].Value;
|
||||
var message = _localizer[$"{reasonCode}_Error_Message"].Value;
|
||||
}
|
||||
|
||||
<!--( Error Message )-->
|
||||
<div class="content border p-5">
|
||||
<p class="text-center mb-2" style="font-size: 40px;">@_localizer["Unexpected_Error"].Value</p>
|
||||
<p class="text-center" style="font-size: 18px;">@_localizer["Unexpected_Error_Message"].Value</p>
|
||||
<p class="text-center mb-2" style="font-size: 40px;">@title</p>
|
||||
<p class="text-center" style="font-size: 18px;">@message</p>
|
||||
</div>
|
||||
56
WeddingShare/Views/Gallery/GalleryOptions.cshtml
Normal file
56
WeddingShare/Views/Gallery/GalleryOptions.cshtml
Normal file
@@ -0,0 +1,56 @@
|
||||
@using System.Text
|
||||
@using WeddingShare.Enums
|
||||
@using WeddingShare.Helpers
|
||||
@using WeddingShare.Views.Gallery
|
||||
@inject Microsoft.Extensions.Localization.IStringLocalizer<GalleryOptionsModel> _localizer
|
||||
@inject WeddingShare.Helpers.IGalleryHelper _gallery
|
||||
@model WeddingShare.Models.PhotoGallery
|
||||
|
||||
@if (!_gallery.GetConfig(Model?.GalleryId, "Disable_QR_Code", false))
|
||||
{
|
||||
<div class="row mb-4">
|
||||
<div class="col-12 text-center">
|
||||
<div id="qrcode"></div>
|
||||
<h6>@_localizer["Share_Code"].Value</h6>
|
||||
<script>
|
||||
document.addEventListener('DOMContentLoaded', function () {
|
||||
if ($('#qrcode').length > 0 && $('#qrcode').is(':visible')) {
|
||||
$(function () {
|
||||
$('#qrcode').qrcode({ width: 150, height: 150, text: '@Html.Raw(ViewBag.QRCodeLink)' });
|
||||
});
|
||||
}
|
||||
}, false);
|
||||
</script>
|
||||
</div>
|
||||
</div>
|
||||
}
|
||||
|
||||
<div class="row px-4 mb-4 presentation-hidden">
|
||||
<div class="col-12 px-5">
|
||||
<div class="btn-group w-100 mb-2">
|
||||
<button class="btn btn-primary btn-sm dropdown-toggle" type="button" data-bs-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
|
||||
@_localizer["View"].Value
|
||||
</button>
|
||||
<div class="dropdown-menu">
|
||||
@foreach (ViewMode mode in Enum.GetValues(typeof(ViewMode)))
|
||||
{
|
||||
<a class="dropdown-item" href="@($"{ViewBag.QRCodeLink}{(ViewBag.QRCodeLink.Contains("?") ? "&" : "?")}mode={(int)mode}")">@mode</a>
|
||||
}
|
||||
</div>
|
||||
</div>
|
||||
<div class="btn-group w-100 mb-2">
|
||||
<button class="btn btn-primary btn-sm dropdown-toggle" type="button" data-bs-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
|
||||
@_localizer["Sort"].Value
|
||||
</button>
|
||||
<div class="dropdown-menu">
|
||||
@foreach (GalleryOrder order in Enum.GetValues(typeof(GalleryOrder)))
|
||||
{
|
||||
if (order != GalleryOrder.None)
|
||||
{
|
||||
<a class="dropdown-item" href="@($"{ViewBag.BaseLink}{(ViewBag.BaseLink.Contains("?") ? "&" : "?")}order={(int)order}")">@order</a>
|
||||
}
|
||||
}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
11
WeddingShare/Views/Gallery/GalleryOptions.cshtml.cs
Normal file
11
WeddingShare/Views/Gallery/GalleryOptions.cshtml.cs
Normal file
@@ -0,0 +1,11 @@
|
||||
using Microsoft.AspNetCore.Mvc.RazorPages;
|
||||
|
||||
namespace WeddingShare.Views.Gallery
|
||||
{
|
||||
public class GalleryOptionsModel : PageModel
|
||||
{
|
||||
public void OnGet()
|
||||
{
|
||||
}
|
||||
}
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user