Compare commits
115 commits
Author | SHA1 | Date | |
---|---|---|---|
a6d2426610 | |||
ef61c58504 | |||
f19aa99ca9 | |||
f9eb0e78f0 | |||
e3935c29f6 | |||
db0787ac32 | |||
a28caf0fba | |||
c3903f4979 | |||
87bab38720 | |||
ab6cc7f413 | |||
eaa11afa70 | |||
5ab5f9fb91 | |||
5d9d261eb9 | |||
b670a1c15f | |||
9bd94843b4 | |||
a20271db6d | |||
9c44d0443e | |||
50925970fc | |||
0c77163f31 | |||
bd32d6c149 | |||
7daa9a9687 | |||
1c3dfd2da1 | |||
a338c65892 | |||
2b2c3ba70e | |||
056ddff816 | |||
ac86d1ee59 | |||
5788a26880 | |||
ec72015524 | |||
19b352c381 | |||
4558dfb8da | |||
a3eb5d29ee | |||
c2be91dfcc | |||
d7ced42d86 | |||
ccd9e8280e | |||
b14a5cfb83 | |||
f5dbde0ed6 | |||
a2654a6f01 | |||
cd5af6b907 | |||
68285fbc39 | |||
3f3cb9ef7c | |||
493d17abfd | |||
bf740857b3 | |||
cd259f244a | |||
d235c56cfb | |||
74ba9d23f0 | |||
0224db1993 | |||
3ed60ae2dd | |||
af22c0d71f | |||
d90f7cdfea | |||
78c8b9ef5c | |||
f0d1e933c6 | |||
5f32fd1142 | |||
ae5606f4be | |||
82f6535765 | |||
0f4a5f547f | |||
b36f1240b0 | |||
181f4bb749 | |||
fb600aa5fc | |||
85bf0fef59 | |||
a134947df6 | |||
c5724d6d4a | |||
51ae13e53b | |||
4c13ddcc48 | |||
fb436f1499 | |||
528fdebe61 | |||
c58aa2f8c0 | |||
5ce72e5d95 | |||
7258af44c3 | |||
2b7ce1d186 | |||
eda0f9336c | |||
bacb2369e0 | |||
e586321faf | |||
e0f263f7f0 | |||
ea731df20d | |||
9376d959f8 | |||
6ae6dd8430 | |||
e0c07091e9 | |||
51471cd58f | |||
5ffb1b4a83 | |||
7672cdc083 | |||
ac0e74c818 | |||
b49a996bf5 | |||
013feabd20 | |||
6e0cb54032 | |||
4bd49c76d4 | |||
fb0144bbe1 | |||
cd93f0031d | |||
87efbf0613 | |||
547d5511fa | |||
b638b4bbce | |||
bd4d611d04 | |||
f29c5ecb03 | |||
aa3794b504 | |||
f469ed6763 | |||
f632b48f32 | |||
aff5e4f03d | |||
ae198541ee | |||
9a87d43459 | |||
f21700f88b | |||
e35304f030 | |||
05d0042a60 | |||
9f64683474 | |||
353c91204a | |||
c2fc8b8ae9 | |||
e4e95b81b2 | |||
f625373e2e | |||
33df90eb96 | |||
88ae856195 | |||
31eb42c1d5 | |||
9d81bd7472 | |||
c8f1d250b5 | |||
857a298b96 | |||
958f04ee11 | |||
ee95038a44 | |||
9f13cbf302 |
25 changed files with 2354 additions and 1378 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -1,2 +1,3 @@
|
||||||
.stack-work/
|
.stack-work/
|
||||||
*~
|
dist-newstyle/
|
||||||
|
*~
|
||||||
|
|
4
.gitmodules
vendored
Normal file
4
.gitmodules
vendored
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
[submodule "zcash-haskell"]
|
||||||
|
path = zcash-haskell
|
||||||
|
url = https://git.vergara.tech/Vergara_Tech/zcash-haskell
|
||||||
|
branch = milestone2
|
63
CHANGELOG.md
63
CHANGELOG.md
|
@ -4,7 +4,68 @@ All notable changes to this project will be documented in this file.
|
||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
## [1.5.0]
|
## [1.8.1]
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Changed license to MIT
|
||||||
|
- Updated to Haskell LTS 21.22
|
||||||
|
- Update to new version of `zcash-haskell`
|
||||||
|
|
||||||
|
## [1.8.0]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Parser for Unified Addresses that validates the address
|
||||||
|
- Tests for UA parsing from wallets
|
||||||
|
- Function to scan new transactions using known viewing keys
|
||||||
|
- Function to identify the owners and VKs needed for tx scans
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Order endpoint updated to ensure orders belong to shop before adding to DB.
|
||||||
|
- MongoDB driver updated to support MongoDB 6.
|
||||||
|
- Full validation of Sapling addresses to parser.
|
||||||
|
|
||||||
|
### Removed
|
||||||
|
|
||||||
|
- `api/orderx` endpoint.
|
||||||
|
- `makeZcashCall` function moved to the generic `zcash-haskell` library.
|
||||||
|
- `RpcResponse`, `RpcCall` types moved to the generic `zcash-haskell` library.
|
||||||
|
|
||||||
|
## [1.7.0]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Parameter to config for number of confirmations for scan
|
||||||
|
- Endpoint for language for invoices
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Modified payment confirmation to use new WooCommerce plugin API endpoint.
|
||||||
|
- Consolidated the `invdata`, `orderid` and `orderx` endpoints
|
||||||
|
- The `xerotoken` endpoint uses `session` for authentication
|
||||||
|
- The order by ID/token endpoint includes shop name
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- The viewing key obfuscation of blank viewing keys
|
||||||
|
|
||||||
|
## [1.6.0]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- New JSON serialization for WooTokens.
|
||||||
|
- New `/api/ownervk` endpoint to save viewing keys
|
||||||
|
- Use of `zcash-haskell` library to validate Sapling viewing keys
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Modified the process of scanning for payments to only scan addresses that have an active ZGo session and have enabled payment confirmations
|
||||||
|
- Modified the process to mark paid orders to ensure only payments to the shop's wallet get marked as paid
|
||||||
|
- Modified the `items` endpoint to use the login session to identify records
|
||||||
|
|
||||||
|
## [1.5.0] - 2023-05-15
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
|
|
7
COPYING
7
COPYING
|
@ -1,7 +0,0 @@
|
||||||
Copyright (c) 2022 Vergara Technologies LLC
|
|
||||||
|
|
||||||
This package ("Original Work") is licensed under the terms of the Bootstrap
|
|
||||||
Open Source License, version 1.0, or at your option, any later version
|
|
||||||
("BOSL"). See the file ./LICENSE for the terms of the Bootstrap Open Source
|
|
||||||
Licence, version 1.0.
|
|
||||||
|
|
191
LICENSE
191
LICENSE
|
@ -1,178 +1,21 @@
|
||||||
Copyright (c) 2023 Vergara Technologies LLC
|
MIT License
|
||||||
|
|
||||||
=======================================================
|
Copyright (c) 2022-2024 Vergara Technologies LLC
|
||||||
Bootstrap Open Source Licence ("BOSL") v. 1.0
|
|
||||||
=======================================================
|
|
||||||
This Bootstrap Open Source Licence (the "License") applies to any original work
|
|
||||||
of authorship (the "Original Work") whose owner (the "Licensor") has placed the
|
|
||||||
following licensing notice adjacent to the copyright notice for the Original
|
|
||||||
Work:
|
|
||||||
|
|
||||||
*Licensed under the Bootstrap Open Source Licence version 1.0*
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
1. **Grant of Copyright License.** Licensor grants You a worldwide,
|
The above copyright notice and this permission notice shall be included in all
|
||||||
royalty-free, non-exclusive, sublicensable license, for the duration of the
|
copies or substantial portions of the Software.
|
||||||
copyright in the Original Work, to do the following:
|
|
||||||
|
|
||||||
a. to reproduce the Original Work in copies, either alone or as part of
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
a collective work;
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
b. to translate, adapt, alter, transform, modify, or arrange the
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
Original Work, thereby creating derivative works ("Derivative Works")
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
based upon the Original Work;
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
c. to distribute or communicate copies of the Original Work and
|
|
||||||
Derivative Works to the public, provided that prior to any such
|
|
||||||
distribution or communication You first place a machine-readable copy
|
|
||||||
of the Source Code of the Original Work and such Derivative Works that
|
|
||||||
You intend to distribute or communicate in an information repository
|
|
||||||
reasonably calculated to permit inexpensive and convenient access
|
|
||||||
thereto by the public (“Information Repository”) for as long as You
|
|
||||||
continue to distribute or communicate said copies, accompanied by an
|
|
||||||
irrevocable offer to license said copies to the public free of charge
|
|
||||||
under this License, said offer valid starting no later than 12 months
|
|
||||||
after You first distribute or communicate said copies;
|
|
||||||
|
|
||||||
d. to perform the Original Work publicly; and
|
|
||||||
|
|
||||||
e. to display the Original Work publicly.
|
|
||||||
|
|
||||||
2. **Grant of Patent License.** Licensor grants You a worldwide, royalty-free,
|
|
||||||
non-exclusive, sublicensable license, under patent claims owned or controlled
|
|
||||||
by the Licensor that are embodied in the Original Work as furnished by the
|
|
||||||
Licensor, for the duration of the patents, to make, use, sell, offer for sale,
|
|
||||||
have made, and import the Original Work and Derivative Works.
|
|
||||||
|
|
||||||
3. **Grant of Source Code License.** The "Source Code" for a work means the
|
|
||||||
preferred form of the work for making modifications to it and all available
|
|
||||||
documentation describing how to modify the work. Licensor agrees to provide a
|
|
||||||
machine-readable copy of the Source Code of the Original Work along with each
|
|
||||||
copy of the Original Work that Licensor distributes. Licensor reserves the
|
|
||||||
right to satisfy this obligation by placing a machine-readable copy of said
|
|
||||||
Source Code in an Information Repository for as long as Licensor continues to
|
|
||||||
distribute the Original Work.
|
|
||||||
|
|
||||||
4. **Exclusions From License Grant.** Neither the names of Licensor, nor the
|
|
||||||
names of any contributors to the Original Work, nor any of their trademarks or
|
|
||||||
service marks, may be used to endorse or promote products derived from this
|
|
||||||
Original Work without express prior permission of the Licensor. Except as
|
|
||||||
expressly stated herein, nothing in this License grants any license to
|
|
||||||
Licensor's trademarks, copyrights, patents, trade secrets or any other
|
|
||||||
intellectual property. No patent license is granted to make, use, sell, offer
|
|
||||||
for sale, have made, or import embodiments of any patent claims other than the
|
|
||||||
licensed claims defined in Section 2. No license is granted to the trademarks
|
|
||||||
of Licensor even if such marks are included in the Original Work. Nothing in
|
|
||||||
this License shall be interpreted to prohibit Licensor from licensing under
|
|
||||||
terms different from this License any Original Work that Licensor otherwise
|
|
||||||
would have a right to license.
|
|
||||||
|
|
||||||
5. **External Deployment.** The term "External Deployment" means the use,
|
|
||||||
distribution, or communication of the Original Work or Derivative Works in any
|
|
||||||
way such that the Original Work or Derivative Works may be used by anyone other
|
|
||||||
than You, whether those works are distributed or communicated to those persons
|
|
||||||
or made available as an application intended for use over a network. As an
|
|
||||||
express condition for the grants of license hereunder, You must treat any
|
|
||||||
External Deployment by You of the Original Work or a Derivative Work as a
|
|
||||||
distribution under section 1(c).
|
|
||||||
|
|
||||||
6. **Attribution Rights.** You must retain, in the Source Code of any
|
|
||||||
Derivative Works that You create, all copyright, patent, or trademark notices
|
|
||||||
from the Source Code of the Original Work, as well as any notices of licensing
|
|
||||||
and any descriptive text identified therein as an "Attribution Notice." You
|
|
||||||
must cause the Source Code for any Derivative Works that You create to carry a
|
|
||||||
prominent Attribution Notice reasonably calculated to inform recipients that
|
|
||||||
You have modified the Original Work.
|
|
||||||
|
|
||||||
7. **Warranty of Provenance and Disclaimer of Warranty.** Licensor warrants
|
|
||||||
that the copyright in and to the Original Work and the patent rights granted
|
|
||||||
herein by Licensor are owned by the Licensor or are sublicensed to You under
|
|
||||||
the terms of this License with the permission of the contributor(s) of those
|
|
||||||
copyrights and patent rights. Except as expressly stated in the immediately
|
|
||||||
preceding sentence, the Original Work is provided under this License on an "AS
|
|
||||||
IS" BASIS and WITHOUT WARRANTY, either express or implied, including, without
|
|
||||||
limitation, the warranties of non-infringement, merchantability or fitness for
|
|
||||||
a particular purpose. THE ENTIRE RISK AS TO THE QUALITY OF THE ORIGINAL WORK IS
|
|
||||||
WITH YOU. This DISCLAIMER OF WARRANTY constitutes an essential part of this
|
|
||||||
License. No license to the Original Work is granted by this License except
|
|
||||||
under this disclaimer.
|
|
||||||
|
|
||||||
8. **Limitation of Liability.** Under no circumstances and under no legal
|
|
||||||
theory, whether in tort (including negligence), contract, or otherwise, shall
|
|
||||||
the Licensor be liable to anyone for any indirect, special, incidental, or
|
|
||||||
consequential damages of any character arising as a result of this License or
|
|
||||||
the use of the Original Work including, without limitation, damages for loss of
|
|
||||||
goodwill, work stoppage, computer failure or malfunction, or any and all other
|
|
||||||
commercial damages or losses. This limitation of liability shall not apply to
|
|
||||||
the extent applicable law prohibits such limitation.
|
|
||||||
|
|
||||||
9. **Acceptance and Termination.** If, at any time, You expressly assented to
|
|
||||||
this License, that assent indicates your clear and irrevocable acceptance of
|
|
||||||
this License and all of its terms and conditions. If You distribute or
|
|
||||||
communicate copies of the Original Work or a Derivative Work, You must make a
|
|
||||||
reasonable effort under the circumstances to obtain the express assent of
|
|
||||||
recipients to the terms of this License. This License conditions your rights to
|
|
||||||
undertake the activities listed in Section 1, including your right to create
|
|
||||||
Derivative Works based upon the Original Work, and doing so without honoring
|
|
||||||
these terms and conditions is prohibited by copyright law and international
|
|
||||||
treaty. Nothing in this License is intended to affect copyright exceptions and
|
|
||||||
limitations (including 'fair use' or 'fair dealing'). This License shall
|
|
||||||
terminate immediately and You may no longer exercise any of the rights granted
|
|
||||||
to You by this License upon your failure to honor the conditions in Section
|
|
||||||
1(c).
|
|
||||||
|
|
||||||
10. **Termination for Patent Action.** This License shall terminate
|
|
||||||
automatically and You may no longer exercise any of the rights granted to You
|
|
||||||
by this License as of the date You commence an action, including a cross-claim
|
|
||||||
or counterclaim, against Licensor or any licensee alleging that the Original
|
|
||||||
Work infringes a patent. This termination provision shall not apply for an
|
|
||||||
action alleging patent infringement by combinations of the Original Work with
|
|
||||||
other software or hardware.
|
|
||||||
|
|
||||||
11. **Jurisdiction, Venue and Governing Law.** Any action or suit relating to
|
|
||||||
this License may be brought only in the courts of a jurisdiction wherein the
|
|
||||||
Licensor resides or in which Licensor conducts its primary business, and under
|
|
||||||
the laws of that jurisdiction excluding its conflict-of-law provisions. The
|
|
||||||
application of the United Nations Convention on Contracts for the International
|
|
||||||
Sale of Goods is expressly excluded. Any use of the Original Work outside the
|
|
||||||
scope of this License or after its termination shall be subject to the
|
|
||||||
requirements and penalties of copyright or patent law in the appropriate
|
|
||||||
jurisdiction. This section shall survive the termination of this License.
|
|
||||||
|
|
||||||
12. **Attorneys' Fees.** In any action to enforce the terms of this License or
|
|
||||||
seeking damages relating thereto, the prevailing party shall be entitled to
|
|
||||||
recover its costs and expenses, including, without limitation, reasonable
|
|
||||||
attorneys' fees and costs incurred in connection with such action, including
|
|
||||||
any appeal of such action. This section shall survive the termination of this
|
|
||||||
License.
|
|
||||||
|
|
||||||
13. **Miscellaneous.** If any provision of this License is held to be
|
|
||||||
unenforceable, such provision shall be reformed only to the extent necessary to
|
|
||||||
make it enforceable.
|
|
||||||
|
|
||||||
14. **Definition of "You" in This License.** "You" throughout this License,
|
|
||||||
whether in upper or lower case, means an individual or a legal entity
|
|
||||||
exercising rights under, and complying with all of the terms of, this License.
|
|
||||||
For legal entities, "You" includes any entity that controls, is controlled by,
|
|
||||||
or is under common control with you. For purposes of this definition, "control"
|
|
||||||
means (i) the power, direct or indirect, to cause the direction or management
|
|
||||||
of such entity, whether by contract or otherwise, or (ii) ownership of fifty
|
|
||||||
percent (50%) or more of the outstanding shares, or (iii) beneficial ownership
|
|
||||||
of such entity.
|
|
||||||
|
|
||||||
15. **Right to Use.** You may use the Original Work in all ways not otherwise
|
|
||||||
restricted or conditioned by this License or by law, and Licensor promises not
|
|
||||||
to interfere with or be responsible for such uses by You.
|
|
||||||
|
|
||||||
16. **Modification of This License.** This License is Copyright © 2007 Zooko
|
|
||||||
Wilcox-O'Hearn. Permission is granted to copy, distribute, or communicate this
|
|
||||||
License without modification. Nothing in this License permits You to modify
|
|
||||||
this License as applied to the Original Work or to Derivative Works. However,
|
|
||||||
You may modify the text of this License and copy, distribute or communicate
|
|
||||||
your modified version (the "Modified License") and apply it to other original
|
|
||||||
works of authorship subject to the following conditions: (i) You may not
|
|
||||||
indicate in any way that your Modified License is the "Bootstrap Open Source
|
|
||||||
Licence" or "BOSL" and you may not use those names in the name of your Modified
|
|
||||||
License; and (ii) You must replace the notice specified in the first paragraph
|
|
||||||
above with the notice "Licensed under <insert your license name here>" or with
|
|
||||||
a notice of your own that is not confusingly similar to the notice in this
|
|
||||||
License.
|
|
||||||
|
|
|
@ -6,7 +6,8 @@ The API server behind the [ZGo.cash](https://zgo.cash) app.
|
||||||
|
|
||||||
## Dependencies
|
## Dependencies
|
||||||
|
|
||||||
- Zcash Full node
|
- Zcash Full node (`zcashd`)
|
||||||
|
- [Zcash Haskell](https://git.vergara.tech/Vergara_Tech/zcash-haskell)
|
||||||
- MongoDB
|
- MongoDB
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
|
@ -23,10 +23,12 @@ main = do
|
||||||
putStrLn "Connected to MongoDB!"
|
putStrLn "Connected to MongoDB!"
|
||||||
checkZcashPrices pipe (c_dbName loadedConfig)
|
checkZcashPrices pipe (c_dbName loadedConfig)
|
||||||
scanZcash' loadedConfig pipe
|
scanZcash' loadedConfig pipe
|
||||||
scanPayments loadedConfig pipe
|
{-scanPayments loadedConfig pipe-}
|
||||||
|
scanTxNative loadedConfig pipe
|
||||||
checkPayments pipe (c_dbName loadedConfig)
|
checkPayments pipe (c_dbName loadedConfig)
|
||||||
expireOwners pipe (c_dbName loadedConfig)
|
expireOwners pipe (c_dbName loadedConfig)
|
||||||
updateLogins pipe loadedConfig
|
updateLogins pipe loadedConfig
|
||||||
expireProSessions pipe (c_dbName loadedConfig)
|
expireProSessions pipe (c_dbName loadedConfig)
|
||||||
|
loadTranslations pipe loadedConfig
|
||||||
close pipe
|
close pipe
|
||||||
else fail "MongoDB connection failed!"
|
else fail "MongoDB connection failed!"
|
||||||
|
|
16
cabal.project
Normal file
16
cabal.project
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
packages:
|
||||||
|
./*.cabal
|
||||||
|
zcash-haskell/zcash-haskell.cabal
|
||||||
|
|
||||||
|
with-compiler: ghc-9.6.5
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://git.vergara.tech/Vergara_Tech/haskell-hexstring.git
|
||||||
|
tag: 39d8da7b11a80269454c2f134a5c834e0f3cb9a7
|
||||||
|
|
||||||
|
source-repository-package
|
||||||
|
type: git
|
||||||
|
location: https://git.vergara.tech/Vergara_Tech/haskell-foreign-rust.git
|
||||||
|
tag: 335e804454cd30da2c526457be37e477f71e4665
|
||||||
|
|
163
package.yaml
163
package.yaml
|
@ -1,163 +0,0 @@
|
||||||
name: zgo-backend
|
|
||||||
version: 1.5.0
|
|
||||||
git: "https://git.vergara.tech/Vergara_Tech/zgo-backend"
|
|
||||||
license: BOSL
|
|
||||||
author: "Rene Vergara"
|
|
||||||
maintainer: "rene@vergara.network"
|
|
||||||
copyright: "Copyright (c) 2023 Vergara Technologies LLC"
|
|
||||||
|
|
||||||
extra-source-files:
|
|
||||||
- README.md
|
|
||||||
- CHANGELOG.md
|
|
||||||
- zgo.cfg
|
|
||||||
|
|
||||||
# Metadata used when publishing your package
|
|
||||||
synopsis: Haskell Back-end for the ZGo point-of-sale application
|
|
||||||
category: Web
|
|
||||||
|
|
||||||
# To avoid duplicated efforts in documentation and dealing with the
|
|
||||||
# complications of embedding Haddock markup inside cabal files, it is
|
|
||||||
# common to point users to the README.md file.
|
|
||||||
description: Please see the README at <https://git.vergara.tech/Vergara_Tech//zgo-backend#readme>
|
|
||||||
|
|
||||||
dependencies:
|
|
||||||
- base >= 4.7 && < 5
|
|
||||||
|
|
||||||
library:
|
|
||||||
source-dirs: src
|
|
||||||
dependencies:
|
|
||||||
- mongoDB
|
|
||||||
- time
|
|
||||||
- text
|
|
||||||
- unordered-containers
|
|
||||||
- bson
|
|
||||||
- aeson
|
|
||||||
- QuickCheck
|
|
||||||
- quickcheck-instances
|
|
||||||
- scotty
|
|
||||||
- http-conduit
|
|
||||||
- wai-extra
|
|
||||||
- http-types
|
|
||||||
- time
|
|
||||||
- securemem
|
|
||||||
- bytestring
|
|
||||||
- regex-base
|
|
||||||
- regex-compat
|
|
||||||
- array
|
|
||||||
- random
|
|
||||||
- vector
|
|
||||||
- wai-cors
|
|
||||||
- warp-tls
|
|
||||||
- hexstring
|
|
||||||
- configurator
|
|
||||||
- scientific
|
|
||||||
- jwt
|
|
||||||
- containers
|
|
||||||
- base64-bytestring
|
|
||||||
- wai
|
|
||||||
- blake3
|
|
||||||
- memory
|
|
||||||
- ghc-prim
|
|
||||||
- network
|
|
||||||
- crypto-rng
|
|
||||||
- megaparsec
|
|
||||||
- uuid
|
|
||||||
|
|
||||||
executables:
|
|
||||||
zgo-backend-exe:
|
|
||||||
main: Server.hs
|
|
||||||
source-dirs: app
|
|
||||||
ghc-options:
|
|
||||||
- -main-is Server
|
|
||||||
- -threaded
|
|
||||||
- -rtsopts
|
|
||||||
- -with-rtsopts=-N
|
|
||||||
- -Wall
|
|
||||||
dependencies:
|
|
||||||
- zgo-backend
|
|
||||||
- base
|
|
||||||
- scotty
|
|
||||||
- wai-extra
|
|
||||||
- securemem
|
|
||||||
- text
|
|
||||||
- aeson
|
|
||||||
- mongoDB
|
|
||||||
- http-types
|
|
||||||
- http-conduit
|
|
||||||
- time
|
|
||||||
- bytestring
|
|
||||||
- configurator
|
|
||||||
- warp-tls
|
|
||||||
- warp
|
|
||||||
- megaparsec
|
|
||||||
zgo-token-refresh:
|
|
||||||
main: TokenRefresh.hs
|
|
||||||
source-dirs: app
|
|
||||||
ghc-options:
|
|
||||||
- -main-is TokenRefresh
|
|
||||||
- -threaded
|
|
||||||
- -rtsopts
|
|
||||||
- -with-rtsopts=-N
|
|
||||||
- -Wall
|
|
||||||
dependencies:
|
|
||||||
- base
|
|
||||||
- zgo-backend
|
|
||||||
- base
|
|
||||||
- scotty
|
|
||||||
- wai-extra
|
|
||||||
- securemem
|
|
||||||
- text
|
|
||||||
- aeson
|
|
||||||
- mongoDB
|
|
||||||
- http-types
|
|
||||||
- http-conduit
|
|
||||||
- time
|
|
||||||
- bytestring
|
|
||||||
- configurator
|
|
||||||
- warp-tls
|
|
||||||
- warp
|
|
||||||
- megaparsec
|
|
||||||
zgo-tasks:
|
|
||||||
main: Tasks.hs
|
|
||||||
source-dirs: app
|
|
||||||
ghc-options:
|
|
||||||
- -main-is Tasks
|
|
||||||
- -threaded
|
|
||||||
- -rtsopts
|
|
||||||
- -with-rtsopts=-N
|
|
||||||
- -Wall
|
|
||||||
dependencies:
|
|
||||||
- base
|
|
||||||
- mongoDB
|
|
||||||
- zgo-backend
|
|
||||||
- scotty
|
|
||||||
- warp-tls
|
|
||||||
- warp
|
|
||||||
- time
|
|
||||||
- megaparsec
|
|
||||||
|
|
||||||
tests:
|
|
||||||
zgo-backend-test:
|
|
||||||
main: Spec.hs
|
|
||||||
source-dirs: test
|
|
||||||
ghc-options:
|
|
||||||
- -threaded
|
|
||||||
- -rtsopts
|
|
||||||
- -with-rtsopts=-N
|
|
||||||
- -main-is Spec
|
|
||||||
dependencies:
|
|
||||||
- zgo-backend
|
|
||||||
- hspec
|
|
||||||
- QuickCheck
|
|
||||||
- text
|
|
||||||
- aeson
|
|
||||||
- http-conduit
|
|
||||||
- http-types
|
|
||||||
- hspec-expectations-json
|
|
||||||
- bytestring
|
|
||||||
- mongoDB
|
|
||||||
- hspec-wai
|
|
||||||
- securemem
|
|
||||||
- time
|
|
||||||
- configurator
|
|
||||||
- scotty
|
|
|
@ -26,6 +26,7 @@ data Config =
|
||||||
, c_smtpPort :: Integer
|
, c_smtpPort :: Integer
|
||||||
, c_smtpUser :: String
|
, c_smtpUser :: String
|
||||||
, c_smtpPwd :: String
|
, c_smtpPwd :: String
|
||||||
|
, c_confirmations :: Integer
|
||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
@ -48,6 +49,7 @@ loadZGoConfig path = do
|
||||||
mailPort <- require config "smtpPort"
|
mailPort <- require config "smtpPort"
|
||||||
mailUser <- require config "smtpUser"
|
mailUser <- require config "smtpUser"
|
||||||
mailPwd <- require config "smtpPwd"
|
mailPwd <- require config "smtpPwd"
|
||||||
|
conf <- require config "confirmations"
|
||||||
return $
|
return $
|
||||||
Config
|
Config
|
||||||
dbHost
|
dbHost
|
||||||
|
@ -66,3 +68,4 @@ loadZGoConfig path = do
|
||||||
mailPort
|
mailPort
|
||||||
mailUser
|
mailUser
|
||||||
mailPwd
|
mailPwd
|
||||||
|
conf
|
||||||
|
|
|
@ -12,6 +12,7 @@ import Data.Time.Clock
|
||||||
import Database.MongoDB
|
import Database.MongoDB
|
||||||
import GHC.Generics
|
import GHC.Generics
|
||||||
import Test.QuickCheck
|
import Test.QuickCheck
|
||||||
|
import User
|
||||||
|
|
||||||
-- | Type to represent a ZGo item
|
-- | Type to represent a ZGo item
|
||||||
data Item =
|
data Item =
|
||||||
|
@ -87,6 +88,9 @@ findItems :: T.Text -> Action IO [Document]
|
||||||
findItems a =
|
findItems a =
|
||||||
rest =<< find (select ["owner" =: a] "items") {sort = ["name" =: (1 :: Int)]}
|
rest =<< find (select ["owner" =: a] "items") {sort = ["name" =: (1 :: Int)]}
|
||||||
|
|
||||||
|
findItemById :: String -> Action IO (Maybe Document)
|
||||||
|
findItemById i = findOne (select ["_id" =: (read i :: ObjectId)] "items")
|
||||||
|
|
||||||
upsertItem :: Item -> Action IO ()
|
upsertItem :: Item -> Action IO ()
|
||||||
upsertItem i = do
|
upsertItem i = do
|
||||||
let item = val i
|
let item = val i
|
||||||
|
|
149
src/Order.hs
149
src/Order.hs
|
@ -12,28 +12,31 @@ import Data.Time.Clock
|
||||||
import Database.MongoDB
|
import Database.MongoDB
|
||||||
import GHC.Generics
|
import GHC.Generics
|
||||||
import Test.QuickCheck
|
import Test.QuickCheck
|
||||||
|
import WooCommerce (WooToken(w_id))
|
||||||
|
|
||||||
-- | Type to represent a ZGo order
|
-- | Type to represent a ZGo order
|
||||||
data ZGoOrder =
|
data ZGoOrder = ZGoOrder
|
||||||
ZGoOrder
|
{ q_id :: Maybe ObjectId
|
||||||
{ q_id :: Maybe ObjectId
|
, qaddress :: T.Text
|
||||||
, qaddress :: T.Text
|
, qsession :: T.Text
|
||||||
, qsession :: T.Text
|
, qtimestamp :: UTCTime
|
||||||
, qtimestamp :: UTCTime
|
, qclosed :: Bool
|
||||||
, qclosed :: Bool
|
, qcurrency :: T.Text
|
||||||
, qcurrency :: T.Text
|
, qprice :: Double
|
||||||
, qprice :: Double
|
, qtotal :: Double
|
||||||
, qtotal :: Double
|
, qtotalZec :: Double
|
||||||
, qtotalZec :: Double
|
, qlines :: [LineItem]
|
||||||
, qlines :: [LineItem]
|
, qpaid :: Bool
|
||||||
, qpaid :: Bool
|
, qexternalInvoice :: T.Text
|
||||||
, qexternalInvoice :: T.Text
|
, qshortCode :: T.Text
|
||||||
, qshortCode :: T.Text
|
, qtoken :: T.Text
|
||||||
}
|
, qtax :: Double
|
||||||
deriving (Eq, Show, Generic)
|
, qvat :: Double
|
||||||
|
, qtip :: Double
|
||||||
|
} deriving (Eq, Show, Generic)
|
||||||
|
|
||||||
instance ToJSON ZGoOrder where
|
instance ToJSON ZGoOrder where
|
||||||
toJSON (ZGoOrder i a s ts c cur p t tZ l paid eI sC) =
|
toJSON (ZGoOrder i a s ts c cur p t tZ l paid eI sC tk qT qV tip) =
|
||||||
case i of
|
case i of
|
||||||
Just oid ->
|
Just oid ->
|
||||||
object
|
object
|
||||||
|
@ -50,6 +53,10 @@ instance ToJSON ZGoOrder where
|
||||||
, "paid" .= paid
|
, "paid" .= paid
|
||||||
, "externalInvoice" .= eI
|
, "externalInvoice" .= eI
|
||||||
, "shortCode" .= sC
|
, "shortCode" .= sC
|
||||||
|
, "token" .= tk
|
||||||
|
, "taxAmount" .= qT
|
||||||
|
, "vatAmount" .= qV
|
||||||
|
, "tipAmount" .= tip
|
||||||
]
|
]
|
||||||
Nothing ->
|
Nothing ->
|
||||||
object
|
object
|
||||||
|
@ -66,6 +73,10 @@ instance ToJSON ZGoOrder where
|
||||||
, "paid" .= paid
|
, "paid" .= paid
|
||||||
, "externalInvoice" .= eI
|
, "externalInvoice" .= eI
|
||||||
, "shortCode" .= sC
|
, "shortCode" .= sC
|
||||||
|
, "token" .= tk
|
||||||
|
, "taxAmount" .= qT
|
||||||
|
, "vatAmount" .= qV
|
||||||
|
, "tipAmount" .= tip
|
||||||
]
|
]
|
||||||
|
|
||||||
instance FromJSON ZGoOrder where
|
instance FromJSON ZGoOrder where
|
||||||
|
@ -84,10 +95,14 @@ instance FromJSON ZGoOrder where
|
||||||
pd <- obj .: "paid"
|
pd <- obj .: "paid"
|
||||||
eI <- obj .: "externalInvoice"
|
eI <- obj .: "externalInvoice"
|
||||||
sC <- obj .: "shortCode"
|
sC <- obj .: "shortCode"
|
||||||
|
tk <- obj .: "token"
|
||||||
|
qT <- obj .: "taxAmount"
|
||||||
|
qV <- obj .: "vatAmount"
|
||||||
|
tip <- obj .: "tipAmount"
|
||||||
pure $
|
pure $
|
||||||
ZGoOrder
|
ZGoOrder
|
||||||
(if not (null i)
|
(if not (null i)
|
||||||
then Just (read i)
|
then Just (read i :: ObjectId)
|
||||||
else Nothing)
|
else Nothing)
|
||||||
a
|
a
|
||||||
s
|
s
|
||||||
|
@ -101,9 +116,13 @@ instance FromJSON ZGoOrder where
|
||||||
pd
|
pd
|
||||||
eI
|
eI
|
||||||
sC
|
sC
|
||||||
|
tk
|
||||||
|
qT
|
||||||
|
qV
|
||||||
|
tip
|
||||||
|
|
||||||
instance Val ZGoOrder where
|
instance Val ZGoOrder where
|
||||||
val (ZGoOrder i a s ts c cur p t tZ l pd eI sC) =
|
val (ZGoOrder i a s ts c cur p t tZ l pd eI sC tk qT qV tip) =
|
||||||
if isJust i
|
if isJust i
|
||||||
then Doc
|
then Doc
|
||||||
[ "_id" =: i
|
[ "_id" =: i
|
||||||
|
@ -119,6 +138,10 @@ instance Val ZGoOrder where
|
||||||
, "paid" =: pd
|
, "paid" =: pd
|
||||||
, "externalInvoice" =: eI
|
, "externalInvoice" =: eI
|
||||||
, "shortCode" =: sC
|
, "shortCode" =: sC
|
||||||
|
, "token" =: tk
|
||||||
|
, "taxAmount" =: qT
|
||||||
|
, "vatAmount" =: qV
|
||||||
|
, "tipAmount" =: tip
|
||||||
]
|
]
|
||||||
else Doc
|
else Doc
|
||||||
[ "address" =: a
|
[ "address" =: a
|
||||||
|
@ -133,6 +156,10 @@ instance Val ZGoOrder where
|
||||||
, "paid" =: pd
|
, "paid" =: pd
|
||||||
, "externalInvoice" =: eI
|
, "externalInvoice" =: eI
|
||||||
, "shortCode" =: sC
|
, "shortCode" =: sC
|
||||||
|
, "token" =: tk
|
||||||
|
, "taxAmount" =: qT
|
||||||
|
, "vatAmount" =: qV
|
||||||
|
, "tipAmount" =: tip
|
||||||
]
|
]
|
||||||
cast' (Doc d) = do
|
cast' (Doc d) = do
|
||||||
i <- B.lookup "_id" d
|
i <- B.lookup "_id" d
|
||||||
|
@ -148,17 +175,19 @@ instance Val ZGoOrder where
|
||||||
pd <- B.lookup "paid" d
|
pd <- B.lookup "paid" d
|
||||||
eI <- B.lookup "externalInvoice" d
|
eI <- B.lookup "externalInvoice" d
|
||||||
sC <- B.lookup "shortCode" d
|
sC <- B.lookup "shortCode" d
|
||||||
Just (ZGoOrder i a s ts c cur p t tZ l pd eI sC)
|
tk <- B.lookup "token" d
|
||||||
|
qT <- B.lookup "taxAmount" d
|
||||||
|
qV <- B.lookup "vatAmount" d
|
||||||
|
tip <- B.lookup "tipAmount" d
|
||||||
|
Just (ZGoOrder i a s ts c cur p t tZ l pd eI sC tk qT qV tip)
|
||||||
cast' _ = Nothing
|
cast' _ = Nothing
|
||||||
|
|
||||||
-- Type to represent an order line item
|
-- Type to represent an order line item
|
||||||
data LineItem =
|
data LineItem = LineItem
|
||||||
LineItem
|
{ lqty :: Double
|
||||||
{ lqty :: Double
|
, lname :: T.Text
|
||||||
, lname :: T.Text
|
, lcost :: Double
|
||||||
, lcost :: Double
|
} deriving (Eq, Show)
|
||||||
}
|
|
||||||
deriving (Eq, Show)
|
|
||||||
|
|
||||||
instance ToJSON LineItem where
|
instance ToJSON LineItem where
|
||||||
toJSON (LineItem q n c) = object ["qty" .= q, "name" .= n, "cost" .= c]
|
toJSON (LineItem q n c) = object ["qty" .= q, "name" .= n, "cost" .= c]
|
||||||
|
@ -181,33 +210,40 @@ instance Val LineItem where
|
||||||
cast' _ = Nothing
|
cast' _ = Nothing
|
||||||
|
|
||||||
-- Database actions
|
-- Database actions
|
||||||
upsertOrder :: ZGoOrder -> Action IO ()
|
upsertOrder :: ZGoOrder -> Double -> Double -> Action IO ()
|
||||||
upsertOrder o = do
|
upsertOrder o taxRate vatRate = do
|
||||||
let order = val $ updateOrderTotals o
|
let order = val $ updateOrderTotals o taxRate vatRate
|
||||||
case order of
|
case order of
|
||||||
Doc d ->
|
Doc d ->
|
||||||
if isJust (q_id o)
|
if isJust (q_id o)
|
||||||
then upsert (select ["_id" =: q_id o] "orders") d
|
then upsert (select ["_id" =: q_id o] "orders") d
|
||||||
else insert_ "orders" d
|
else insert_ "orders" d
|
||||||
_ -> return ()
|
_ -> return ()
|
||||||
|
|
||||||
insertWooOrder :: ZGoOrder -> Action IO Database.MongoDB.Value
|
insertWooOrder :: ZGoOrder -> Action IO Database.MongoDB.Value
|
||||||
insertWooOrder o = do
|
insertWooOrder o = do
|
||||||
let order = val $ updateOrderTotals o
|
let order = val $ updateOrderTotals o 0 0
|
||||||
case order of
|
case order of
|
||||||
Doc d -> insert "orders" d
|
Doc d -> insert "orders" d
|
||||||
_ -> fail "Couldn't parse order"
|
_ -> fail "Couldn't parse order"
|
||||||
|
|
||||||
upsertXeroOrder :: ZGoOrder -> Action IO ()
|
upsertXeroOrder :: ZGoOrder -> Action IO ()
|
||||||
upsertXeroOrder o = do
|
upsertXeroOrder o = do
|
||||||
let order = val $ updateOrderTotals o
|
let order = val $ updateOrderTotals o 0 0
|
||||||
case order of
|
case order of
|
||||||
Doc d -> upsert (select ["externalInvoice" =: qexternalInvoice o, "shortCode" =: qshortCode o] "orders") d
|
Doc d ->
|
||||||
|
upsert
|
||||||
|
(select
|
||||||
|
[ "externalInvoice" =: qexternalInvoice o
|
||||||
|
, "shortCode" =: qshortCode o
|
||||||
|
]
|
||||||
|
"orders")
|
||||||
|
d
|
||||||
_ -> return ()
|
_ -> return ()
|
||||||
|
|
||||||
-- | Function to update order totals from items
|
-- | Function to update order totals from items
|
||||||
updateOrderTotals :: ZGoOrder -> ZGoOrder
|
updateOrderTotals :: ZGoOrder -> Double -> Double -> ZGoOrder
|
||||||
updateOrderTotals o =
|
updateOrderTotals o taxRate vatRate =
|
||||||
ZGoOrder
|
ZGoOrder
|
||||||
(q_id o)
|
(q_id o)
|
||||||
(qaddress o)
|
(qaddress o)
|
||||||
|
@ -216,31 +252,51 @@ updateOrderTotals o =
|
||||||
(qclosed o)
|
(qclosed o)
|
||||||
(qcurrency o)
|
(qcurrency o)
|
||||||
(qprice o)
|
(qprice o)
|
||||||
(newTotal o)
|
(newTotal o taxRate vatRate)
|
||||||
(if qprice o /= 0
|
(if qprice o /= 0
|
||||||
then roundZec (newTotal o / qprice o)
|
then roundZec (newTotal o taxRate vatRate / qprice o)
|
||||||
else 0)
|
else 0)
|
||||||
(qlines o)
|
(qlines o)
|
||||||
(qpaid o)
|
(qpaid o)
|
||||||
(qexternalInvoice o)
|
(qexternalInvoice o)
|
||||||
(qshortCode o)
|
(qshortCode o)
|
||||||
|
(qtoken o)
|
||||||
|
(updateTax o taxRate)
|
||||||
|
(updateTax o vatRate)
|
||||||
|
(qtip o)
|
||||||
where
|
where
|
||||||
newTotal :: ZGoOrder -> Double
|
updateTax :: ZGoOrder -> Double -> Double
|
||||||
newTotal x = foldr tallyItems 0 (qlines x)
|
updateTax x t = roundFiat $ itemsTotal (qlines x) * t / 100.0
|
||||||
|
itemsTotal :: [LineItem] -> Double
|
||||||
|
itemsTotal = foldr tallyItems 0
|
||||||
|
newTotal :: ZGoOrder -> Double -> Double -> Double
|
||||||
|
newTotal x tR vR =
|
||||||
|
itemsTotal (qlines x) + updateTax x tR + updateTax x vR + qtip x
|
||||||
tallyItems :: LineItem -> Double -> Double
|
tallyItems :: LineItem -> Double -> Double
|
||||||
tallyItems y z = (lqty y * lcost y) + z
|
tallyItems y z = (lqty y * lcost y) + z
|
||||||
|
|
||||||
|
setOrderToken :: T.Text -> ZGoOrder -> ZGoOrder
|
||||||
|
setOrderToken token (ZGoOrder i a s ts c cur p t tZ l pd eI sC tk qT qV tip) =
|
||||||
|
ZGoOrder i a s ts c cur p t tZ l pd eI sC token qT qV tip
|
||||||
|
|
||||||
findOrder :: T.Text -> Action IO (Maybe Document)
|
findOrder :: T.Text -> Action IO (Maybe Document)
|
||||||
findOrder s = findOne (select ["session" =: s, "closed" =: False] "orders")
|
findOrder s = findOne (select ["session" =: s, "closed" =: False] "orders")
|
||||||
|
|
||||||
findXeroOrder :: T.Text -> T.Text -> T.Text -> Action IO (Maybe Document)
|
findXeroOrder :: T.Text -> T.Text -> T.Text -> Action IO (Maybe Document)
|
||||||
findXeroOrder a i s = findOne (select ["address" =: a, "externalInvoice" =: i, "shortCode" =: s] "orders")
|
findXeroOrder a i s =
|
||||||
|
findOne
|
||||||
|
(select ["address" =: a, "externalInvoice" =: i, "shortCode" =: s] "orders")
|
||||||
|
|
||||||
findOrderById :: String -> Action IO (Maybe Document)
|
findOrderById :: String -> Action IO (Maybe Document)
|
||||||
|
findOrderById "0" = return Nothing
|
||||||
findOrderById i = findOne (select ["_id" =: (read i :: B.ObjectId)] "orders")
|
findOrderById i = findOne (select ["_id" =: (read i :: B.ObjectId)] "orders")
|
||||||
|
|
||||||
findAllOrders :: T.Text -> Action IO [Document]
|
findAllOrders :: T.Text -> Action IO [Document]
|
||||||
findAllOrders a = rest =<< find (select ["address" =: a] "orders") {sort = ["timestamp" =: (negate 1 :: Int)]}
|
findAllOrders a =
|
||||||
|
rest =<<
|
||||||
|
find
|
||||||
|
(select ["address" =: a] "orders")
|
||||||
|
{sort = ["timestamp" =: (negate 1 :: Int)]}
|
||||||
|
|
||||||
deleteOrder :: String -> Action IO ()
|
deleteOrder :: String -> Action IO ()
|
||||||
deleteOrder i = deleteOne (select ["_id" =: (read i :: B.ObjectId)] "orders")
|
deleteOrder i = deleteOne (select ["_id" =: (read i :: B.ObjectId)] "orders")
|
||||||
|
@ -255,3 +311,6 @@ markOrderPaid (i, a) = do
|
||||||
-- | Helper function to round to 8 decimal places
|
-- | Helper function to round to 8 decimal places
|
||||||
roundZec :: Double -> Double
|
roundZec :: Double -> Double
|
||||||
roundZec n = fromInteger (round $ n * (10 ^ 8)) / (10.0 ^^ 8)
|
roundZec n = fromInteger (round $ n * (10 ^ 8)) / (10.0 ^^ 8)
|
||||||
|
|
||||||
|
roundFiat :: Double -> Double
|
||||||
|
roundFiat n = fromInteger (round $ n * (10 ^ 2)) / (10.0 ^^ 2)
|
||||||
|
|
192
src/Owner.hs
192
src/Owner.hs
|
@ -14,38 +14,37 @@ import Database.MongoDB
|
||||||
import GHC.Generics
|
import GHC.Generics
|
||||||
|
|
||||||
-- | Type to represent a ZGo shop owner/business
|
-- | Type to represent a ZGo shop owner/business
|
||||||
data Owner =
|
data Owner = Owner
|
||||||
Owner
|
{ o_id :: Maybe ObjectId
|
||||||
{ o_id :: Maybe ObjectId
|
, oaddress :: T.Text
|
||||||
, oaddress :: T.Text
|
, oname :: T.Text
|
||||||
, oname :: T.Text
|
, ocurrency :: T.Text
|
||||||
, ocurrency :: T.Text
|
, otax :: Bool
|
||||||
, otax :: Bool
|
, otaxValue :: Double
|
||||||
, otaxValue :: Double
|
, ovat :: Bool
|
||||||
, ovat :: Bool
|
, ovatValue :: Double
|
||||||
, ovatValue :: Double
|
, ofirst :: T.Text
|
||||||
, ofirst :: T.Text
|
, olast :: T.Text
|
||||||
, olast :: T.Text
|
, oemail :: T.Text
|
||||||
, oemail :: T.Text
|
, ostreet :: T.Text
|
||||||
, ostreet :: T.Text
|
, ocity :: T.Text
|
||||||
, ocity :: T.Text
|
, ostate :: T.Text
|
||||||
, ostate :: T.Text
|
, opostal :: T.Text
|
||||||
, opostal :: T.Text
|
, ophone :: T.Text
|
||||||
, ophone :: T.Text
|
, owebsite :: T.Text
|
||||||
, owebsite :: T.Text
|
, ocountry :: T.Text
|
||||||
, ocountry :: T.Text
|
, opaid :: Bool
|
||||||
, opaid :: Bool
|
, ozats :: Bool
|
||||||
, ozats :: Bool
|
, oinvoices :: Bool
|
||||||
, oinvoices :: Bool
|
, oexpiration :: UTCTime
|
||||||
, oexpiration :: UTCTime
|
, opayconf :: Bool
|
||||||
, opayconf :: Bool
|
, oviewkey :: T.Text
|
||||||
, oviewkey :: T.Text
|
, ocrmToken :: T.Text
|
||||||
, ocrmToken :: T.Text
|
, otips :: Bool
|
||||||
}
|
} deriving (Eq, Show, Generic, Typeable)
|
||||||
deriving (Eq, Show, Generic, Typeable)
|
|
||||||
|
|
||||||
instance ToJSON Owner where
|
instance ToJSON Owner where
|
||||||
toJSON (Owner i a n c t tV v vV f l e s ct st p ph w co paid zats inv eTs pc vk cT) =
|
toJSON (Owner i a n c t tV v vV f l e s ct st p ph w co paid zats inv eTs pc vk cT oT) =
|
||||||
case i of
|
case i of
|
||||||
Just oid ->
|
Just oid ->
|
||||||
object
|
object
|
||||||
|
@ -74,6 +73,7 @@ instance ToJSON Owner where
|
||||||
, "payconf" .= pc
|
, "payconf" .= pc
|
||||||
, "viewkey" .= vk
|
, "viewkey" .= vk
|
||||||
, "crmToken" .= cT
|
, "crmToken" .= cT
|
||||||
|
, "tips" .= oT
|
||||||
]
|
]
|
||||||
Nothing ->
|
Nothing ->
|
||||||
object
|
object
|
||||||
|
@ -102,6 +102,7 @@ instance ToJSON Owner where
|
||||||
, "payconf" .= pc
|
, "payconf" .= pc
|
||||||
, "viewkey" .= vk
|
, "viewkey" .= vk
|
||||||
, "crmToken" .= cT
|
, "crmToken" .= cT
|
||||||
|
, "tips" .= oT
|
||||||
]
|
]
|
||||||
|
|
||||||
instance FromJSON Owner where
|
instance FromJSON Owner where
|
||||||
|
@ -132,6 +133,7 @@ instance FromJSON Owner where
|
||||||
pc <- obj .:? "payconf"
|
pc <- obj .:? "payconf"
|
||||||
vk <- obj .:? "viewkey"
|
vk <- obj .:? "viewkey"
|
||||||
cT <- obj .:? "crmToken"
|
cT <- obj .:? "crmToken"
|
||||||
|
oT <- obj .:? "tips"
|
||||||
pure $
|
pure $
|
||||||
Owner
|
Owner
|
||||||
(if not (null i)
|
(if not (null i)
|
||||||
|
@ -161,6 +163,7 @@ instance FromJSON Owner where
|
||||||
(fromMaybe False pc)
|
(fromMaybe False pc)
|
||||||
(fromMaybe "" vk)
|
(fromMaybe "" vk)
|
||||||
(fromMaybe "" cT)
|
(fromMaybe "" cT)
|
||||||
|
(fromMaybe False oT)
|
||||||
|
|
||||||
instance Val Owner where
|
instance Val Owner where
|
||||||
cast' (Doc d) = do
|
cast' (Doc d) = do
|
||||||
|
@ -189,6 +192,7 @@ instance Val Owner where
|
||||||
pc <- B.lookup "payconf" d
|
pc <- B.lookup "payconf" d
|
||||||
vk <- B.lookup "viewKey" d
|
vk <- B.lookup "viewKey" d
|
||||||
cT <- B.lookup "crmToken" d
|
cT <- B.lookup "crmToken" d
|
||||||
|
oT <- B.lookup "tips" d
|
||||||
Just
|
Just
|
||||||
(Owner
|
(Owner
|
||||||
i
|
i
|
||||||
|
@ -215,9 +219,10 @@ instance Val Owner where
|
||||||
ets
|
ets
|
||||||
pc
|
pc
|
||||||
vk
|
vk
|
||||||
cT)
|
cT
|
||||||
|
oT)
|
||||||
cast' _ = Nothing
|
cast' _ = Nothing
|
||||||
val (Owner i a n c t tV v vV f l e s ct st p ph w co paid zats inv ets pc vk cT) =
|
val (Owner i a n c t tV v vV f l e s ct st p ph w co paid zats inv ets pc vk cT oT) =
|
||||||
case i of
|
case i of
|
||||||
Just oid ->
|
Just oid ->
|
||||||
Doc
|
Doc
|
||||||
|
@ -246,6 +251,7 @@ instance Val Owner where
|
||||||
, "payconf" =: pc
|
, "payconf" =: pc
|
||||||
, "viewKey" =: vk
|
, "viewKey" =: vk
|
||||||
, "crmToken" =: cT
|
, "crmToken" =: cT
|
||||||
|
, "tips" =: oT
|
||||||
]
|
]
|
||||||
Nothing ->
|
Nothing ->
|
||||||
Doc
|
Doc
|
||||||
|
@ -273,24 +279,23 @@ instance Val Owner where
|
||||||
, "payconf" =: pc
|
, "payconf" =: pc
|
||||||
, "viewKey" =: vk
|
, "viewKey" =: vk
|
||||||
, "crmToken" =: cT
|
, "crmToken" =: cT
|
||||||
|
, "tips" =: oT
|
||||||
]
|
]
|
||||||
|
|
||||||
-- | Type to represent informational data for Owners from UI
|
-- | Type to represent informational data for Owners from UI
|
||||||
data OwnerData =
|
data OwnerData = OwnerData
|
||||||
OwnerData
|
{ od_first :: T.Text
|
||||||
{ od_first :: T.Text
|
, od_last :: T.Text
|
||||||
, od_last :: T.Text
|
, od_name :: T.Text
|
||||||
, od_name :: T.Text
|
, od_street :: T.Text
|
||||||
, od_street :: T.Text
|
, od_city :: T.Text
|
||||||
, od_city :: T.Text
|
, od_state :: T.Text
|
||||||
, od_state :: T.Text
|
, od_postal :: T.Text
|
||||||
, od_postal :: T.Text
|
, od_country :: T.Text
|
||||||
, od_country :: T.Text
|
, od_email :: T.Text
|
||||||
, od_email :: T.Text
|
, od_website :: T.Text
|
||||||
, od_website :: T.Text
|
, od_phone :: T.Text
|
||||||
, od_phone :: T.Text
|
} deriving (Eq, Show, Generic)
|
||||||
}
|
|
||||||
deriving (Eq, Show, Generic)
|
|
||||||
|
|
||||||
instance FromJSON OwnerData where
|
instance FromJSON OwnerData where
|
||||||
parseJSON =
|
parseJSON =
|
||||||
|
@ -308,25 +313,24 @@ instance FromJSON OwnerData where
|
||||||
ph <- obj .: "phone"
|
ph <- obj .: "phone"
|
||||||
pure $ OwnerData f l n s c st p co e w ph
|
pure $ OwnerData f l n s c st p co e w ph
|
||||||
|
|
||||||
data OwnerSettings =
|
data OwnerSettings = OwnerSettings
|
||||||
OwnerSettings
|
{ os_id :: Maybe ObjectId
|
||||||
{ os_id :: Maybe ObjectId
|
, os_address :: T.Text
|
||||||
, os_address :: T.Text
|
, os_name :: T.Text
|
||||||
, os_name :: T.Text
|
, os_currency :: T.Text
|
||||||
, os_currency :: T.Text
|
, os_tax :: Bool
|
||||||
, os_tax :: Bool
|
, os_taxValue :: Double
|
||||||
, os_taxValue :: Double
|
, os_vat :: Bool
|
||||||
, os_vat :: Bool
|
, os_vatValue :: Double
|
||||||
, os_vatValue :: Double
|
, os_paid :: Bool
|
||||||
, os_paid :: Bool
|
, os_zats :: Bool
|
||||||
, os_zats :: Bool
|
, os_invoices :: Bool
|
||||||
, os_invoices :: Bool
|
, os_expiration :: UTCTime
|
||||||
, os_expiration :: UTCTime
|
, os_payconf :: Bool
|
||||||
, os_payconf :: Bool
|
, os_crmToken :: T.Text
|
||||||
, os_crmToken :: T.Text
|
, os_viewKey :: T.Text
|
||||||
, os_viewKey :: T.Text
|
, os_tips :: Bool
|
||||||
}
|
} deriving (Eq, Show, Generic)
|
||||||
deriving (Eq, Show, Generic)
|
|
||||||
|
|
||||||
instance FromJSON OwnerSettings where
|
instance FromJSON OwnerSettings where
|
||||||
parseJSON =
|
parseJSON =
|
||||||
|
@ -346,11 +350,28 @@ instance FromJSON OwnerSettings where
|
||||||
pc <- obj .: "payconf"
|
pc <- obj .: "payconf"
|
||||||
cT <- obj .: "crmToken"
|
cT <- obj .: "crmToken"
|
||||||
vK <- obj .: "viewkey"
|
vK <- obj .: "viewkey"
|
||||||
|
oT <- obj .: "tips"
|
||||||
pure $
|
pure $
|
||||||
OwnerSettings ((Just . read) =<< i) a n c t tV v vV p z inv e pc cT vK
|
OwnerSettings
|
||||||
|
((Just . read) =<< i)
|
||||||
|
a
|
||||||
|
n
|
||||||
|
c
|
||||||
|
t
|
||||||
|
tV
|
||||||
|
v
|
||||||
|
vV
|
||||||
|
p
|
||||||
|
z
|
||||||
|
inv
|
||||||
|
e
|
||||||
|
pc
|
||||||
|
cT
|
||||||
|
vK
|
||||||
|
oT
|
||||||
|
|
||||||
instance ToJSON OwnerSettings where
|
instance ToJSON OwnerSettings where
|
||||||
toJSON (OwnerSettings i a n c t tV v vV p z inv e pc cT vK) =
|
toJSON (OwnerSettings i a n c t tV v vV p z inv e pc cT vK oT) =
|
||||||
object
|
object
|
||||||
[ "_id" .= maybe "" show i
|
[ "_id" .= maybe "" show i
|
||||||
, "address" .= a
|
, "address" .= a
|
||||||
|
@ -366,8 +387,13 @@ instance ToJSON OwnerSettings where
|
||||||
, "expiration" .= e
|
, "expiration" .= e
|
||||||
, "payconf" .= pc
|
, "payconf" .= pc
|
||||||
, "crmToken" .= cT
|
, "crmToken" .= cT
|
||||||
, "viewkey" .= (T.take 8 vK <> "...." <> T.takeEnd 8 vK)
|
, "viewkey" .= keyObfuscate vK
|
||||||
|
, "tips" .= oT
|
||||||
]
|
]
|
||||||
|
where
|
||||||
|
keyObfuscate s
|
||||||
|
| s == "" = ""
|
||||||
|
| otherwise = T.take 8 s <> "...." <> T.takeEnd 8 s
|
||||||
|
|
||||||
-- Helper Functions
|
-- Helper Functions
|
||||||
getOwnerSettings :: Owner -> OwnerSettings
|
getOwnerSettings :: Owner -> OwnerSettings
|
||||||
|
@ -388,6 +414,7 @@ getOwnerSettings o =
|
||||||
(opayconf o)
|
(opayconf o)
|
||||||
(ocrmToken o)
|
(ocrmToken o)
|
||||||
(oviewkey o)
|
(oviewkey o)
|
||||||
|
(otips o)
|
||||||
|
|
||||||
-- Database actions
|
-- Database actions
|
||||||
-- | Function to upsert an Owner
|
-- | Function to upsert an Owner
|
||||||
|
@ -407,6 +434,10 @@ findOwnerById :: T.Text -> Action IO (Maybe Document)
|
||||||
findOwnerById i =
|
findOwnerById i =
|
||||||
findOne (select ["_id" =: (read (T.unpack i) :: ObjectId)] "owners")
|
findOne (select ["_id" =: (read (T.unpack i) :: ObjectId)] "owners")
|
||||||
|
|
||||||
|
findActiveOwners :: Action IO [Document]
|
||||||
|
findActiveOwners =
|
||||||
|
rest =<< find (select ["paid" =: True, "payconf" =: True] "owners")
|
||||||
|
|
||||||
-- | Function to find Owners about to expire
|
-- | Function to find Owners about to expire
|
||||||
findExpiringOwners :: UTCTime -> Action IO [Document]
|
findExpiringOwners :: UTCTime -> Action IO [Document]
|
||||||
findExpiringOwners now =
|
findExpiringOwners now =
|
||||||
|
@ -416,6 +447,10 @@ findExpiringOwners now =
|
||||||
["paid" =: True, "expiration" =: ["$lte" =: addUTCTime 172800 now]]
|
["paid" =: True, "expiration" =: ["$lte" =: addUTCTime 172800 now]]
|
||||||
"owners")
|
"owners")
|
||||||
|
|
||||||
|
findWithKeys :: Action IO [Document]
|
||||||
|
findWithKeys =
|
||||||
|
rest =<< find (select ["paid" =: True, "payconf" =: True] "owners")
|
||||||
|
|
||||||
removePro :: T.Text -> Action IO ()
|
removePro :: T.Text -> Action IO ()
|
||||||
removePro o =
|
removePro o =
|
||||||
modify (select ["address" =: o] "owners") ["$set" =: ["invoices" =: False]]
|
modify (select ["address" =: o] "owners") ["$set" =: ["invoices" =: False]]
|
||||||
|
@ -434,18 +469,21 @@ updateOwnerSettings os =
|
||||||
, "zats" =: os_zats os
|
, "zats" =: os_zats os
|
||||||
, "payconf" =: os_payconf os
|
, "payconf" =: os_payconf os
|
||||||
, "crmToken" =: os_crmToken os
|
, "crmToken" =: os_crmToken os
|
||||||
|
, "tips" =: os_tips os
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
|
|
||||||
|
upsertViewingKey :: Owner -> String -> Action IO ()
|
||||||
|
upsertViewingKey o vk =
|
||||||
|
modify (select ["_id" =: o_id o] "owners") ["$set" =: ["viewKey" =: vk]]
|
||||||
|
|
||||||
-- | Type for a pro session
|
-- | Type for a pro session
|
||||||
data ZGoProSession =
|
data ZGoProSession = ZGoProSession
|
||||||
ZGoProSession
|
{ ps_id :: Maybe ObjectId
|
||||||
{ ps_id :: Maybe ObjectId
|
, psaddress :: T.Text
|
||||||
, psaddress :: T.Text
|
, psexpiration :: UTCTime
|
||||||
, psexpiration :: UTCTime
|
, psclosed :: Bool
|
||||||
, psclosed :: Bool
|
} deriving (Eq, Show)
|
||||||
}
|
|
||||||
deriving (Eq, Show)
|
|
||||||
|
|
||||||
instance Val ZGoProSession where
|
instance Val ZGoProSession where
|
||||||
cast' (Doc d) = do
|
cast' (Doc d) = do
|
||||||
|
|
33
src/User.hs
33
src/User.hs
|
@ -69,6 +69,36 @@ instance FromJSON User where
|
||||||
""
|
""
|
||||||
v
|
v
|
||||||
|
|
||||||
|
instance Val User where
|
||||||
|
cast' (Doc d) = do
|
||||||
|
i <- B.lookup "_id" d
|
||||||
|
a <- B.lookup "address" d
|
||||||
|
s <- B.lookup "session" d
|
||||||
|
b <- B.lookup "blocktime" d
|
||||||
|
p <- B.lookup "pin" d
|
||||||
|
v <- B.lookup "validated" d
|
||||||
|
Just $ User i a s b p v
|
||||||
|
cast' _ = Nothing
|
||||||
|
val (User i a s b p v) =
|
||||||
|
case i of
|
||||||
|
Just oid ->
|
||||||
|
Doc
|
||||||
|
[ "_id" =: oid
|
||||||
|
, "address" =: a
|
||||||
|
, "session" =: s
|
||||||
|
, "blocktime" =: b
|
||||||
|
, "pin" =: p
|
||||||
|
, "validated" =: v
|
||||||
|
]
|
||||||
|
Nothing ->
|
||||||
|
Doc
|
||||||
|
[ "address" =: a
|
||||||
|
, "session" =: s
|
||||||
|
, "blocktime" =: b
|
||||||
|
, "pin" =: p
|
||||||
|
, "validated" =: v
|
||||||
|
]
|
||||||
|
|
||||||
parseUserBson :: B.Document -> Maybe User
|
parseUserBson :: B.Document -> Maybe User
|
||||||
parseUserBson d = do
|
parseUserBson d = do
|
||||||
i <- B.lookup "_id" d
|
i <- B.lookup "_id" d
|
||||||
|
@ -84,6 +114,9 @@ parseUserBson d = do
|
||||||
findUser :: T.Text -> Action IO (Maybe Document)
|
findUser :: T.Text -> Action IO (Maybe Document)
|
||||||
findUser s = findOne (select ["session" =: s] "users")
|
findUser s = findOne (select ["session" =: s] "users")
|
||||||
|
|
||||||
|
findUserById :: String -> Action IO (Maybe Document)
|
||||||
|
findUserById i = findOne (select ["_id" =: (read i :: B.ObjectId)] "users")
|
||||||
|
|
||||||
-- | Function to delete user by ID
|
-- | Function to delete user by ID
|
||||||
deleteUser :: String -> Action IO ()
|
deleteUser :: String -> Action IO ()
|
||||||
deleteUser i = deleteOne (select ["_id" =: (read i :: B.ObjectId)] "users")
|
deleteUser i = deleteOne (select ["_id" =: (read i :: B.ObjectId)] "users")
|
||||||
|
|
|
@ -28,6 +28,29 @@ data WooToken =
|
||||||
}
|
}
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
|
instance FromJSON WooToken where
|
||||||
|
parseJSON =
|
||||||
|
withObject "WooToken" $ \obj -> do
|
||||||
|
i <- obj .:? "_id"
|
||||||
|
o <- obj .: "ownerid"
|
||||||
|
t <- obj .: "token"
|
||||||
|
u <- obj .: "siteurl"
|
||||||
|
pure $ WooToken (read <$> i) (read o) t u
|
||||||
|
|
||||||
|
instance ToJSON WooToken where
|
||||||
|
toJSON (WooToken i o t u) =
|
||||||
|
case i of
|
||||||
|
Just oid ->
|
||||||
|
object
|
||||||
|
["_id" .= show oid, "ownerid" .= show o, "token" .= t, "siteurl" .= u]
|
||||||
|
Nothing ->
|
||||||
|
object
|
||||||
|
[ "_id" .= ("" :: String)
|
||||||
|
, "ownerid" .= show o
|
||||||
|
, "token" .= t
|
||||||
|
, "siteurl" .= u
|
||||||
|
]
|
||||||
|
|
||||||
instance Val WooToken where
|
instance Val WooToken where
|
||||||
val (WooToken i o t u) =
|
val (WooToken i o t u) =
|
||||||
if isJust i
|
if isJust i
|
||||||
|
@ -47,8 +70,11 @@ instance Val WooToken where
|
||||||
cast' _ = Nothing
|
cast' _ = Nothing
|
||||||
|
|
||||||
-- Database actions
|
-- Database actions
|
||||||
findWooToken :: ObjectId -> Action IO (Maybe Document)
|
findWooToken :: Maybe ObjectId -> Action IO (Maybe Document)
|
||||||
findWooToken oid = findOne (select ["owner" =: oid] "wootokens")
|
findWooToken oid =
|
||||||
|
case oid of
|
||||||
|
Nothing -> return Nothing
|
||||||
|
Just o -> findOne (select ["owner" =: o] "wootokens")
|
||||||
|
|
||||||
addUrl :: WooToken -> T.Text -> Action IO ()
|
addUrl :: WooToken -> T.Text -> Action IO ()
|
||||||
addUrl t u =
|
addUrl t u =
|
||||||
|
@ -63,8 +89,9 @@ payWooOrder ::
|
||||||
-> BS.ByteString -- Total ZEC for order
|
-> BS.ByteString -- Total ZEC for order
|
||||||
-> IO ()
|
-> IO ()
|
||||||
payWooOrder u i o t p z = do
|
payWooOrder u i o t p z = do
|
||||||
wooReq <- parseRequest $ u ++ "/wc-api/zpmtcallback"
|
wooReq <- parseRequest u
|
||||||
let req =
|
let req =
|
||||||
|
setRequestPath "/wp-json/wc/v3/zgocallback" $
|
||||||
setRequestQueryString
|
setRequestQueryString
|
||||||
[ ("token", Just t)
|
[ ("token", Just t)
|
||||||
, ("orderid", Just o)
|
, ("orderid", Just o)
|
||||||
|
@ -77,23 +104,15 @@ payWooOrder u i o t p z = do
|
||||||
res <- httpLBS req
|
res <- httpLBS req
|
||||||
if getResponseStatus res == ok200
|
if getResponseStatus res == ok200
|
||||||
then return ()
|
then return ()
|
||||||
else error "Failed to report payment to WooCommerce"
|
else do
|
||||||
|
print $ getResponseStatus res
|
||||||
|
error "Failed to report payment to WooCommerce"
|
||||||
|
|
||||||
generateWooToken :: Owner -> Action IO ()
|
generateWooToken :: Owner -> String -> Action IO ()
|
||||||
generateWooToken o =
|
generateWooToken o s =
|
||||||
case o_id o of
|
case o_id o of
|
||||||
Just ownerid -> do
|
Just ownerid -> do
|
||||||
let tokenHash =
|
let wooToken = val $ WooToken Nothing ownerid (T.pack s) Nothing
|
||||||
BLK.hash
|
|
||||||
[ BA.pack . BS.unpack . C.pack . T.unpack $ oname o <> oaddress o :: BA.Bytes
|
|
||||||
]
|
|
||||||
let wooToken =
|
|
||||||
val $
|
|
||||||
WooToken
|
|
||||||
Nothing
|
|
||||||
ownerid
|
|
||||||
(T.pack . show $ (tokenHash :: BLK.Digest BLK.DEFAULT_DIGEST_LEN))
|
|
||||||
Nothing
|
|
||||||
case wooToken of
|
case wooToken of
|
||||||
Doc wT -> insert_ "wootokens" wT
|
Doc wT -> insert_ "wootokens" wT
|
||||||
_ -> error "Couldn't create the WooCommerce token"
|
_ -> error "Couldn't create the WooCommerce token"
|
||||||
|
|
21
src/Xero.hs
21
src/Xero.hs
|
@ -171,6 +171,26 @@ instance FromJSON XeroTenant where
|
||||||
--u <- obj .: "updatedDateUtc"
|
--u <- obj .: "updatedDateUtc"
|
||||||
pure $ XeroTenant i aei tI tT tN
|
pure $ XeroTenant i aei tI tT tN
|
||||||
|
|
||||||
|
data XeroInvoiceRequest =
|
||||||
|
XeroInvoiceRequest
|
||||||
|
{ xr_owner :: T.Text
|
||||||
|
, xr_invNo :: T.Text
|
||||||
|
, xr_amount :: Double
|
||||||
|
, xr_currency :: T.Text
|
||||||
|
, xr_shortCode :: T.Text
|
||||||
|
}
|
||||||
|
deriving (Show, Eq)
|
||||||
|
|
||||||
|
instance FromJSON XeroInvoiceRequest where
|
||||||
|
parseJSON =
|
||||||
|
withObject "XeroInvoiceRequest" $ \obj -> do
|
||||||
|
o <- obj .: "ownerId"
|
||||||
|
i <- obj .: "invoice"
|
||||||
|
a <- obj .: "amount"
|
||||||
|
c <- obj .: "currency"
|
||||||
|
s <- obj .: "shortcode"
|
||||||
|
pure $ XeroInvoiceRequest o i a c s
|
||||||
|
|
||||||
data XeroInvoice =
|
data XeroInvoice =
|
||||||
XeroInvoice
|
XeroInvoice
|
||||||
{ xi_id :: Maybe ObjectId
|
{ xi_id :: Maybe ObjectId
|
||||||
|
@ -443,5 +463,6 @@ payXeroInvoice pipe dbName inv address amt zec = do
|
||||||
setRequestHost "api.xero.com" $
|
setRequestHost "api.xero.com" $
|
||||||
setRequestMethod "PUT" defaultRequest
|
setRequestMethod "PUT" defaultRequest
|
||||||
res <- httpJSON req :: IO (Response Object)
|
res <- httpJSON req :: IO (Response Object)
|
||||||
|
print res
|
||||||
return ()
|
return ()
|
||||||
else error "Invalid parameters"
|
else error "Invalid parameters"
|
||||||
|
|
1633
src/ZGoBackend.hs
1633
src/ZGoBackend.hs
File diff suppressed because it is too large
Load diff
82
src/ZGoTx.hs
82
src/ZGoTx.hs
|
@ -9,26 +9,27 @@ import qualified Data.Bson as B
|
||||||
import Data.Char
|
import Data.Char
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
|
import qualified Data.Text.Encoding as E
|
||||||
import qualified Data.UUID as U
|
import qualified Data.UUID as U
|
||||||
import Data.Void
|
import Data.Void
|
||||||
import Database.MongoDB
|
import Database.MongoDB
|
||||||
import GHC.Generics
|
import GHC.Generics
|
||||||
import Text.Megaparsec hiding (State)
|
import Text.Megaparsec hiding (State)
|
||||||
import Text.Megaparsec.Char
|
import Text.Megaparsec.Char
|
||||||
|
import ZcashHaskell.Orchard
|
||||||
|
import ZcashHaskell.Sapling (isValidShieldedAddress)
|
||||||
|
|
||||||
-- | Type to model a ZGo transaction
|
-- | Type to model a ZGo transaction
|
||||||
data ZGoTx =
|
data ZGoTx = ZGoTx
|
||||||
ZGoTx
|
{ _id :: Maybe ObjectId
|
||||||
{ _id :: Maybe ObjectId
|
, address :: T.Text
|
||||||
, address :: T.Text
|
, session :: T.Text
|
||||||
, session :: T.Text
|
, confirmations :: Integer
|
||||||
, confirmations :: Integer
|
, blocktime :: Integer
|
||||||
, blocktime :: Integer
|
, amount :: Double
|
||||||
, amount :: Double
|
, txid :: T.Text
|
||||||
, txid :: T.Text
|
, memo :: T.Text
|
||||||
, memo :: T.Text
|
} deriving (Eq, Show, Generic)
|
||||||
}
|
|
||||||
deriving (Eq, Show, Generic)
|
|
||||||
|
|
||||||
parseZGoTxBson :: B.Document -> Maybe ZGoTx
|
parseZGoTxBson :: B.Document -> Maybe ZGoTx
|
||||||
parseZGoTxBson d = do
|
parseZGoTxBson d = do
|
||||||
|
@ -100,26 +101,25 @@ instance Val ZGoTx where
|
||||||
]
|
]
|
||||||
|
|
||||||
-- | Type to represent and parse ZGo memos
|
-- | Type to represent and parse ZGo memos
|
||||||
data ZGoMemo =
|
data ZGoMemo = ZGoMemo
|
||||||
ZGoMemo
|
{ m_session :: Maybe U.UUID
|
||||||
{ m_session :: Maybe U.UUID
|
, m_address :: Maybe T.Text
|
||||||
, m_address :: Maybe T.Text
|
, m_payment :: Bool
|
||||||
, m_payment :: Bool
|
, m_orderId :: Maybe T.Text
|
||||||
}
|
} deriving (Eq, Show)
|
||||||
deriving (Eq, Show)
|
|
||||||
|
|
||||||
data MemoToken
|
data MemoToken
|
||||||
= Login !U.UUID
|
= Login !U.UUID
|
||||||
| PayMsg !U.UUID
|
| PayMsg !U.UUID
|
||||||
| Address !T.Text
|
| Address !T.Text
|
||||||
| Msg !T.Text
|
| Msg !T.Text
|
||||||
|
| OrderId !T.Text
|
||||||
deriving (Show, Eq)
|
deriving (Show, Eq)
|
||||||
|
|
||||||
type Parser = Parsec Void T.Text
|
type Parser = Parsec Void T.Text
|
||||||
|
|
||||||
pSession :: Parser MemoToken
|
pSession :: Parser MemoToken
|
||||||
pSession = do
|
pSession = do
|
||||||
optional spaceChar
|
|
||||||
string "ZGO"
|
string "ZGO"
|
||||||
pay <- optional $ char 'p'
|
pay <- optional $ char 'p'
|
||||||
string "::"
|
string "::"
|
||||||
|
@ -136,19 +136,38 @@ pSaplingAddress :: Parser MemoToken
|
||||||
pSaplingAddress = do
|
pSaplingAddress = do
|
||||||
string "zs"
|
string "zs"
|
||||||
a <- some alphaNumChar
|
a <- some alphaNumChar
|
||||||
if length a /= 76
|
if isValidShieldedAddress (E.encodeUtf8 $ "zs" <> T.pack a)
|
||||||
then fail "Failed to parse Sapling address"
|
then pure $ Address $ T.pack ("zs" <> a)
|
||||||
else pure $ Address $ T.pack ("zs" <> a)
|
else fail "Failed to parse Sapling address"
|
||||||
|
|
||||||
|
pUnifiedAddress :: Parser MemoToken
|
||||||
|
pUnifiedAddress = do
|
||||||
|
string "u1"
|
||||||
|
a <- some alphaNumChar
|
||||||
|
case isValidUnifiedAddress (E.encodeUtf8 $ "u1" <> T.pack a) of
|
||||||
|
Just u -> pure $ Address $ T.pack ("u1" <> a)
|
||||||
|
Nothing -> fail "Failed to parse Unified Address"
|
||||||
|
|
||||||
|
pOrderId :: Parser MemoToken
|
||||||
|
pOrderId = do
|
||||||
|
string "ZGo Order::"
|
||||||
|
a <- some hexDigitChar
|
||||||
|
pure $ OrderId . T.pack $ a
|
||||||
|
|
||||||
pMsg :: Parser MemoToken
|
pMsg :: Parser MemoToken
|
||||||
pMsg = do
|
pMsg = do
|
||||||
Msg . T.pack <$>
|
msg <-
|
||||||
some (alphaNumChar <|> punctuationChar <|> charCategory OtherSymbol)
|
some
|
||||||
|
(alphaNumChar <|> punctuationChar <|> symbolChar <|>
|
||||||
|
charCategory OtherSymbol)
|
||||||
|
pure $ Msg . T.pack $ msg
|
||||||
|
|
||||||
pMemo :: Parser MemoToken
|
pMemo :: Parser MemoToken
|
||||||
pMemo = do
|
pMemo = do
|
||||||
optional spaceChar
|
optional $ some spaceChar
|
||||||
pSession <|> pSaplingAddress <|> pMsg
|
t <- pSession <|> pSaplingAddress <|> pUnifiedAddress <|> pOrderId <|> pMsg
|
||||||
|
optional $ some spaceChar
|
||||||
|
return t
|
||||||
|
|
||||||
isMemoToken :: T.Text -> MemoToken -> Bool
|
isMemoToken :: T.Text -> MemoToken -> Bool
|
||||||
isMemoToken kind t =
|
isMemoToken kind t =
|
||||||
|
@ -171,8 +190,15 @@ isMemoToken kind t =
|
||||||
pZGoMemo :: Parser ZGoMemo
|
pZGoMemo :: Parser ZGoMemo
|
||||||
pZGoMemo = do
|
pZGoMemo = do
|
||||||
tks <- some pMemo
|
tks <- some pMemo
|
||||||
pure $ ZGoMemo (isSession tks) (isAddress tks) (isPayment tks)
|
pure $ ZGoMemo (isSession tks) (isAddress tks) (isPayment tks) (isOrder tks)
|
||||||
where
|
where
|
||||||
|
isOrder [] = Nothing
|
||||||
|
isOrder tks =
|
||||||
|
if not (null tks)
|
||||||
|
then case head tks of
|
||||||
|
OrderId x -> Just x
|
||||||
|
_ -> isOrder $ tail tks
|
||||||
|
else Nothing
|
||||||
isPayment [] = False
|
isPayment [] = False
|
||||||
isPayment tks =
|
isPayment tks =
|
||||||
not (null tks) &&
|
not (null tks) &&
|
||||||
|
|
71
stack.yaml
71
stack.yaml
|
@ -1,71 +0,0 @@
|
||||||
# This file was automatically generated by 'stack init'
|
|
||||||
#
|
|
||||||
# Some commonly used options have been documented as comments in this file.
|
|
||||||
# For advanced use and comprehensive documentation of the format, please see:
|
|
||||||
# https://docs.haskellstack.org/en/stable/yaml_configuration/
|
|
||||||
|
|
||||||
# Resolver to choose a 'specific' stackage snapshot or a compiler version.
|
|
||||||
# A snapshot resolver dictates the compiler version and the set of packages
|
|
||||||
# to be used for project dependencies. For example:
|
|
||||||
#
|
|
||||||
# resolver: lts-3.5
|
|
||||||
# resolver: nightly-2015-09-21
|
|
||||||
# resolver: ghc-7.10.2
|
|
||||||
#
|
|
||||||
# The location of a snapshot can be provided as a file or url. Stack assumes
|
|
||||||
# a snapshot provided as a file might change, whereas a url resource does not.
|
|
||||||
#
|
|
||||||
# resolver: ./custom-snapshot.yaml
|
|
||||||
# resolver: https://example.com/snapshots/2018-01-01.yaml
|
|
||||||
resolver: lts-20.19
|
|
||||||
#url: https://raw.githubusercontent.com/commercialhaskell/stackage-snapshots/master/lts/19/4.yaml
|
|
||||||
|
|
||||||
# User packages to be built.
|
|
||||||
# Various formats can be used as shown in the example below.
|
|
||||||
#
|
|
||||||
# packages:
|
|
||||||
# - some-directory
|
|
||||||
# - https://example.com/foo/bar/baz-0.0.2.tar.gz
|
|
||||||
# subdirs:
|
|
||||||
# - auto-update
|
|
||||||
# - wai
|
|
||||||
packages:
|
|
||||||
- .
|
|
||||||
# Dependency packages to be pulled from upstream that are not in the resolver.
|
|
||||||
# These entries can reference officially published versions as well as
|
|
||||||
# forks / in-progress versions pinned to a git hash. For example:
|
|
||||||
#
|
|
||||||
# extra-deps:
|
|
||||||
# - acme-missiles-0.3
|
|
||||||
# - git: https://github.com/commercialhaskell/stack.git
|
|
||||||
# commit: e7b331f14bcffb8367cd58fbfc8b40ec7642100a
|
|
||||||
#
|
|
||||||
# extra-deps: []
|
|
||||||
extra-deps:
|
|
||||||
- git: https://github.com/reach-sh/haskell-hexstring.git
|
|
||||||
commit: 085c16fb21b9f856a435a3faab980e7e0b319341
|
|
||||||
- blake3-0.2@sha256:d1146b9a51ccfbb0532780778b6d016a614e3d44c05d8c1923dde9a8be869045,2448
|
|
||||||
- crypto-rng-0.3.0.1@sha256:04f4ae75943ecad8b794950985054130f272d65a588b6b6528f56df0bfedc4dc,1565
|
|
||||||
# Override default flag values for local packages and extra-deps
|
|
||||||
# flags: {}
|
|
||||||
|
|
||||||
# Extra package databases containing global packages
|
|
||||||
# extra-package-dbs: []
|
|
||||||
|
|
||||||
# Control whether we use the GHC we find on the path
|
|
||||||
# system-ghc: true
|
|
||||||
#
|
|
||||||
# Require a specific version of stack, using version ranges
|
|
||||||
# require-stack-version: -any # Default
|
|
||||||
# require-stack-version: ">=2.7"
|
|
||||||
#
|
|
||||||
# Override the architecture used by stack, especially useful on Windows
|
|
||||||
# arch: i386
|
|
||||||
# arch: x86_64
|
|
||||||
#
|
|
||||||
# Extra directories used by stack for building
|
|
||||||
# extra-include-dirs: [/path/to/dir]
|
|
||||||
# extra-lib-dirs: [/path/to/dir]
|
|
||||||
#
|
|
||||||
# Allow a newer minor version of GHC than the snapshot specifies
|
|
||||||
# compiler-check: newer-minor
|
|
|
@ -1,37 +0,0 @@
|
||||||
# This file was autogenerated by Stack.
|
|
||||||
# You should not edit this file by hand.
|
|
||||||
# For more information, please see the documentation at:
|
|
||||||
# https://docs.haskellstack.org/en/stable/lock_files
|
|
||||||
|
|
||||||
packages:
|
|
||||||
- completed:
|
|
||||||
commit: 085c16fb21b9f856a435a3faab980e7e0b319341
|
|
||||||
git: https://github.com/reach-sh/haskell-hexstring.git
|
|
||||||
name: hexstring
|
|
||||||
pantry-tree:
|
|
||||||
sha256: 9ecf67856f59dfb382b283eceb42e4fc1865935d1a7e59111556ed381c6a2ffd
|
|
||||||
size: 687
|
|
||||||
version: 0.11.1
|
|
||||||
original:
|
|
||||||
commit: 085c16fb21b9f856a435a3faab980e7e0b319341
|
|
||||||
git: https://github.com/reach-sh/haskell-hexstring.git
|
|
||||||
- completed:
|
|
||||||
hackage: blake3-0.2@sha256:d1146b9a51ccfbb0532780778b6d016a614e3d44c05d8c1923dde9a8be869045,2448
|
|
||||||
pantry-tree:
|
|
||||||
sha256: 0264ef3e7919e7b0d668c4153f6ce0d88e6965626b52d9dfd2cafd70309501d3
|
|
||||||
size: 1433
|
|
||||||
original:
|
|
||||||
hackage: blake3-0.2@sha256:d1146b9a51ccfbb0532780778b6d016a614e3d44c05d8c1923dde9a8be869045,2448
|
|
||||||
- completed:
|
|
||||||
hackage: crypto-rng-0.3.0.1@sha256:04f4ae75943ecad8b794950985054130f272d65a588b6b6528f56df0bfedc4dc,1565
|
|
||||||
pantry-tree:
|
|
||||||
sha256: 1caccafe35d1ae3063f057c31188742a8e794f4f4e4530bab4019c0a514ee54f
|
|
||||||
size: 455
|
|
||||||
original:
|
|
||||||
hackage: crypto-rng-0.3.0.1@sha256:04f4ae75943ecad8b794950985054130f272d65a588b6b6528f56df0bfedc4dc,1565
|
|
||||||
snapshots:
|
|
||||||
- completed:
|
|
||||||
sha256: 42f77c84b34f68c30c2cd0bf8c349f617a0f428264362426290847a6a2019b64
|
|
||||||
size: 649618
|
|
||||||
url: https://raw.githubusercontent.com/commercialhaskell/stackage-snapshots/master/lts/20/19.yaml
|
|
||||||
original: lts-20.19
|
|
952
test/Spec.hs
952
test/Spec.hs
File diff suppressed because it is too large
Load diff
1
zcash-haskell
Submodule
1
zcash-haskell
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 90c8a7c3028bd6836dea5655221277a25d457653
|
|
@ -1,18 +1,18 @@
|
||||||
cabal-version: 1.12
|
cabal-version: 3.0
|
||||||
|
|
||||||
-- This file has been generated from package.yaml by hpack version 0.35.1.
|
-- This file has been generated from package.yaml by hpack version 0.36.0.
|
||||||
--
|
--
|
||||||
-- see: https://github.com/sol/hpack
|
-- see: https://github.com/sol/hpack
|
||||||
|
|
||||||
name: zgo-backend
|
name: zgo-backend
|
||||||
version: 1.5.0
|
version: 1.9.0
|
||||||
synopsis: Haskell Back-end for the ZGo point-of-sale application
|
synopsis: Haskell Back-end for the ZGo point-of-sale application
|
||||||
description: Please see the README at <https://git.vergara.tech/Vergara_Tech//zgo-backend#readme>
|
description: Please see the README at <https://git.vergara.tech/Vergara_Tech//zgo-backend#readme>
|
||||||
category: Web
|
category: Web
|
||||||
author: Rene Vergara
|
author: Rene Vergara
|
||||||
maintainer: rene@vergara.network
|
maintainer: rene@vergara.network
|
||||||
copyright: Copyright (c) 2023 Vergara Technologies LLC
|
copyright: 2022-2024 Vergara Technologies LLC
|
||||||
license: BOSL
|
license: MIT
|
||||||
license-file: LICENSE
|
license-file: LICENSE
|
||||||
build-type: Simple
|
build-type: Simple
|
||||||
extra-source-files:
|
extra-source-files:
|
||||||
|
@ -20,10 +20,6 @@ extra-source-files:
|
||||||
CHANGELOG.md
|
CHANGELOG.md
|
||||||
zgo.cfg
|
zgo.cfg
|
||||||
|
|
||||||
source-repository head
|
|
||||||
type: git
|
|
||||||
location: https://git.vergara.tech/Vergara_Tech/zgo-backend
|
|
||||||
|
|
||||||
library
|
library
|
||||||
exposed-modules:
|
exposed-modules:
|
||||||
Config
|
Config
|
||||||
|
@ -37,8 +33,6 @@ library
|
||||||
Xero
|
Xero
|
||||||
ZGoBackend
|
ZGoBackend
|
||||||
ZGoTx
|
ZGoTx
|
||||||
other-modules:
|
|
||||||
Paths_zgo_backend
|
|
||||||
hs-source-dirs:
|
hs-source-dirs:
|
||||||
src
|
src
|
||||||
build-depends:
|
build-depends:
|
||||||
|
@ -78,17 +72,16 @@ library
|
||||||
, wai-cors
|
, wai-cors
|
||||||
, wai-extra
|
, wai-extra
|
||||||
, warp-tls
|
, warp-tls
|
||||||
|
, zcash-haskell
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
|
||||||
executable zgo-backend-exe
|
executable zgo-backend-exe
|
||||||
main-is: Server.hs
|
main-is: Server.hs
|
||||||
other-modules:
|
|
||||||
Tasks
|
|
||||||
TokenRefresh
|
|
||||||
Paths_zgo_backend
|
|
||||||
hs-source-dirs:
|
hs-source-dirs:
|
||||||
app
|
app
|
||||||
ghc-options: -main-is Server -threaded -rtsopts -with-rtsopts=-N -Wall
|
ghc-options: -main-is Server -threaded -rtsopts -with-rtsopts=-N -Wall
|
||||||
|
pkgconfig-depends:
|
||||||
|
rustzcash_wrapper
|
||||||
build-depends:
|
build-depends:
|
||||||
aeson
|
aeson
|
||||||
, base
|
, base
|
||||||
|
@ -97,7 +90,7 @@ executable zgo-backend-exe
|
||||||
, http-conduit
|
, http-conduit
|
||||||
, http-types
|
, http-types
|
||||||
, megaparsec
|
, megaparsec
|
||||||
, mongoDB
|
, mongoDB >=2.7.1.4
|
||||||
, scotty
|
, scotty
|
||||||
, securemem
|
, securemem
|
||||||
, text
|
, text
|
||||||
|
@ -110,13 +103,11 @@ executable zgo-backend-exe
|
||||||
|
|
||||||
executable zgo-tasks
|
executable zgo-tasks
|
||||||
main-is: Tasks.hs
|
main-is: Tasks.hs
|
||||||
other-modules:
|
|
||||||
Server
|
|
||||||
TokenRefresh
|
|
||||||
Paths_zgo_backend
|
|
||||||
hs-source-dirs:
|
hs-source-dirs:
|
||||||
app
|
app
|
||||||
ghc-options: -main-is Tasks -threaded -rtsopts -with-rtsopts=-N -Wall
|
ghc-options: -main-is Tasks -threaded -rtsopts -with-rtsopts=-N -Wall
|
||||||
|
pkgconfig-depends:
|
||||||
|
rustzcash_wrapper
|
||||||
build-depends:
|
build-depends:
|
||||||
base
|
base
|
||||||
, megaparsec
|
, megaparsec
|
||||||
|
@ -130,13 +121,11 @@ executable zgo-tasks
|
||||||
|
|
||||||
executable zgo-token-refresh
|
executable zgo-token-refresh
|
||||||
main-is: TokenRefresh.hs
|
main-is: TokenRefresh.hs
|
||||||
other-modules:
|
|
||||||
Server
|
|
||||||
Tasks
|
|
||||||
Paths_zgo_backend
|
|
||||||
hs-source-dirs:
|
hs-source-dirs:
|
||||||
app
|
app
|
||||||
ghc-options: -main-is TokenRefresh -threaded -rtsopts -with-rtsopts=-N -Wall
|
ghc-options: -main-is TokenRefresh -threaded -rtsopts -with-rtsopts=-N -Wall
|
||||||
|
pkgconfig-depends:
|
||||||
|
rustzcash_wrapper
|
||||||
build-depends:
|
build-depends:
|
||||||
aeson
|
aeson
|
||||||
, base
|
, base
|
||||||
|
@ -159,11 +148,11 @@ executable zgo-token-refresh
|
||||||
test-suite zgo-backend-test
|
test-suite zgo-backend-test
|
||||||
type: exitcode-stdio-1.0
|
type: exitcode-stdio-1.0
|
||||||
main-is: Spec.hs
|
main-is: Spec.hs
|
||||||
other-modules:
|
|
||||||
Paths_zgo_backend
|
|
||||||
hs-source-dirs:
|
hs-source-dirs:
|
||||||
test
|
test
|
||||||
ghc-options: -threaded -rtsopts -with-rtsopts=-N -main-is Spec
|
ghc-options: -threaded -rtsopts -with-rtsopts=-N -main-is Spec
|
||||||
|
pkgconfig-depends:
|
||||||
|
rustzcash_wrapper
|
||||||
build-depends:
|
build-depends:
|
||||||
QuickCheck
|
QuickCheck
|
||||||
, aeson
|
, aeson
|
||||||
|
@ -174,11 +163,15 @@ test-suite zgo-backend-test
|
||||||
, hspec-expectations-json
|
, hspec-expectations-json
|
||||||
, hspec-wai
|
, hspec-wai
|
||||||
, http-conduit
|
, http-conduit
|
||||||
|
, HUnit
|
||||||
, http-types
|
, http-types
|
||||||
|
, megaparsec
|
||||||
, mongoDB
|
, mongoDB
|
||||||
, scotty
|
, scotty
|
||||||
, securemem
|
, securemem
|
||||||
, text
|
, text
|
||||||
, time
|
, time
|
||||||
|
, uuid
|
||||||
|
, zcash-haskell
|
||||||
, zgo-backend
|
, zgo-backend
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
|
1
zgo.cfg
1
zgo.cfg
|
@ -6,6 +6,7 @@ dbUser = "zgo"
|
||||||
dbPassword = "zcashrules"
|
dbPassword = "zcashrules"
|
||||||
nodeUser = "zecwallet"
|
nodeUser = "zecwallet"
|
||||||
nodePassword = "rdsxlun6v4a"
|
nodePassword = "rdsxlun6v4a"
|
||||||
|
confirmations = 100
|
||||||
port = 3000
|
port = 3000
|
||||||
tls = false
|
tls = false
|
||||||
certificate = "/path/to/cert.pem"
|
certificate = "/path/to/cert.pem"
|
||||||
|
|
|
@ -6,6 +6,7 @@ dbUser = "zgo"
|
||||||
dbPassword = "zcashrules"
|
dbPassword = "zcashrules"
|
||||||
nodeUser = "zecwallet"
|
nodeUser = "zecwallet"
|
||||||
nodePassword = "rdsxlun6v4a"
|
nodePassword = "rdsxlun6v4a"
|
||||||
|
confirmations = 100
|
||||||
port = 3000
|
port = 3000
|
||||||
tls = false
|
tls = false
|
||||||
certificate = "/path/to/cert.pem"
|
certificate = "/path/to/cert.pem"
|
||||||
|
|
Loading…
Reference in a new issue