mirror of https://github.com/lukechilds/lnbits.git
Arc
5 years ago
committed by
GitHub
39 changed files with 2231 additions and 365 deletions
@ -0,0 +1,8 @@ |
|||
from flask import Blueprint |
|||
|
|||
|
|||
core_app = Blueprint("core", __name__, template_folder="templates") |
|||
|
|||
|
|||
from .views_api import * # noqa |
|||
from .views import * # noqa |
After Width: | Height: | Size: 23 KiB |
@ -0,0 +1,14 @@ |
|||
from flask import render_template, send_from_directory |
|||
from os import path |
|||
|
|||
from lnbits.core import core_app |
|||
|
|||
|
|||
@core_app.route("/favicon.ico") |
|||
def favicon(): |
|||
return send_from_directory(path.join(core_app.root_path, "static"), "favicon.ico") |
|||
|
|||
|
|||
@core_app.route("/") |
|||
def home(): |
|||
return render_template("index.html") |
@ -1,73 +0,0 @@ |
|||
|
|||
![Lightning network wallet](https://i.imgur.com/arUWZbH.png) |
|||
# LNbits |
|||
Simple free and open-source Python lightning-network wallet/accounts system. Use https://lnbits.com, or run your own LNbits server! |
|||
|
|||
LNbits is a very simple server that sits on top of a funding source, and can be used as: |
|||
* Accounts system to mitigate the risk of exposing applications to your full balance, via unique API keys for each wallet! |
|||
* Fallback wallet for the LNURL scheme |
|||
* Instant wallet for LN demonstrations |
|||
|
|||
The wallet can run on top of any lightning-network funding source such as LND, lntxbot, paywall, opennode, etc. This first BETA release runs on top of lntxbot, other releases coming soon, although if you are impatient, it could be ported to other funding sources relatively easily. Contributors very welcome :) |
|||
|
|||
LNbits is still in BETA. Please report any vulnerabilities responsibly |
|||
## LNbits as an account system |
|||
LNbits is packaged with tools to help manage funds, such as a table of transactions, line chart of spending, export to csv + more to come.. |
|||
|
|||
|
|||
![Lightning network wallet](https://i.imgur.com/Sd4ri3T.png) |
|||
|
|||
Each wallet also comes with its own API keys, to help partition the exposure of your funding source. |
|||
|
|||
(LNbits M5StackSats available here https://github.com/arcbtc/M5StackSats) |
|||
|
|||
![lnurl ATM](https://i.imgur.com/ABruzAn.png) |
|||
|
|||
## LNbits as an LNURL-withdraw fallback |
|||
LNURL has a fallback scheme, so if scanned by a regular QR code reader it can default to a URL. LNbits exploits this to generate an instant wallet using the LNURL-withdraw. |
|||
|
|||
![lnurl fallback](https://i.imgur.com/CPBKHIv.png) |
|||
https://github.com/btcontract/lnurl-rfc/blob/master/spec.md |
|||
|
|||
Adding **/lnurl?lightning="LNURL-WITHDRAW"** will trigger a withdraw that builds an LNbits wallet. |
|||
Example use would be an ATM, which utilises LNURL, if the user scans the QR with a regular QR code scanner app, they will stilll be able to access the funds. |
|||
|
|||
![lnurl ATM](https://i.imgur.com/Gi6bn3L.jpg) |
|||
|
|||
## LNbits as an insta-wallet |
|||
Wallets can be easily generated and given out to people at events (one click multi-wallet generation to be added soon). |
|||
"Go to this website", has a lot less friction than "Download this app". |
|||
|
|||
![lnurl ATM](https://i.imgur.com/SF5KoIe.png) |
|||
|
|||
# Running LNbits locally |
|||
Download this repo |
|||
|
|||
LNbits uses [Flask](http://flask.pocoo.org/). |
|||
Feel free to contribute to the project. |
|||
|
|||
Application dependencies |
|||
------------------------ |
|||
The application uses [Pipenv][pipenv] to manage Python packages. |
|||
While in development, you will need to install all dependencies: |
|||
|
|||
$ pipenv shell |
|||
$ pipenv install --dev |
|||
|
|||
You will need to set the variables in .env.example, and rename the file to .env |
|||
|
|||
![lnurl ATM](https://i.imgur.com/ri2zOe8.png) |
|||
|
|||
Running the server |
|||
------------------ |
|||
|
|||
$ flask run |
|||
|
|||
There is an environment variable called `FLASK_ENV` that has to be set to `development` |
|||
if you want to run Flask in debug mode with autoreload |
|||
|
|||
[pipenv]: https://docs.pipenv.org/#install-pipenv-today |
|||
|
|||
# Tip me |
|||
If you like this project and might even use or extend it, why not send some tip love! |
|||
https://paywall.link/to/f4e4e |
@ -1,73 +0,0 @@ |
|||
|
|||
![Lightning network wallet](https://i.imgur.com/arUWZbH.png) |
|||
# LNbits |
|||
Simple free and open-source Python lightning-network wallet/accounts system. Use https://lnbits.com, or run your own LNbits server! |
|||
|
|||
LNbits is a very simple server that sits on top of a funding source, and can be used as: |
|||
* Accounts system to mitigate the risk of exposing applications to your full balance, via unique API keys for each wallet! |
|||
* Fallback wallet for the LNURL scheme |
|||
* Instant wallet for LN demonstrations |
|||
|
|||
The wallet can run on top of any lightning-network funding source such as LND, lntxbot, paywall, opennode, etc. This first BETA release runs on top of lntxbot, other releases coming soon, although if you are impatient, it could be ported to other funding sources relatively easily. Contributors very welcome :) |
|||
|
|||
LNbits is still in BETA. Please report any vulnerabilities responsibly |
|||
## LNbits as an account system |
|||
LNbits is packaged with tools to help manage funds, such as a table of transactions, line chart of spending, export to csv + more to come.. |
|||
|
|||
|
|||
![Lightning network wallet](https://i.imgur.com/Sd4ri3T.png) |
|||
|
|||
Each wallet also comes with its own API keys, to help partition the exposure of your funding source. |
|||
|
|||
(LNbits M5StackSats available here https://github.com/arcbtc/M5StackSats) |
|||
|
|||
![lnurl ATM](https://i.imgur.com/ABruzAn.png) |
|||
|
|||
## LNbits as an LNURL-withdraw fallback |
|||
LNURL has a fallback scheme, so if scanned by a regular QR code reader it can default to a URL. LNbits exploits this to generate an instant wallet using the LNURL-withdraw. |
|||
|
|||
![lnurl fallback](https://i.imgur.com/CPBKHIv.png) |
|||
https://github.com/btcontract/lnurl-rfc/blob/master/spec.md |
|||
|
|||
Adding **/lnurl?lightning="LNURL-WITHDRAW"** will trigger a withdraw that builds an LNbits wallet. |
|||
Example use would be an ATM, which utilises LNURL, if the user scans the QR with a regular QR code scanner app, they will stilll be able to access the funds. |
|||
|
|||
![lnurl ATM](https://i.imgur.com/Gi6bn3L.jpg) |
|||
|
|||
## LNbits as an insta-wallet |
|||
Wallets can be easily generated and given out to people at events (one click multi-wallet generation to be added soon). |
|||
"Go to this website", has a lot less friction than "Download this app". |
|||
|
|||
![lnurl ATM](https://i.imgur.com/SF5KoIe.png) |
|||
|
|||
# Running LNbits locally |
|||
Download this repo |
|||
|
|||
LNbits uses [Flask](http://flask.pocoo.org/). |
|||
Feel free to contribute to the project. |
|||
|
|||
Application dependencies |
|||
------------------------ |
|||
The application uses [Pipenv][pipenv] to manage Python packages. |
|||
While in development, you will need to install all dependencies: |
|||
|
|||
$ pipenv shell |
|||
$ pipenv install --dev |
|||
|
|||
You will need to set the variables in .env.example, and rename the file to .env |
|||
|
|||
![lnurl ATM](https://i.imgur.com/ri2zOe8.png) |
|||
|
|||
Running the server |
|||
------------------ |
|||
|
|||
$ flask run |
|||
|
|||
There is an environment variable called `FLASK_ENV` that has to be set to `development` |
|||
if you want to run Flask in debug mode with autoreload |
|||
|
|||
[pipenv]: https://docs.pipenv.org/#install-pipenv-today |
|||
|
|||
# Tip me |
|||
If you like this project and might even use or extend it, why not send some tip love! |
|||
https://paywall.link/to/f4e4e |
@ -1,73 +0,0 @@ |
|||
|
|||
![Lightning network wallet](https://i.imgur.com/arUWZbH.png) |
|||
# LNbits |
|||
Simple free and open-source Python lightning-network wallet/accounts system. Use https://lnbits.com, or run your own LNbits server! |
|||
|
|||
LNbits is a very simple server that sits on top of a funding source, and can be used as: |
|||
* Accounts system to mitigate the risk of exposing applications to your full balance, via unique API keys for each wallet! |
|||
* Fallback wallet for the LNURL scheme |
|||
* Instant wallet for LN demonstrations |
|||
|
|||
The wallet can run on top of any lightning-network funding source such as LND, lntxbot, paywall, opennode, etc. This first BETA release runs on top of lntxbot, other releases coming soon, although if you are impatient, it could be ported to other funding sources relatively easily. Contributors very welcome :) |
|||
|
|||
LNbits is still in BETA. Please report any vulnerabilities responsibly |
|||
## LNbits as an account system |
|||
LNbits is packaged with tools to help manage funds, such as a table of transactions, line chart of spending, export to csv + more to come.. |
|||
|
|||
|
|||
![Lightning network wallet](https://i.imgur.com/Sd4ri3T.png) |
|||
|
|||
Each wallet also comes with its own API keys, to help partition the exposure of your funding source. |
|||
|
|||
(LNbits M5StackSats available here https://github.com/arcbtc/M5StackSats) |
|||
|
|||
![lnurl ATM](https://i.imgur.com/ABruzAn.png) |
|||
|
|||
## LNbits as an LNURL-withdraw fallback |
|||
LNURL has a fallback scheme, so if scanned by a regular QR code reader it can default to a URL. LNbits exploits this to generate an instant wallet using the LNURL-withdraw. |
|||
|
|||
![lnurl fallback](https://i.imgur.com/CPBKHIv.png) |
|||
https://github.com/btcontract/lnurl-rfc/blob/master/spec.md |
|||
|
|||
Adding **/lnurl?lightning="LNURL-WITHDRAW"** will trigger a withdraw that builds an LNbits wallet. |
|||
Example use would be an ATM, which utilises LNURL, if the user scans the QR with a regular QR code scanner app, they will stilll be able to access the funds. |
|||
|
|||
![lnurl ATM](https://i.imgur.com/Gi6bn3L.jpg) |
|||
|
|||
## LNbits as an insta-wallet |
|||
Wallets can be easily generated and given out to people at events (one click multi-wallet generation to be added soon). |
|||
"Go to this website", has a lot less friction than "Download this app". |
|||
|
|||
![lnurl ATM](https://i.imgur.com/SF5KoIe.png) |
|||
|
|||
# Running LNbits locally |
|||
Download this repo |
|||
|
|||
LNbits uses [Flask](http://flask.pocoo.org/). |
|||
Feel free to contribute to the project. |
|||
|
|||
Application dependencies |
|||
------------------------ |
|||
The application uses [Pipenv][pipenv] to manage Python packages. |
|||
While in development, you will need to install all dependencies: |
|||
|
|||
$ pipenv shell |
|||
$ pipenv install --dev |
|||
|
|||
You will need to set the variables in .env.example, and rename the file to .env |
|||
|
|||
![lnurl ATM](https://i.imgur.com/ri2zOe8.png) |
|||
|
|||
Running the server |
|||
------------------ |
|||
|
|||
$ flask run |
|||
|
|||
There is an environment variable called `FLASK_ENV` that has to be set to `development` |
|||
if you want to run Flask in debug mode with autoreload |
|||
|
|||
[pipenv]: https://docs.pipenv.org/#install-pipenv-today |
|||
|
|||
# Tip me |
|||
If you like this project and might even use or extend it, why not send some tip love! |
|||
https://paywall.link/to/f4e4e |
Binary file not shown.
@ -0,0 +1,8 @@ |
|||
from flask import Blueprint |
|||
|
|||
|
|||
withdraw_ext = Blueprint("withdraw", __name__, static_folder="static", template_folder="templates") |
|||
|
|||
|
|||
from .views_api import * # noqa |
|||
from .views import * # noqa |
Binary file not shown.
@ -0,0 +1,530 @@ |
|||
<!-- @format --> |
|||
|
|||
<!DOCTYPE html> |
|||
<html> |
|||
<head> |
|||
<meta charset="UTF-8" /> |
|||
<title>LNBits Wallet</title> |
|||
<meta |
|||
content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no" |
|||
name="viewport" |
|||
/> |
|||
<!-- Bootstrap 3.3.2 --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='bootstrap/css/bootstrap.min.css') }}" |
|||
/> |
|||
<!-- FontAwesome 4.3.0 --> |
|||
<link |
|||
href="https://maxcdn.bootstrapcdn.com/font-awesome/4.3.0/css/font-awesome.min.css" |
|||
rel="stylesheet" |
|||
type="text/css" |
|||
/> |
|||
<!-- Ionicons 2.0.0 --> |
|||
<link |
|||
href="https://code.ionicframework.com/ionicons/2.0.0/css/ionicons.min.css" |
|||
rel="stylesheet" |
|||
type="text/css" |
|||
/> |
|||
|
|||
<!-- Theme style --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='dist/css/AdminLTE.min.css') }}" |
|||
/> |
|||
<!-- AdminLTE Skins. Choose a skin from the css/skins |
|||
folder instead of downloading all of them to reduce the load. --> |
|||
|
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='dist/css/skins/_all-skins.min.css') }}" |
|||
/> |
|||
|
|||
<!-- Morris chart --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='plugins/morris/morris.css') }}" |
|||
/> |
|||
|
|||
<!-- jvectormap --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='plugins/jvectormap/jquery-jvectormap-1.2.2.css') }}" |
|||
/> |
|||
|
|||
<!-- bootstrap wysihtml5 - text editor --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='plugins/bootstrap-wysihtml5/bootstrap3-wysihtml5.min.css') }}" |
|||
/> |
|||
|
|||
<!-- HTML5 Shim and Respond.js IE8 support of HTML5 elements and media queries --> |
|||
<!-- WARNING: Respond.js doesn't work if you view the page via file:// --> |
|||
<!--[if lt IE 9]> |
|||
<script src="https://oss.maxcdn.com/libs/html5shiv/3.7.0/html5shiv.js"></script> |
|||
<script src="https://oss.maxcdn.com/libs/respond.js/1.3.0/respond.min.js"></script> |
|||
<![endif]--> |
|||
|
|||
<style> |
|||
.small-box > .small-box-footer { |
|||
text-align: left; |
|||
padding-left: 10px; |
|||
} |
|||
|
|||
#loadingMessage { |
|||
text-align: center; |
|||
padding: 40px; |
|||
background-color: #eee; |
|||
} |
|||
|
|||
#canvas { |
|||
width: 100%; |
|||
} |
|||
|
|||
#output { |
|||
margin-top: 20px; |
|||
background: #eee; |
|||
padding: 10px; |
|||
padding-bottom: 0; |
|||
} |
|||
|
|||
#output div { |
|||
padding-bottom: 10px; |
|||
word-wrap: break-word; |
|||
} |
|||
|
|||
#noQRFound { |
|||
text-align: center; |
|||
} |
|||
</style> |
|||
|
|||
<!-- jQuery 2.1.3 --> |
|||
<script src="{{ url_for('static', filename='plugins/jQuery/jQuery-2.1.3.min.js') }}"></script> |
|||
<!-- jQuery UI 1.11.2 --> |
|||
<script |
|||
src="https://code.jquery.com/ui/1.11.2/jquery-ui.min.js" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Resolve conflict in jQuery UI tooltip with Bootstrap tooltip --> |
|||
<script> |
|||
$.widget.bridge('uibutton', $.ui.button) |
|||
</script> |
|||
<!-- Bootstrap 3.3.2 JS --> |
|||
<script |
|||
src="{{ url_for('static', filename='bootstrap/js/bootstrap.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Morris.js charts --> |
|||
<script src="https://cdnjs.cloudflare.com/ajax/libs/raphael/2.1.0/raphael-min.js"></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/morris/morris.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Sparkline --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/sparkline/jquery.sparkline.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- jvectormap --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jvectormap/jquery-jvectormap-1.2.2.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jvectormap/jquery-jvectormap-world-mill-en.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- jQuery Knob Chart --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/knob/jquery.knob.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Bootstrap WYSIHTML5 --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/bootstrap-wysihtml5/bootstrap3-wysihtml5.all.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Slimscroll --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/slimScroll/jquery.slimscroll.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- FastClick --> |
|||
<script src="{{ url_for('static', filename='plugins/fastclick/fastclick.min.js') }}"></script> |
|||
<!-- AdminLTE App --> |
|||
<script |
|||
src="{{ url_for('static', filename='dist/js/app.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
<!-- AdminLTE dashboard demo (This is only for demo purposes) --> |
|||
<script |
|||
src="{{ url_for('static', filename='dist/js/pages/dashboard.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
<!-- AdminLTE for demo purposes --> |
|||
<script |
|||
src="{{ url_for('static', filename='dist/js/demo.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
<script |
|||
src="{{ url_for('static', filename='plugins/datatables/jquery.dataTables.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<link |
|||
rel="stylesheet" |
|||
href="//cdnjs.cloudflare.com/ajax/libs/morris.js/0.5.1/morris.css" |
|||
/> |
|||
<script src="https://cdnjs.cloudflare.com/ajax/libs/raphael/2.1.0/raphael-min.js"></script> |
|||
<script src="https://cdnjs.cloudflare.com/ajax/libs/morris.js/0.5.1/morris.min.js"></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jscam/JS.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jscam/qrcode.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/bolt11/decoder.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/bolt11/utils.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
|
|||
|
|||
<style> |
|||
|
|||
//GOOFY CSS HACK TO GO DARK |
|||
|
|||
.skin-blue .wrapper { |
|||
background: |
|||
#1f2234; |
|||
} |
|||
|
|||
body { |
|||
color: #fff; |
|||
} |
|||
|
|||
.skin-blue .sidebar-menu > li.active > a { |
|||
color: #fff; |
|||
background:#1f2234; |
|||
border-left-color:#8964a9; |
|||
} |
|||
|
|||
.skin-blue .main-header .navbar { |
|||
background-color: |
|||
#2e507d; |
|||
} |
|||
|
|||
.content-wrapper, .right-side { |
|||
background-color: |
|||
#1f2234; |
|||
} |
|||
.skin-blue .main-header .logo { |
|||
background-color: |
|||
#1f2234; |
|||
color: |
|||
#fff; |
|||
} |
|||
|
|||
.skin-blue .sidebar-menu > li.header { |
|||
color: |
|||
#4b646f; |
|||
background: |
|||
#1f2234; |
|||
} |
|||
.skin-blue .wrapper, .skin-blue .main-sidebar, .skin-blue .left-side { |
|||
background: |
|||
#1f2234; |
|||
} |
|||
|
|||
.skin-blue .sidebar-menu > li > .treeview-menu { |
|||
margin: 0 1px; |
|||
background: |
|||
#1f2234; |
|||
} |
|||
|
|||
.skin-blue .sidebar-menu > li > a { |
|||
border-left: 3px solid |
|||
transparent; |
|||
margin-right: 1px; |
|||
} |
|||
.skin-blue .sidebar-menu > li > a:hover, .skin-blue .sidebar-menu > li.active > a { |
|||
|
|||
color: #fff; |
|||
background:#3e355a; |
|||
border-left-color:#8964a9; |
|||
|
|||
} |
|||
|
|||
|
|||
.skin-blue .main-header .logo:hover { |
|||
background: |
|||
#3e355a; |
|||
} |
|||
|
|||
.skin-blue .main-header .navbar .sidebar-toggle:hover { |
|||
background-color: |
|||
#3e355a; |
|||
} |
|||
.main-footer { |
|||
background-color: #1f2234; |
|||
padding: 15px; |
|||
color: #fff; |
|||
border-top: 0px; |
|||
} |
|||
|
|||
.skin-blue .main-header .navbar { |
|||
background-color: #1f2234; |
|||
} |
|||
|
|||
.bg-red, .callout.callout-danger, .alert-danger, .alert-error, .label-danger, .modal-danger .modal-body { |
|||
background-color: |
|||
#1f2234 !important; |
|||
} |
|||
.alert-danger, .alert-error { |
|||
|
|||
border-color: #fff; |
|||
border: 1px solid |
|||
|
|||
#fff; |
|||
border-radius: 7px; |
|||
|
|||
} |
|||
|
|||
.skin-blue .main-header .navbar .nav > li > a:hover, .skin-blue .main-header .navbar .nav > li > a:active, .skin-blue .main-header .navbar .nav > li > a:focus, .skin-blue .main-header .navbar .nav .open > a, .skin-blue .main-header .navbar .nav .open > a:hover, .skin-blue .main-header .navbar .nav .open > a:focus { |
|||
color: |
|||
#f6f6f6; |
|||
background-color: #3e355a; |
|||
} |
|||
.bg-aqua, .callout.callout-info, .alert-info, .label-info, .modal-info .modal-body { |
|||
background-color: |
|||
#3e355a !important; |
|||
} |
|||
|
|||
.box { |
|||
position: relative; |
|||
border-radius: 3px; |
|||
background-color: #333646; |
|||
border-top: 3px solid #8964a9; |
|||
margin-bottom: 20px; |
|||
width: 100%; |
|||
|
|||
} |
|||
.table-striped > tbody > tr:nth-of-type(2n+1) { |
|||
background-color: |
|||
#333646; |
|||
} |
|||
|
|||
.box-header { |
|||
color: #fff; |
|||
|
|||
} |
|||
|
|||
|
|||
.box.box-danger { |
|||
border-top-color: #8964a9; |
|||
} |
|||
.box.box-primary { |
|||
border-top-color: #8964a9; |
|||
} |
|||
|
|||
a { |
|||
color: #8964a9; |
|||
} |
|||
.box-header.with-border { |
|||
border-bottom: none; |
|||
} |
|||
|
|||
a:hover, a:active, a:focus { |
|||
outline: none; |
|||
text-decoration: none; |
|||
color: #fff; |
|||
} |
|||
// .modal.in .modal-dialog{ |
|||
// color:#000; |
|||
// } |
|||
|
|||
.form-control { |
|||
|
|||
background-color:#333646; |
|||
color: #fff; |
|||
|
|||
} |
|||
.box-footer { |
|||
|
|||
border-top: none; |
|||
|
|||
background-color: |
|||
#333646; |
|||
} |
|||
.modal-footer { |
|||
|
|||
border-top: none; |
|||
|
|||
} |
|||
.modal-content { |
|||
|
|||
background-color: |
|||
#333646; |
|||
} |
|||
.modal.in .modal-dialog { |
|||
|
|||
background-color: #333646; |
|||
|
|||
} |
|||
|
|||
.layout-boxed { |
|||
background: none; |
|||
background-color: rgba(0, 0, 0, 0); |
|||
background-color: |
|||
#3e355a; |
|||
} |
|||
|
|||
.skin-blue .sidebar-menu > li > a:hover, .skin-blue .sidebar-menu > li.active > a { |
|||
|
|||
background: none; |
|||
} |
|||
</style> |
|||
|
|||
|
|||
</head> |
|||
<body class="skin-blue layout-boxed sidebar-collapse sidebar-open"> |
|||
<div class="wrapper"> |
|||
<header class="main-header"> |
|||
<!-- Logo --> |
|||
<a href="{{ url_for('core.home') }}" class="logo"><b>LN</b>bits</a> |
|||
<!-- Header Navbar: style can be found in header.less --> |
|||
<nav class="navbar navbar-static-top" role="navigation"> |
|||
<!-- Sidebar toggle button--> |
|||
<a |
|||
href="#" |
|||
class="sidebar-toggle" |
|||
data-toggle="offcanvas" |
|||
role="button" |
|||
> |
|||
<span class="sr-only">Toggle navigation</span> |
|||
</a> |
|||
<div class="navbar-custom-menu"> |
|||
<ul class="nav navbar-nav"> |
|||
<!-- Messages: style can be found in dropdown.less--> |
|||
<li class="dropdown messages-menu"> |
|||
|
|||
{% block messages %}{% endblock %} |
|||
</li> |
|||
</ul> |
|||
</div> |
|||
</nav> |
|||
</header> |
|||
|
|||
<aside class="main-sidebar"> |
|||
<!-- sidebar: style can be found in sidebar.less --> |
|||
<section class="sidebar" style="height: auto;"> |
|||
<!-- Sidebar user panel --> |
|||
|
|||
<!-- /.search form --> |
|||
<!-- sidebar menu: : style can be found in sidebar.less --> |
|||
<ul class="sidebar-menu"> |
|||
<li><br/><br/><a href="https://where39.com/"><p>Where39 anon locations</p><img src="static/where39.png" style="width:170px"></a></li> |
|||
<li><br/><a href="https://github.com/arcbtc/Quickening"><p>The Quickening <$8 PoS</p><img src="static/quick.gif" style="width:170px"></a></li> |
|||
<li><br/><a href="https://jigawatt.co/"><p>Buy BTC stamps + electronics</p><img src="static/stamps.jpg" style="width:170px"></a></li> |
|||
<li><br/><a href="mailto:ben@arc.wales"><h3>Advertise here!</h3></a></li> |
|||
|
|||
</ul> |
|||
</section> |
|||
<!-- /.sidebar --> |
|||
</aside> |
|||
|
|||
<!-- Right side column. Contains the navbar and content of the page --> |
|||
<div class="content-wrapper"> |
|||
<!-- Content Header (Page header) --> |
|||
<section class="content-header"> |
|||
<h1> |
|||
LNURL Withdraw Link |
|||
<small>Use LNURL compatible bitcoin wallet</small> |
|||
</h1> |
|||
|
|||
</section> |
|||
|
|||
<!-- Main content --> |
|||
<section class="content"><br/><br/> |
|||
<center><h1 style="font-size:500%">Withdraw Link: {{ user_fau[0][6] }}</h1></center> |
|||
|
|||
<center><br/><br/> <div id="qrcode" style="width: 340px;"></div><br/><br/> |
|||
<div style="width:55%;word-wrap: break-word;" id="qrcodetxt"></div> <br/></center> |
|||
|
|||
</section><!-- /.content --> |
|||
</div><!-- /.content-wrapper --> |
|||
</div> |
|||
</body> |
|||
|
|||
<script> |
|||
function getAjax(url, thekey, success) { |
|||
var xhr = window.XMLHttpRequest |
|||
? new XMLHttpRequest() |
|||
: new ActiveXObject('Microsoft.XMLHTTP') |
|||
xhr.open('GET', url, true) |
|||
xhr.onreadystatechange = function() { |
|||
if (xhr.readyState > 3 && xhr.status == 200) { |
|||
success(xhr.responseText) |
|||
} |
|||
} |
|||
xhr.setRequestHeader('Grpc-Metadata-macaroon', thekey) |
|||
xhr.setRequestHeader('Content-Type', 'application/json') |
|||
|
|||
xhr.send() |
|||
return xhr |
|||
} |
|||
|
|||
|
|||
function drawwithdraw(data) { |
|||
|
|||
|
|||
console.log(data) |
|||
|
|||
|
|||
getAjax('/withdraw/api/v1/lnurlencode/'+ window.location.hostname + "/" + data, "filla", function(datab) { |
|||
if (JSON.parse(datab).status == 'TRUE') { |
|||
console.log(JSON.parse(datab).status) |
|||
lnurlfau = (JSON.parse(datab).lnurl) |
|||
|
|||
|
|||
new QRCode(document.getElementById('qrcode'), { |
|||
text: lnurlfau, |
|||
width: 300, |
|||
height: 300, |
|||
colorDark: '#000000', |
|||
colorLight: '#ffffff', |
|||
correctLevel: QRCode.CorrectLevel.M |
|||
}) |
|||
document.getElementById("qrcode").style.backgroundColor = "white"; |
|||
document.getElementById("qrcode").style.padding = "20px"; |
|||
|
|||
document.getElementById('qrcodetxt').innerHTML = lnurlfau + "<br/><br/>" |
|||
|
|||
} |
|||
else { |
|||
|
|||
data = "Failed to build LNURL" |
|||
} |
|||
}) |
|||
} |
|||
drawwithdraw("{{ user_fau[0][5] }}") |
|||
|
|||
|
|||
|
|||
|
|||
</script> |
|||
</html> |
@ -0,0 +1,498 @@ |
|||
<!-- @format --> |
|||
|
|||
{% extends "base.html" %} {% block messages %} |
|||
<a href="#" class="dropdown-toggle" data-toggle="dropdown"> |
|||
<i class="fa fa-bell-o"></i> |
|||
<span class="label label-danger">!</span> |
|||
</a> |
|||
<ul class="dropdown-menu"> |
|||
<li class="header"><b>Instant wallet, bookmark to save</b></li> |
|||
<li></li> |
|||
</ul> |
|||
{% endblock %} {% block menuitems %} |
|||
<li class="treeview"> |
|||
<a href="#"> |
|||
<i class="fa fa-bitcoin"></i> <span>Wallets</span> |
|||
<i class="fa fa-angle-left pull-right"></i> |
|||
</a> |
|||
<ul class="treeview-menu"> |
|||
{% for w in user_wallets %} |
|||
<li> |
|||
<a href="{{ url_for('wallet') }}?wal={{ w.id }}&usr={{ w.user }}" |
|||
><i class="fa fa-bolt"></i> {{ w.name }}</a |
|||
> |
|||
</li> |
|||
{% endfor %} |
|||
<li><a onclick="sidebarmake()">Add a wallet +</a></li> |
|||
<div id="sidebarmake"></div> |
|||
</ul> |
|||
</li> |
|||
|
|||
<li class="active treeview"> |
|||
<a href="#"> |
|||
<i class="fa fa-th"></i> <span>Extensions</span> |
|||
<i class="fa fa-angle-left pull-right"></i> |
|||
</a> |
|||
<ul class="treeview-menu"> |
|||
{% if user_ext[0][3] %} |
|||
<li> |
|||
<a href="{{ url_for('withdraw.index') }}?usr={{ user_ext[0][0]}}" |
|||
><i class="fa fa-plus"></i> LNURLw</a> |
|||
</li> |
|||
{% endif %} |
|||
<li> |
|||
<a href="{{ url_for('extensions') }}?usr={{ user }}" |
|||
>Manager </a> |
|||
</li> |
|||
|
|||
|
|||
</ul> |
|||
</li> |
|||
|
|||
{% endblock %} {% block body %} |
|||
<!-- Right side column. Contains the navbar and content of the page --> |
|||
<div class="content-wrapper"> |
|||
<!-- Content Header (Page header) --> |
|||
<section class="content-header"> |
|||
<h1> |
|||
Withdraw link maker |
|||
<small>powered by LNURL</small> |
|||
|
|||
</h1> |
|||
<ol class="breadcrumb"> |
|||
<li> |
|||
<a href="{{ url_for('wallet') }}?usr={{ user }}"><i class="fa fa-dashboard"></i> Home</a> |
|||
</li> |
|||
<li> |
|||
<a href="{{ url_for('extensions') }}?usr={{ user }}"><li class="fa fa-dashboard">Extensions</li></a> |
|||
</li> |
|||
<li> |
|||
<i class="active" class="fa fa-dashboard">Withdraw link maker</i> |
|||
</li> |
|||
</ol> |
|||
<br /><br /> |
|||
</section> |
|||
|
|||
<!-- Main content --> |
|||
<section class="content"> |
|||
<!-- Small boxes (Stat box) --> |
|||
<div class="row"> |
|||
|
|||
<div class="col-md-6"> |
|||
<!-- general form elements --> |
|||
<div class="box box-primary"> |
|||
<div class="box-header"> |
|||
<h3 class="box-title"> Make a link</h3> |
|||
</div><!-- /.box-header --> |
|||
<!-- form start --> |
|||
<form role="form"> |
|||
<div class="box-body"> |
|||
|
|||
<div class="form-group"> |
|||
<label for="exampleInputEmail1">Link title</label> |
|||
<input id="tit" type="text" pattern="^[A-Za-z]+$" class="form-control" > |
|||
</div> |
|||
<!-- select --> |
|||
<div class="form-group"> |
|||
<label>Select a wallet</label> |
|||
<select id="wal" class="form-control"> |
|||
<option></option> |
|||
{% for w in user_wallets %} |
|||
<option>{{w.name}}-{{w.id}}</option> |
|||
{% endfor %} |
|||
</select> |
|||
</div> |
|||
|
|||
|
|||
<div class="form-group"> |
|||
<label for="exampleInputEmail1">Max withdraw:</label> |
|||
<input id="maxamt" type="number" class="form-control" placeholder="1"></input> |
|||
</div> |
|||
<div class="form-group"> |
|||
<label for="exampleInputEmail1">Min withdraw:</label> |
|||
<input id="minamt" type="number" class="form-control" placeholder="1"></input> |
|||
</div> |
|||
<div class="form-group"> |
|||
<label for="exampleInputPassword1">Amount of uses:</label> |
|||
<input id="amt" type="number" class="form-control" placeholder="1"></input> |
|||
</div> |
|||
<div class="form-group"> |
|||
<label for="exampleInputPassword1">Time between withdrawals:</label> |
|||
<input id="tme" type="number" class="form-control" placeholder="0" max="86400"></input> |
|||
</div> |
|||
<div class="checkbox"> |
|||
<label> |
|||
<input id="uniq" type="checkbox"> Unique links |
|||
</label> |
|||
</div> |
|||
</div><!-- /.box-body --> |
|||
|
|||
<div class="box-footer"> |
|||
|
|||
<button onclick="postfau()" type="button" class="btn btn-info">Create link(s)</button><p style="color:red;" id="error"></p> |
|||
</div> |
|||
</form> |
|||
</div><!-- /.box --> |
|||
</div> |
|||
|
|||
|
|||
|
|||
|
|||
|
|||
|
|||
<div class="col-md-6"> |
|||
<!-- general form elements --> |
|||
<div class="box box-primary"> |
|||
<div class="box-header"> |
|||
<h3 class="box-title">Select a link</h3> |
|||
</div><!-- /.box-header --> |
|||
<form role="form"> |
|||
<div class="box-body"> |
|||
<div class="form-group"> |
|||
|
|||
<select class="form-control" id="fauselect" onchange="drawwithdraw()"> |
|||
<option></option> |
|||
{% for w in user_fau %} |
|||
<option id="{{w.uni}}" >{{w.tit}}-{{w.uni}}-{{w.inc}}</option> |
|||
{% endfor %} |
|||
</select> |
|||
</div> |
|||
|
|||
<center> <br/><div id="qrcode" style="width:340px" ></div><br/><div style="width:75%;word-wrap: break-word;" id="qrcodetxt" ></div></center> |
|||
</div> |
|||
</form> |
|||
|
|||
</div><!-- /.box --> |
|||
</div> |
|||
|
|||
|
|||
|
|||
</div> |
|||
|
|||
<div class="row"> |
|||
<div class="col-md-6"> |
|||
<div class="box"> |
|||
<div class="box-header"> |
|||
<h3 class="box-title">Withdraw links <b id="withdraws"></b></h3> |
|||
</div> |
|||
<!-- /.box-header --> |
|||
<div class="box-body no-padding"> |
|||
<table id="pagnation" class="table table-bordered table-striped"> |
|||
<tr> |
|||
<th>Title</th> |
|||
<th style="width:15%">Link/ID</th> |
|||
<th style="width:15%">Max Withdraw</th> |
|||
<th style="width:15%">No. uses</th> |
|||
<th style="width:15%">Wait</th> |
|||
<th style="width:15%">Wallet</th> |
|||
<th style="width:10%">Edit</th> |
|||
<th style="width:10%">Del</th> |
|||
</tr> |
|||
<tbody id="transactions"></tbody> |
|||
</table> |
|||
</div> |
|||
<!-- /.box-body --> |
|||
</div> |
|||
<!-- /.box --> |
|||
</div> |
|||
</div> |
|||
|
|||
|
|||
|
|||
<div id="editlink"></div> |
|||
|
|||
<!-- /.content --> |
|||
</section> |
|||
|
|||
<script> |
|||
window.user = {{ user | megajson | safe }} |
|||
window.user_wallets = {{ user_wallets | megajson | safe }} |
|||
window.user_ext = {{ user_ext | megajson | safe }} |
|||
window.user_fau = {{ user_fau | megajson | safe }} |
|||
|
|||
const user_fau = window.user_fau |
|||
console.log(user_fau) |
|||
|
|||
|
|||
function drawChart(user_fau) { |
|||
var transactionsHTML = '' |
|||
|
|||
for (var i = 0; i < user_fau.length; i++) { |
|||
var tx = user_fau[i] |
|||
console.log(tx.nme) |
|||
// make the transactions table |
|||
transactionsHTML = |
|||
"<tr><td style='width: 50%'>" + |
|||
tx.tit + |
|||
'</td><td >' + |
|||
"<a href='" + "{{ url_for('withdraw.display') }}?id=" + tx.uni + "'>" + tx.uni.substring(0, 4) + "...</a>" + |
|||
'</td><td>' + |
|||
tx.maxamt + |
|||
'</td><td>' + |
|||
tx.inc + |
|||
'</td><td>' + |
|||
tx.tme + |
|||
'</td><td>' + |
|||
"<a href='{{ url_for('wallet') }}?usr="+ user +"'>" + tx.uni.substring(0, 4) + "...</a>" + |
|||
'</td><td>' + |
|||
"<i onclick='editlink("+ i +")'' class='fa fa-edit'></i>" + |
|||
'</td><td>' + |
|||
"<b><a style='color:red;' href='" + "{{ url_for('withdraw.index') }}?del=" + tx.uni + "&usr=" + user +"'>" + "<i class='fa fa-trash'></i>" + "</a></b>" + |
|||
'</td></tr>' + |
|||
transactionsHTML |
|||
document.getElementById('transactions').innerHTML = transactionsHTML |
|||
|
|||
} |
|||
} |
|||
|
|||
if (user_fau.length) { |
|||
drawChart(user_fau) |
|||
} |
|||
|
|||
|
|||
//draws withdraw QR code |
|||
function drawwithdraw() { |
|||
|
|||
|
|||
walname = document.getElementById("fauselect").value |
|||
|
|||
thewithdraw = walname.split("-"); |
|||
console.log(window.location.hostname + "-" + thewithdraw[1]) |
|||
|
|||
getAjax("/withdraw/api/v1/lnurlencode/"+ window.location.hostname + "/" + thewithdraw[1], "filla", function(datab) { |
|||
if (JSON.parse(datab).STATUS == 'TRUE') { |
|||
console.log(JSON.parse(datab).STATUS) |
|||
lnurlfau = (JSON.parse(datab).LNURL) |
|||
|
|||
|
|||
new QRCode(document.getElementById('qrcode'), { |
|||
text: lnurlfau, |
|||
width: 300, |
|||
height: 300, |
|||
colorDark: '#000000', |
|||
colorLight: '#ffffff', |
|||
correctLevel: QRCode.CorrectLevel.M |
|||
}) |
|||
|
|||
if (thewithdraw[2] > 0){ |
|||
document.getElementById('qrcodetxt').innerHTML = lnurlfau |
|||
+ |
|||
"<a target='_blank' href='{{ url_for('withdraw.display') }}?id=" + thewithdraw[1] + "'><h4>Shareable link</h4></a>" + |
|||
"<a target='_blank' href='/withdraw/printwithdraw/" + window.location.hostname + "/?id=" + thewithdraw[1] + "'><h4>Print all withdraws</h4></a>" |
|||
document.getElementById("qrcode").style.backgroundColor = "white"; |
|||
document.getElementById("qrcode").style.padding = "20px"; |
|||
} |
|||
else{ |
|||
document.getElementById('qrcode').innerHTML = "" |
|||
document.getElementById('qrcodetxt').innerHTML = "<h1>No more uses left in link!</h1><br/><br/>" |
|||
|
|||
|
|||
} |
|||
|
|||
} |
|||
else { |
|||
|
|||
thewithdraw[1] = "Failed to build LNURL" |
|||
} |
|||
}) |
|||
|
|||
} |
|||
|
|||
|
|||
function postfau(){ |
|||
|
|||
wal = document.getElementById('wal').value |
|||
tit = document.getElementById('tit').value |
|||
amt = document.getElementById('amt').value |
|||
maxamt = document.getElementById('maxamt').value |
|||
minamt = document.getElementById('minamt').value |
|||
tme = document.getElementById('tme').value |
|||
uniq = document.getElementById('uniq').checked |
|||
|
|||
|
|||
if (tit == "") { |
|||
document.getElementById("error").innerHTML = "Only use letters in title" |
|||
return amt |
|||
} |
|||
if (wal == "") { |
|||
document.getElementById("error").innerHTML = "No wallet selected" |
|||
return amt |
|||
} |
|||
|
|||
if (isNaN(maxamt) || maxamt < 10 || maxamt > 1000000) { |
|||
document.getElementById("error").innerHTML = "Max 15 - 1000000 and must be higher than min" |
|||
return amt |
|||
} |
|||
if (isNaN(minamt) || minamt < 1 || minamt > 1000000 || minamt > maxamt) { |
|||
document.getElementById("error").innerHTML = "Min 1 - 1000000 and must be lower than max" |
|||
return amt |
|||
} |
|||
|
|||
if (isNaN(amt) || amt < 1 || amt > 1000) { |
|||
document.getElementById("error").innerHTML = "Amount of uses must be between 1 - 1000" |
|||
return amt |
|||
} |
|||
|
|||
if (isNaN(tme) || tme < 1 || tme > 86400) { |
|||
document.getElementById("error").innerHTML = "Max waiting time 1 day (86400 secs)" |
|||
return amt |
|||
} |
|||
|
|||
|
|||
postAjax( |
|||
"{{ url_for('withdraw.create') }}", |
|||
JSON.stringify({"tit": tit, "amt": amt, "maxamt": maxamt, "minamt": minamt, "tme": tme, "wal": wal, "usr": user, "uniq": uniq}), |
|||
"filla", |
|||
|
|||
function(data) { location.replace("{{ url_for('withdraw.index') }}?usr=" + user) |
|||
}) |
|||
|
|||
|
|||
} |
|||
|
|||
|
|||
function editlink(linknum){ |
|||
|
|||
faudetails = user_fau[linknum] |
|||
|
|||
console.log(faudetails) |
|||
wallpick = "" |
|||
|
|||
checkbox = "" |
|||
if (faudetails.uniq == 1){ |
|||
checkbox = "checked"} |
|||
|
|||
document.getElementById('editlink').innerHTML = "<div class='row'>"+ |
|||
"<div class='col-md-6'>"+ |
|||
" <!-- general form elements -->"+ |
|||
"<div class='box box-primary'>"+ |
|||
"<div class='box-header'>"+ |
|||
"<h3 class='box-title'> Edit: <i id='unid'>" + faudetails.tit + "-" + faudetails.uni + "</i> </h3>"+ |
|||
"<div class='box-tools pull-right'>" + |
|||
"<button class='btn btn-box-tool' data-widget='remove'><i class='fa fa-times'></i></button>" + |
|||
"</div>" + |
|||
" </div><!-- /.box-header -->"+ |
|||
" <!-- form start -->"+ |
|||
"<form role='form'>"+ |
|||
"<div class='box-body'>"+ |
|||
"<div class='col-sm-3 col-md-4'>"+ |
|||
"<div class='form-group'>"+ |
|||
"<label for='exampleInputEmail1'>Link title</label>"+ |
|||
"<input id='edittit' type='text' class='form-control' value='"+ |
|||
faudetails.tit + |
|||
"'></input> </div>"+ |
|||
" </div>"+ |
|||
" <div class='col-sm-4 col-md-4'>"+ |
|||
" <!-- select -->"+ |
|||
" <div class='form-group'>"+ |
|||
" <label>Select a wallet</label>"+ |
|||
"<select id='editwal' class='form-control'>"+ |
|||
" <option>" + faudetails.walnme + "-" + faudetails.wal + "</option>"+ |
|||
" {% for w in user_wallets %}"+ |
|||
|
|||
" <option>{{w.name}}-{{w.id}}</option>"+ |
|||
|
|||
" {% endfor %}"+ |
|||
" </select>"+ |
|||
" </div>"+ |
|||
" </div>"+ |
|||
" <div class='col-sm-3 col-md-4'>"+ |
|||
"<div class='form-group'>"+ |
|||
" <label for='exampleInputPassword1'>Time between withdrawals:</label>"+ |
|||
" <input id='edittme' type='number' class='form-control' placeholder='0' max='86400' value='"+ |
|||
faudetails.tme + |
|||
"'></input>"+ |
|||
"</div> </div>"+ |
|||
" <div class='col-sm-3 col-md-4'>"+ |
|||
"<div class='form-group'>"+ |
|||
"<label for='exampleInputEmail1'>Max withdraw:</label>"+ |
|||
" <input id='editmaxamt' type='number' class='form-control' placeholder='1' value='"+ |
|||
faudetails.maxamt + |
|||
"'></input>"+ |
|||
" </div></div>"+ |
|||
" <div class='col-sm-3 col-md-4'>"+ |
|||
" <div class='form-group'>"+ |
|||
" <label for='exampleInputEmail1'>Min withdraw:</label>"+ |
|||
" <input id='editminamt' type='number' class='form-control' placeholder='1' value='"+ |
|||
faudetails.minamt + |
|||
"'></input>"+ |
|||
" </div></div>"+ |
|||
" <div class='col-sm-3 col-md-4'>"+ |
|||
"<div class='form-group'>"+ |
|||
" <label for='exampleInputPassword1'>Amount of uses:</label>"+ |
|||
" <input id='editamt' type='number' class='form-control' placeholder='1' value='"+ |
|||
faudetails.inc + |
|||
"'></input>"+ |
|||
" </div> </div>"+ |
|||
" <div class='col-sm-3 col-md-4'>"+ |
|||
" <div class='checkbox'>"+ |
|||
"<label data-toggle='tooltip' title='Some tooltip text!'><input id='edituniq' type='checkbox' "+ |
|||
checkbox + |
|||
">"+ |
|||
"Unique links</label>"+ |
|||
"</div></div><!-- /.box-body -->"+ |
|||
" </div><br/>"+ |
|||
" <div class='box-footer'>"+ |
|||
" <button onclick='editlinkcont()' type='button' class='btn btn-info'>Edit link(s)</button><p style='color:red;' id='error2'></p>"+ |
|||
" </div></form></div><!-- /.box --></div></div>" |
|||
|
|||
|
|||
} |
|||
|
|||
function editlinkcont(){ |
|||
|
|||
unid = document.getElementById('unid').innerHTML |
|||
wal = document.getElementById('editwal').value |
|||
tit = document.getElementById('edittit').value |
|||
amt = document.getElementById('editamt').value |
|||
maxamt = document.getElementById('editmaxamt').value |
|||
minamt = document.getElementById('editminamt').value |
|||
tme = document.getElementById('edittme').value |
|||
uniq = document.getElementById('edituniq').checked |
|||
|
|||
|
|||
|
|||
if (tit == "") { |
|||
document.getElementById("error2").innerHTML = "Only use letters in title" |
|||
return amt |
|||
} |
|||
if (wal == "") { |
|||
document.getElementById("error2").innerHTML = "No wallet selected" |
|||
return amt |
|||
} |
|||
|
|||
if (isNaN(maxamt) || maxamt < 10 || maxamt > 1000000) { |
|||
document.getElementById("error2").innerHTML = "Max 10 - 1000000 and must be higher than min" |
|||
return amt |
|||
} |
|||
if (isNaN(minamt) || minamt < 1 || minamt > 1000000 || minamt > maxamt) { |
|||
document.getElementById("error2").innerHTML = "Min 1 - 1000000 and must be lower than max" |
|||
return amt |
|||
} |
|||
|
|||
if (isNaN(amt) || amt < 1 || amt > 1000) { |
|||
document.getElementById("error2").innerHTML = "Amount of uses must be between 1 - 1000" |
|||
return amt |
|||
} |
|||
|
|||
if (isNaN(tme) || tme < 1 || tme > 86400) { |
|||
document.getElementById("error2").innerHTML = "Max waiting time 1 day (86400 secs)" |
|||
return amt |
|||
} |
|||
|
|||
|
|||
postAjax( |
|||
"{{ url_for('withdraw.create') }}", |
|||
JSON.stringify({"id": unid, "tit": tit, "amt": amt, "maxamt": maxamt, "minamt": minamt, "tme": tme, "wal": wal, "usr": user, "uniq": uniq}), |
|||
"filla", |
|||
|
|||
function(data) { location.replace("{{ url_for('withdraw.index') }}?usr=" + user) |
|||
}) |
|||
|
|||
|
|||
|
|||
} |
|||
|
|||
</script> |
|||
</div> |
|||
{% endblock %} |
@ -0,0 +1,291 @@ |
|||
<!DOCTYPE html> |
|||
<html style="background-color:grey;"> |
|||
<head> |
|||
<meta charset="UTF-8" /> |
|||
<title>LNBits Wallet</title> |
|||
<meta |
|||
content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no" |
|||
name="viewport" |
|||
/> |
|||
<!-- Bootstrap 3.3.2 --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='bootstrap/css/bootstrap.min.css') }}" |
|||
/> |
|||
<!-- FontAwesome 4.3.0 --> |
|||
<link |
|||
href="https://maxcdn.bootstrapcdn.com/font-awesome/4.3.0/css/font-awesome.min.css" |
|||
rel="stylesheet" |
|||
type="text/css" |
|||
/> |
|||
<!-- Ionicons 2.0.0 --> |
|||
<link |
|||
href="https://code.ionicframework.com/ionicons/2.0.0/css/ionicons.min.css" |
|||
rel="stylesheet" |
|||
type="text/css" |
|||
/> |
|||
|
|||
<!-- Theme style --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='dist/css/AdminLTE.min.css') }}" |
|||
/> |
|||
<!-- AdminLTE Skins. Choose a skin from the css/skins |
|||
folder instead of downloading all of them to reduce the load. --> |
|||
|
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='dist/css/skins/_all-skins.min.css') }}" |
|||
/> |
|||
|
|||
<!-- Morris chart --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='plugins/morris/morris.css') }}" |
|||
/> |
|||
|
|||
<!-- jvectormap --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='plugins/jvectormap/jquery-jvectormap-1.2.2.css') }}" |
|||
/> |
|||
|
|||
<!-- bootstrap wysihtml5 - text editor --> |
|||
<link |
|||
rel="stylesheet" |
|||
media="screen" |
|||
href="{{ url_for('static', filename='plugins/bootstrap-wysihtml5/bootstrap3-wysihtml5.min.css') }}" |
|||
/> |
|||
|
|||
<!-- HTML5 Shim and Respond.js IE8 support of HTML5 elements and media queries --> |
|||
<!-- WARNING: Respond.js doesn't work if you view the page via file:// --> |
|||
<!--[if lt IE 9]> |
|||
<script src="https://oss.maxcdn.com/libs/html5shiv/3.7.0/html5shiv.js"></script> |
|||
<script src="https://oss.maxcdn.com/libs/respond.js/1.3.0/respond.min.js"></script> |
|||
<![endif]--> |
|||
|
|||
<style> |
|||
.small-box > .small-box-footer { |
|||
text-align: left; |
|||
padding-left: 10px; |
|||
} |
|||
|
|||
#loadingMessage { |
|||
text-align: center; |
|||
padding: 40px; |
|||
background-color: #eee; |
|||
} |
|||
|
|||
#canvas { |
|||
width: 100%; |
|||
} |
|||
|
|||
#output { |
|||
margin-top: 20px; |
|||
background: #eee; |
|||
padding: 10px; |
|||
padding-bottom: 0; |
|||
} |
|||
|
|||
#output div { |
|||
padding-bottom: 10px; |
|||
word-wrap: break-word; |
|||
} |
|||
|
|||
#noQRFound { |
|||
text-align: center; |
|||
} |
|||
.layout-boxed { |
|||
background: white; |
|||
} |
|||
</style> |
|||
|
|||
<!-- jQuery 2.1.3 --> |
|||
<script src="{{ url_for('static', filename='plugins/jQuery/jQuery-2.1.3.min.js') }}"></script> |
|||
<!-- jQuery UI 1.11.2 --> |
|||
<script |
|||
src="https://code.jquery.com/ui/1.11.2/jquery-ui.min.js" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Resolve conflict in jQuery UI tooltip with Bootstrap tooltip --> |
|||
<script> |
|||
$.widget.bridge('uibutton', $.ui.button) |
|||
</script> |
|||
<!-- Bootstrap 3.3.2 JS --> |
|||
<script |
|||
src="{{ url_for('static', filename='bootstrap/js/bootstrap.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Morris.js charts --> |
|||
<script src="https://cdnjs.cloudflare.com/ajax/libs/raphael/2.1.0/raphael-min.js"></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/morris/morris.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Sparkline --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/sparkline/jquery.sparkline.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- jvectormap --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jvectormap/jquery-jvectormap-1.2.2.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jvectormap/jquery-jvectormap-world-mill-en.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- jQuery Knob Chart --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/knob/jquery.knob.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Bootstrap WYSIHTML5 --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/bootstrap-wysihtml5/bootstrap3-wysihtml5.all.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- Slimscroll --> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/slimScroll/jquery.slimscroll.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<!-- FastClick --> |
|||
<script src="{{ url_for('static', filename='plugins/fastclick/fastclick.min.js') }}"></script> |
|||
<!-- AdminLTE App --> |
|||
<script |
|||
src="{{ url_for('static', filename='dist/js/app.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
<!-- AdminLTE dashboard demo (This is only for demo purposes) --> |
|||
<script |
|||
src="{{ url_for('static', filename='dist/js/pages/dashboard.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
<!-- AdminLTE for demo purposes --> |
|||
<script |
|||
src="{{ url_for('static', filename='dist/js/demo.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
|
|||
<script |
|||
src="{{ url_for('static', filename='plugins/datatables/jquery.dataTables.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<link |
|||
rel="stylesheet" |
|||
href="//cdnjs.cloudflare.com/ajax/libs/morris.js/0.5.1/morris.css" |
|||
/> |
|||
<script src="https://cdnjs.cloudflare.com/ajax/libs/raphael/2.1.0/raphael-min.js"></script> |
|||
<script src="https://cdnjs.cloudflare.com/ajax/libs/morris.js/0.5.1/morris.min.js"></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jscam/JS.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/jscam/qrcode.min.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/bolt11/decoder.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
<script |
|||
src="{{ url_for('static', filename='plugins/bolt11/utils.js') }}" |
|||
type="text/javascript" |
|||
></script> |
|||
</head> |
|||
<body class="skin-white layout-boxed sidebar-collapse sidebar-open"> |
|||
<div class="wrapper"> |
|||
|
|||
<style> |
|||
body { |
|||
width: 210mm; |
|||
/* to centre page on screen*/ |
|||
margin-left: auto; |
|||
margin-right: auto; |
|||
background-color:white; |
|||
} |
|||
.layout-boxed .wrapper { |
|||
box-shadow: none; |
|||
|
|||
} |
|||
</style> |
|||
</head> |
|||
<body> |
|||
|
|||
<div id="allqrs"></div> |
|||
|
|||
|
|||
</body> |
|||
|
|||
<script> |
|||
function getAjax(url, thekey, success) { |
|||
var xhr = window.XMLHttpRequest |
|||
? new XMLHttpRequest() |
|||
: new ActiveXObject('Microsoft.XMLHTTP') |
|||
xhr.open('GET', url, true) |
|||
xhr.onreadystatechange = function() { |
|||
if (xhr.readyState > 3 && xhr.status == 200) { |
|||
success(xhr.responseText) |
|||
} |
|||
} |
|||
xhr.setRequestHeader('Grpc-Metadata-macaroon', thekey) |
|||
xhr.setRequestHeader('Content-Type', 'application/json') |
|||
|
|||
xhr.send() |
|||
return xhr |
|||
} |
|||
|
|||
window.user_fau = {{ user_fau | megajson | safe }} |
|||
|
|||
|
|||
function drawwithdraw(data, id) { |
|||
|
|||
new QRCode(document.getElementById(id), { |
|||
text: data, |
|||
width: 120, |
|||
height: 120, |
|||
colorDark: '#000000', |
|||
colorLight: '#ffffff', |
|||
correctLevel: QRCode.CorrectLevel.M |
|||
} ) |
|||
|
|||
|
|||
} |
|||
lnurlar = {{ lnurlar|tojson }} |
|||
lnurlamt = user_fau["inc"] |
|||
console.log(user_fau) |
|||
allqr = "" |
|||
|
|||
for (i = 0; i < lnurlamt; i++) { |
|||
allqr += "<div style='float:left;padding:20px; background-image: url(/static/note.jpg); width: 500px;height: 248px;'><div style='width:120px;float:right;margin-top:-16px;margin-right:-19px;background-color: white;'><div id='qrcode" + i + "'></div><center><p>{{user_fau[7]}} FREE SATS! <br/> <small style='font-size: 52%;'>SCAN AND FOLLOW LINK OR<br/>USE LN BITCOIN WALLET</small></p></center></div></div>" |
|||
} |
|||
|
|||
document.getElementById("allqrs").innerHTML = allqr |
|||
|
|||
if (typeof lnurlar[1] != 'undefined'){ |
|||
for (i = 0; i < lnurlamt; i++) { |
|||
drawwithdraw(lnurlar[i], "qrcode" + i) |
|||
} |
|||
window.print() |
|||
} |
|||
else{ |
|||
for (i = 0; i < lnurlamt; i++) { |
|||
drawwithdraw(lnurlar[0], "qrcode" + i) |
|||
} |
|||
window.print() |
|||
} |
|||
|
|||
|
|||
|
|||
</script> |
|||
</html> |
@ -0,0 +1,160 @@ |
|||
import uuid |
|||
|
|||
from flask import jsonify, render_template, request, redirect, url_for |
|||
from lnurl import encode as lnurl_encode |
|||
from datetime import datetime |
|||
|
|||
from lnbits.db import open_db, open_ext_db |
|||
from lnbits.extensions.withdraw import withdraw_ext |
|||
|
|||
|
|||
@withdraw_ext.route("/") |
|||
def index(): |
|||
"""Main withdraw link page.""" |
|||
|
|||
usr = request.args.get("usr") |
|||
|
|||
if usr: |
|||
if not len(usr) > 20: |
|||
return redirect(url_for("home")) |
|||
|
|||
# Get all the data |
|||
with open_db() as db: |
|||
user_wallets = db.fetchall("SELECT * FROM wallets WHERE user = ?", (usr,)) |
|||
|
|||
with open_ext_db() as ext_db: |
|||
user_ext = ext_db.fetchall("SELECT * FROM overview WHERE user = ?", (usr,)) |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE usr = ?", (usr,)) |
|||
|
|||
# If del is selected by user from withdraw page, the withdraw link is to be deleted |
|||
faudel = request.args.get("del") |
|||
if faudel: |
|||
withdraw_ext_db.execute("DELETE FROM withdraws WHERE uni = ?", (faudel,)) |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE usr = ?", (usr,)) |
|||
|
|||
return render_template( |
|||
"withdraw/index.html", user_wallets=user_wallets, user=usr, user_ext=user_ext, user_fau=user_fau |
|||
) |
|||
|
|||
|
|||
@withdraw_ext.route("/create", methods=["GET", "POST"]) |
|||
def create(): |
|||
""".""" |
|||
|
|||
data = request.json |
|||
amt = data["amt"] |
|||
tit = data["tit"] |
|||
wal = data["wal"] |
|||
minamt = data["minamt"] |
|||
maxamt = data["maxamt"] |
|||
tme = data["tme"] |
|||
uniq = data["uniq"] |
|||
usr = data["usr"] |
|||
wall = wal.split("-") |
|||
|
|||
# Form validation |
|||
if ( |
|||
int(amt) < 0 |
|||
or not tit.replace(" ", "").isalnum() |
|||
or wal == "" |
|||
or int(minamt) < 0 |
|||
or int(maxamt) < 0 |
|||
or int(minamt) > int(maxamt) |
|||
or int(tme) < 0 |
|||
): |
|||
return jsonify({"ERROR": "FORM ERROR"}), 401 |
|||
|
|||
# If id that means its a link being edited, delet the record first |
|||
if "id" in data: |
|||
unid = data["id"].split("-") |
|||
uni = unid[1] |
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
withdraw_ext_db.execute("DELETE FROM withdraws WHERE uni = ?", (unid[1],)) |
|||
else: |
|||
uni = uuid.uuid4().hex |
|||
|
|||
# Randomiser for random QR option |
|||
rand = "" |
|||
if uniq > 0: |
|||
for x in range(0, int(amt)): |
|||
rand += uuid.uuid4().hex[0:5] + "," |
|||
else: |
|||
rand = uuid.uuid4().hex[0:5] + "," |
|||
|
|||
with open_db() as dbb: |
|||
user_wallets = dbb.fetchall("SELECT * FROM wallets WHERE user = ? AND id = ?", (usr, wall[1],)) |
|||
if not user_wallets: |
|||
return jsonify({"ERROR": "NO WALLET USER"}), 401 |
|||
|
|||
# Get time |
|||
dt = datetime.now() |
|||
seconds = dt.timestamp() |
|||
|
|||
# Add to DB |
|||
with open_ext_db("withdraw") as db: |
|||
db.execute( |
|||
"INSERT OR IGNORE INTO withdraws (usr, wal, walnme, adm, uni, tit, maxamt, minamt, spent, inc, tme, uniq, withdrawals, tmestmp, rand) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)", |
|||
( |
|||
usr, |
|||
wall[1], |
|||
user_wallets[0][1], |
|||
user_wallets[0][3], |
|||
uni, |
|||
tit, |
|||
maxamt, |
|||
minamt, |
|||
0, |
|||
amt, |
|||
tme, |
|||
uniq, |
|||
0, |
|||
seconds, |
|||
rand, |
|||
), |
|||
) |
|||
|
|||
# Get updated records |
|||
with open_ext_db() as ext_db: |
|||
user_ext = ext_db.fetchall("SELECT * FROM overview WHERE user = ?", (usr,)) |
|||
if not user_ext: |
|||
return jsonify({"ERROR": "NO WALLET USER"}), 401 |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE usr = ?", (usr,)) |
|||
if not user_fau: |
|||
return jsonify({"ERROR": "NO WALLET USER"}), 401 |
|||
|
|||
return render_template( |
|||
"withdraw/index.html", user_wallets=user_wallets, user=usr, user_ext=user_ext, user_fau=user_fau |
|||
) |
|||
|
|||
|
|||
@withdraw_ext.route("/display", methods=["GET", "POST"]) |
|||
def display(): |
|||
"""Simple shareable link.""" |
|||
fauid = request.args.get("id") |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE uni = ?", (fauid,)) |
|||
|
|||
return render_template("withdraw/display.html", user_fau=user_fau,) |
|||
|
|||
|
|||
@withdraw_ext.route("/print/<urlstr>/", methods=["GET", "POST"]) |
|||
def print_qr(urlstr): |
|||
"""Simple printable page of links.""" |
|||
fauid = request.args.get("id") |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE uni = ?", (fauid,)) |
|||
randar = user_fau[0][15].split(",") |
|||
randar = randar[:-1] |
|||
lnurlar = [] |
|||
|
|||
for d in range(len(randar)): |
|||
url = url_for("withdraw.api_lnurlfetch", _external=True, urlstr=urlstr, parstr=fauid, rand=randar[d]) |
|||
lnurlar.append(lnurl_encode(url.replace("http", "https"))) |
|||
|
|||
return render_template("withdraw/print.html", lnurlar=lnurlar, user_fau=user_fau[0],) |
@ -0,0 +1,116 @@ |
|||
import uuid |
|||
import json |
|||
import requests |
|||
|
|||
from flask import jsonify, request, url_for |
|||
from lnurl import LnurlWithdrawResponse, encode as lnurl_encode |
|||
from datetime import datetime |
|||
|
|||
from lnbits.db import open_ext_db |
|||
from lnbits.extensions.withdraw import withdraw_ext |
|||
|
|||
|
|||
@withdraw_ext.route("/api/v1/lnurlencode/<urlstr>/<parstr>", methods=["GET"]) |
|||
def api_lnurlencode(urlstr, parstr): |
|||
"""Returns encoded LNURL if web url and parameter gieven.""" |
|||
|
|||
if not urlstr: |
|||
return jsonify({"status": "FALSE"}), 200 |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE uni = ?", (parstr,)) |
|||
randar = user_fau[0][15].split(",") |
|||
# randar = randar[:-1] |
|||
# If "Unique links" selected get correct rand, if not there is only one rand |
|||
if user_fau[0][12] > 0: |
|||
rand = randar[user_fau[0][10] - 2] |
|||
else: |
|||
rand = randar[0] |
|||
|
|||
url = url_for("withdraw.api_lnurlfetch", _external=True, urlstr=urlstr, parstr=parstr, rand=rand) |
|||
|
|||
return jsonify({"status": "TRUE", "lnurl": lnurl_encode(url.replace("http", "https"))}), 200 |
|||
|
|||
|
|||
@withdraw_ext.route("/api/v1/lnurlfetch/<urlstr>/<parstr>/<rand>", methods=["GET"]) |
|||
def api_lnurlfetch(parstr, urlstr, rand): |
|||
"""Returns LNURL json.""" |
|||
|
|||
if not parstr: |
|||
return jsonify({"status": "FALSE", "ERROR": "NO WALL ID"}), 200 |
|||
|
|||
if not urlstr: |
|||
|
|||
return jsonify({"status": "FALSE", "ERROR": "NO URL"}), 200 |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE uni = ?", (parstr,)) |
|||
k1str = uuid.uuid4().hex |
|||
withdraw_ext_db.execute("UPDATE withdraws SET withdrawals = ? WHERE uni = ?", (k1str, parstr,)) |
|||
|
|||
res = LnurlWithdrawResponse( |
|||
callback=url_for("withdraw.api_lnurlwithdraw", _external=True, rand=rand).replace("http", "https"), |
|||
k1=k1str, |
|||
min_withdrawable=user_fau[0][8] * 1000, |
|||
max_withdrawable=user_fau[0][7] * 1000, |
|||
default_description="LNbits LNURL withdraw", |
|||
) |
|||
|
|||
return res.json(), 200 |
|||
|
|||
|
|||
@withdraw_ext.route("/api/v1/lnurlwithdraw/<rand>/", methods=["GET"]) |
|||
def api_lnurlwithdraw(rand): |
|||
"""Pays invoice if passed k1 invoice and rand.""" |
|||
|
|||
k1 = request.args.get("k1") |
|||
pr = request.args.get("pr") |
|||
|
|||
if not k1: |
|||
return jsonify({"status": "FALSE", "ERROR": "NO k1"}), 200 |
|||
|
|||
if not pr: |
|||
return jsonify({"status": "FALSE", "ERROR": "NO PR"}), 200 |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE withdrawals = ?", (k1,)) |
|||
|
|||
if not user_fau: |
|||
return jsonify({"status": "ERROR", "reason": "NO AUTH"}), 400 |
|||
|
|||
if user_fau[0][10] < 1: |
|||
return jsonify({"status": "ERROR", "reason": "withdraw SPENT"}), 400 |
|||
|
|||
# Check withdraw time |
|||
dt = datetime.now() |
|||
seconds = dt.timestamp() |
|||
secspast = seconds - user_fau[0][14] |
|||
|
|||
if secspast < user_fau[0][11]: |
|||
return jsonify({"status": "ERROR", "reason": "WAIT " + str(int(user_fau[0][11] - secspast)) + "s"}), 400 |
|||
|
|||
randar = user_fau[0][15].split(",") |
|||
if rand not in randar: |
|||
return jsonify({"status": "ERROR", "reason": "BAD AUTH"}), 400 |
|||
if len(randar) > 2: |
|||
randar.remove(rand) |
|||
randstr = ",".join(randar) |
|||
|
|||
# Update time and increments |
|||
upinc = int(user_fau[0][10]) - 1 |
|||
withdraw_ext_db.execute( |
|||
"UPDATE withdraws SET inc = ?, rand = ?, tmestmp = ? WHERE withdrawals = ?", (upinc, randstr, seconds, k1,) |
|||
) |
|||
|
|||
header = {"Content-Type": "application/json", "Grpc-Metadata-macaroon": str(user_fau[0][4])} |
|||
data = {"payment_request": pr} |
|||
r = requests.post(url=url_for("api_transactions", _external=True), headers=header, data=json.dumps(data)) |
|||
r_json = r.json() |
|||
|
|||
if "ERROR" in r_json: |
|||
return jsonify({"status": "ERROR", "reason": r_json["ERROR"]}), 400 |
|||
|
|||
with open_ext_db("withdraw") as withdraw_ext_db: |
|||
user_fau = withdraw_ext_db.fetchall("SELECT * FROM withdraws WHERE withdrawals = ?", (k1,)) |
|||
|
|||
return jsonify({"status": "OK"}), 200 |
@ -1,15 +1,15 @@ |
|||
import os |
|||
|
|||
from .wallets import LntxbotWallet # OR LndWallet OR OpennodeWallet |
|||
from .wallets import OpenNodeWallet # OR LndWallet OR OpennodeWallet |
|||
|
|||
#WALLET = OpenNodeWallet(endpoint=os.getenv("OPENNODE_API_ENDPOINT"),admin_key=os.getenv("OPENNODE_ADMIN_KEY"),invoice_key=os.getenv("OPENNODE_INVOICE_KEY")) |
|||
WALLET = LntxbotWallet(endpoint=os.getenv("LNTXBOT_API_ENDPOINT"),admin_key=os.getenv("LNTXBOT_ADMIN_KEY"),invoice_key=os.getenv("LNTXBOT_INVOICE_KEY")) |
|||
WALLET = OpenNodeWallet(endpoint=os.getenv("OPENNODE_API_ENDPOINT"),admin_key=os.getenv("OPENNODE_ADMIN_KEY"),invoice_key=os.getenv("OPENNODE_INVOICE_KEY")) |
|||
#WALLET = LntxbotWallet(endpoint=os.getenv("LNTXBOT_API_ENDPOINT"),admin_key=os.getenv("LNTXBOT_ADMIN_KEY"),invoice_key=os.getenv("LNTXBOT_INVOICE_KEY")) |
|||
#WALLET = LndWallet(endpoint=os.getenv("LND_API_ENDPOINT"),admin_macaroon=os.getenv("LND_ADMIN_MACAROON"),invoice_macaroon=os.getenv("LND_INVOICE_MACAROON"),read_macaroon=os.getenv("LND_READ_MACAROON")) |
|||
#WALLET = LNPayWallet(endpoint=os.getenv("LNPAY_API_ENDPOINT"),admin_key=os.getenv("LNPAY_ADMIN_KEY"),invoice_key=os.getenv("LNPAY_INVOICE_KEY"),api_key=os.getenv("LNPAY_API_KEY"),read_key=os.getenv("LNPAY_READ_KEY")) |
|||
|
|||
|
|||
LNBITS_PATH = os.path.dirname(os.path.realpath(__file__)) |
|||
DATABASE_PATH = os.getenv("DATABASE_PATH") or os.path.join(LNBITS_PATH, "data", "database.sqlite3") |
|||
DEFAULT_USER_WALLET_NAME = os.getenv("DEFAULT_USER_WALLET_NAME") or "Bitcoin LN Wallet" |
|||
DATABASE_PATH = os.getenv("DATABASE_PATH", os.path.join(LNBITS_PATH, "data", "database.sqlite3")) |
|||
|
|||
FEE_RESERVE = float(os.getenv("FEE_RESERVE") or 0) |
|||
DEFAULT_USER_WALLET_NAME = os.getenv("DEFAULT_USER_WALLET_NAME", "Bitcoin LN Wallet") |
|||
FEE_RESERVE = float(os.getenv("FEE_RESERVE", 0)) |
|||
|
After Width: | Height: | Size: 79 KiB |
After Width: | Height: | Size: 1.2 MiB |
After Width: | Height: | Size: 14 KiB |
After Width: | Height: | Size: 59 KiB |
@ -0,0 +1,165 @@ |
|||
<!-- @format --> |
|||
|
|||
{% extends "base.html" %} {% block messages %} |
|||
<a href="#" class="dropdown-toggle" data-toggle="dropdown"> |
|||
<i class="fa fa-bell-o"></i> |
|||
<span class="label label-danger">!</span> |
|||
</a> |
|||
<ul class="dropdown-menu"> |
|||
<li class="header"><b>Instant wallet, bookmark to save</b></li> |
|||
<li></li> |
|||
</ul> |
|||
{% endblock %} {% block menuitems %} |
|||
<li class="treeview"> |
|||
<a href="#"> |
|||
<i class="fa fa-bitcoin"></i> <span>Wallets</span> |
|||
<i class="fa fa-angle-left pull-right"></i> |
|||
</a> |
|||
<ul class="treeview-menu"> |
|||
{% for w in user_wallets %} |
|||
<li> |
|||
<a href="wallet?wal={{ w.id }}&usr={{ w.user }}" |
|||
><i class="fa fa-bolt"></i> {{ w.name }}</a |
|||
> |
|||
</li> |
|||
{% endfor %} |
|||
<li><a onclick="sidebarmake()">Add a wallet +</a></li> |
|||
<div id="sidebarmake"></div> |
|||
</ul> |
|||
</li> |
|||
|
|||
<li class="active treeview"> |
|||
<a href="#"> |
|||
<i class="fa fa-th"></i> <span>Extensions</span> |
|||
<i class="fa fa-angle-left pull-right"></i> |
|||
</a> |
|||
<ul class="treeview-menu"> |
|||
|
|||
|
|||
{% if user_ext[0][1] %} |
|||
<li> |
|||
<a href="lnevents?usr={{ user_ext[0][0]}}" |
|||
><i class="fa fa-plus"></i> LNEvents</a> |
|||
</li> |
|||
{% endif %} |
|||
|
|||
{% if user_ext[0][2] %} |
|||
<li> |
|||
<a href="lnjoust?usr={{ user_ext[0][0]}}" |
|||
><i class="fa fa-plus"></i> LNJoust</a> |
|||
</li> |
|||
{% endif %} |
|||
|
|||
{% if user_ext[0][3] %} |
|||
<li> |
|||
<a href="withdraw?usr={{ user_ext[0][0]}}" |
|||
><i class="fa fa-plus"></i> LNURLw</a> |
|||
</li> |
|||
{% endif %} |
|||
<li> |
|||
<a href="extensions?usr={{ user }}" |
|||
>Manager </a> |
|||
</li> |
|||
|
|||
|
|||
</ul> |
|||
</li> |
|||
|
|||
{% endblock %} {% block body %} |
|||
<!-- Right side column. Contains the navbar and content of the page --> |
|||
<div class="content-wrapper"> |
|||
<!-- Content Header (Page header) --> |
|||
<section class="content-header"> |
|||
<h1> |
|||
Wallet |
|||
<small |
|||
>Control panel |
|||
<div id="wonga"></div |
|||
></small> |
|||
</h1> |
|||
<ol class="breadcrumb"> |
|||
<li> |
|||
<a href="#"><i class="fa fa-dashboard"></i> Home</a> |
|||
</li> |
|||
<li class="active">Extensions</li> |
|||
|
|||
</ol> |
|||
<br /><br /> |
|||
<div class="alert alert-danger alert-dismissable"> |
|||
<h4> |
|||
Bookmark to save your wallet. Wallet is in BETA, use with caution. |
|||
</h4> |
|||
</div> |
|||
</section> |
|||
|
|||
<!-- Main content --> |
|||
<section class="content"> |
|||
<!-- Small boxes (Stat box) --> |
|||
<div class="row"> |
|||
|
|||
|
|||
{% if not user_ext[0][3] %} |
|||
<div class="col-lg-3 col-xs-6"> |
|||
<!-- small box --> |
|||
<div class="small-box bg-blue"> |
|||
<div class="inner"> |
|||
<h3> |
|||
LNURLw |
|||
</h3> |
|||
<p> |
|||
Make LNURL withdraw links |
|||
</p> |
|||
</div> |
|||
<div class="icon"> |
|||
<i class="ion ion-beer"></i> |
|||
</div> |
|||
<a href="extensions?usr={{user}}&withdraw=1" class="small-box-footer"> |
|||
Activate <i class="fa fa-arrow-circle-right"></i> |
|||
</a> |
|||
</div> |
|||
</div><!-- ./col --> |
|||
{% else %} |
|||
|
|||
<div class="col-lg-3 col-xs-6"> |
|||
<!-- small box --> |
|||
|
|||
<div class="small-box bg-blue"> |
|||
|
|||
<div class="inner"> |
|||
<a href="withdraw?usr={{user}}" style="color: inherit;"> |
|||
<h3> |
|||
LNURLw |
|||
</h3> |
|||
<p> |
|||
Make LNURL withdraw links |
|||
</p> |
|||
|
|||
</div> |
|||
<div class="icon"> |
|||
|
|||
<i class="ion ion-beer"></i> |
|||
|
|||
</div> |
|||
</a> |
|||
<a href="extensions?usr={{user}}&withdraw=0" class="small-box-footer"> |
|||
Deactivate <i class="fa fa-arrow-circle-right"></i> |
|||
</a> |
|||
|
|||
</div> |
|||
|
|||
</div><!-- ./col --> |
|||
|
|||
{% endif %} |
|||
</div> |
|||
|
|||
<!-- /.content --> |
|||
</section> |
|||
|
|||
<script> |
|||
window.user = {{ user | megajson | safe }} |
|||
window.user_wallets = {{ user_wallets | megajson | safe }} |
|||
window.user_ext = {{ user_ext | megajson | safe }} |
|||
|
|||
</script> |
|||
</div> |
|||
{% endblock %} |
After Width: | Height: | Size: 79 KiB |
Loading…
Reference in new issue