Merge pull request #295 from lnbits/lnticketsflat
Returns lnticket to polling, until listeners are properly tested, and adds a flat rate
This commit is contained in:
commit
1359c1ee39
7 changed files with 160 additions and 71 deletions
|
|
@ -49,7 +49,7 @@ async def set_ticket_paid(payment_hash: str) -> Tickets:
|
||||||
amount = formdata.amountmade + row[7]
|
amount = formdata.amountmade + row[7]
|
||||||
await db.execute(
|
await db.execute(
|
||||||
"""
|
"""
|
||||||
UPDATE lnticket.form
|
UPDATE lnticket.forms
|
||||||
SET amountmade = ?
|
SET amountmade = ?
|
||||||
WHERE id = ?
|
WHERE id = ?
|
||||||
""",
|
""",
|
||||||
|
|
@ -108,34 +108,35 @@ async def create_form(
|
||||||
name: str,
|
name: str,
|
||||||
webhook: Optional[str] = None,
|
webhook: Optional[str] = None,
|
||||||
description: str,
|
description: str,
|
||||||
costpword: int,
|
amount: int,
|
||||||
|
flatrate: int,
|
||||||
) -> Forms:
|
) -> Forms:
|
||||||
form_id = urlsafe_short_hash()
|
form_id = urlsafe_short_hash()
|
||||||
await db.execute(
|
await db.execute(
|
||||||
"""
|
"""
|
||||||
INSERT INTO lnticket.form (id, wallet, name, webhook, description, costpword, amountmade)
|
INSERT INTO lnticket.forms (id, wallet, name, webhook, description, flatrate, amount, amountmade)
|
||||||
VALUES (?, ?, ?, ?, ?, ?, ?)
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
||||||
""",
|
""",
|
||||||
(form_id, wallet, name, webhook, description, costpword, 0),
|
(form_id, wallet, name, webhook, description, flatrate, amount, 0),
|
||||||
)
|
)
|
||||||
|
|
||||||
form = await get_form(form_id)
|
form = await get_form(form_id)
|
||||||
assert form, "Newly created form couldn't be retrieved"
|
assert form, "Newly created forms couldn't be retrieved"
|
||||||
return form
|
return form
|
||||||
|
|
||||||
|
|
||||||
async def update_form(form_id: str, **kwargs) -> Forms:
|
async def update_form(form_id: str, **kwargs) -> Forms:
|
||||||
q = ", ".join([f"{field[0]} = ?" for field in kwargs.items()])
|
q = ", ".join([f"{field[0]} = ?" for field in kwargs.items()])
|
||||||
await db.execute(
|
await db.execute(
|
||||||
f"UPDATE lnticket.form SET {q} WHERE id = ?", (*kwargs.values(), form_id)
|
f"UPDATE lnticket.forms SET {q} WHERE id = ?", (*kwargs.values(), form_id)
|
||||||
)
|
)
|
||||||
row = await db.fetchone("SELECT * FROM lnticket.form WHERE id = ?", (form_id,))
|
row = await db.fetchone("SELECT * FROM lnticket.forms WHERE id = ?", (form_id,))
|
||||||
assert row, "Newly updated form couldn't be retrieved"
|
assert row, "Newly updated form couldn't be retrieved"
|
||||||
return Forms(**row)
|
return Forms(**row)
|
||||||
|
|
||||||
|
|
||||||
async def get_form(form_id: str) -> Optional[Forms]:
|
async def get_form(form_id: str) -> Optional[Forms]:
|
||||||
row = await db.fetchone("SELECT * FROM lnticket.form WHERE id = ?", (form_id,))
|
row = await db.fetchone("SELECT * FROM lnticket.forms WHERE id = ?", (form_id,))
|
||||||
return Forms(**row) if row else None
|
return Forms(**row) if row else None
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -145,11 +146,11 @@ async def get_forms(wallet_ids: Union[str, List[str]]) -> List[Forms]:
|
||||||
|
|
||||||
q = ",".join(["?"] * len(wallet_ids))
|
q = ",".join(["?"] * len(wallet_ids))
|
||||||
rows = await db.fetchall(
|
rows = await db.fetchall(
|
||||||
f"SELECT * FROM lnticket.form WHERE wallet IN ({q})", (*wallet_ids,)
|
f"SELECT * FROM lnticket.forms WHERE wallet IN ({q})", (*wallet_ids,)
|
||||||
)
|
)
|
||||||
|
|
||||||
return [Forms(**row) for row in rows]
|
return [Forms(**row) for row in rows]
|
||||||
|
|
||||||
|
|
||||||
async def delete_form(form_id: str) -> None:
|
async def delete_form(form_id: str) -> None:
|
||||||
await db.execute("DELETE FROM lnticket.form WHERE id = ?", (form_id,))
|
await db.execute("DELETE FROM lnticket.forms WHERE id = ?", (form_id,))
|
||||||
|
|
|
||||||
|
|
@ -145,3 +145,58 @@ async def m003_changed(db):
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
await db.execute("DROP TABLE lnticket.forms")
|
await db.execute("DROP TABLE lnticket.forms")
|
||||||
|
|
||||||
|
|
||||||
|
async def m004_changed(db):
|
||||||
|
|
||||||
|
await db.execute(
|
||||||
|
"""
|
||||||
|
CREATE TABLE lnticket.forms (
|
||||||
|
id TEXT PRIMARY KEY,
|
||||||
|
wallet TEXT NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
webhook TEXT,
|
||||||
|
description TEXT NOT NULL,
|
||||||
|
flatrate INTEGER DEFAULT 0,
|
||||||
|
amount INTEGER NOT NULL,
|
||||||
|
amountmade INTEGER NOT NULL,
|
||||||
|
time TIMESTAMP NOT NULL DEFAULT """
|
||||||
|
+ db.timestamp_now
|
||||||
|
+ """
|
||||||
|
);
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
|
||||||
|
for row in [list(row) for row in await db.fetchall("SELECT * FROM lnticket.form")]:
|
||||||
|
usescsv = ""
|
||||||
|
|
||||||
|
for i in range(row[5]):
|
||||||
|
if row[7]:
|
||||||
|
usescsv += "," + str(i + 1)
|
||||||
|
else:
|
||||||
|
usescsv += "," + str(1)
|
||||||
|
usescsv = usescsv[1:]
|
||||||
|
await db.execute(
|
||||||
|
"""
|
||||||
|
INSERT INTO lnticket.forms (
|
||||||
|
id,
|
||||||
|
wallet,
|
||||||
|
name,
|
||||||
|
webhook,
|
||||||
|
description,
|
||||||
|
amount,
|
||||||
|
amountmade,
|
||||||
|
)
|
||||||
|
VALUES (?, ?, ?, ?, ?, ?, ?)
|
||||||
|
""",
|
||||||
|
(
|
||||||
|
row[0],
|
||||||
|
row[1],
|
||||||
|
row[2],
|
||||||
|
row[3],
|
||||||
|
row[4],
|
||||||
|
row[5],
|
||||||
|
row[6],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
await db.execute("DROP TABLE lnticket.form")
|
||||||
|
|
|
||||||
|
|
@ -7,7 +7,8 @@ class Forms(NamedTuple):
|
||||||
name: str
|
name: str
|
||||||
webhook: str
|
webhook: str
|
||||||
description: str
|
description: str
|
||||||
costpword: int
|
amount: int
|
||||||
|
flatrate: int
|
||||||
amountmade: int
|
amountmade: int
|
||||||
time: int
|
time: int
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -23,11 +23,20 @@
|
||||||
label="Your email (optional, if you want a reply)"
|
label="Your email (optional, if you want a reply)"
|
||||||
></q-input>
|
></q-input>
|
||||||
<q-input
|
<q-input
|
||||||
|
v-if="flatrate"
|
||||||
filled
|
filled
|
||||||
dense
|
dense
|
||||||
v-model.number="formDialog.data.text"
|
v-model.number="formDialog.data.text"
|
||||||
type="textarea"
|
type="textarea"
|
||||||
label="{{ form_costpword }} sats per word"
|
label="{{ form_amount }} sats"
|
||||||
|
></q-input>
|
||||||
|
<q-input
|
||||||
|
v-else
|
||||||
|
filled
|
||||||
|
dense
|
||||||
|
v-model.number="formDialog.data.text"
|
||||||
|
type="textarea"
|
||||||
|
label="{{ form_amount }} sats per word"
|
||||||
></q-input>
|
></q-input>
|
||||||
<p>{% raw %}{{amountWords}}{% endraw %}</p>
|
<p>{% raw %}{{amountWords}}{% endraw %}</p>
|
||||||
<div class="row q-mt-lg">
|
<div class="row q-mt-lg">
|
||||||
|
|
@ -87,13 +96,16 @@
|
||||||
return {
|
return {
|
||||||
paymentReq: null,
|
paymentReq: null,
|
||||||
redirectUrl: null,
|
redirectUrl: null,
|
||||||
|
flatrate: parseInt('{{form_flatrate}}'),
|
||||||
formDialog: {
|
formDialog: {
|
||||||
show: false,
|
show: false,
|
||||||
data: {
|
data: {
|
||||||
name: '',
|
name: '',
|
||||||
email: '',
|
email: '',
|
||||||
text: ''
|
text: ''
|
||||||
}
|
},
|
||||||
|
dismissMsg: null,
|
||||||
|
paymentChecker: null
|
||||||
},
|
},
|
||||||
receive: {
|
receive: {
|
||||||
show: false,
|
show: false,
|
||||||
|
|
@ -108,21 +120,24 @@
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
amountWords() {
|
amountWords() {
|
||||||
var regex = /\s+/gi
|
if (this.formDialog.data.text.length == '') {
|
||||||
var nwords = this.formDialog.data.text
|
|
||||||
.trim()
|
|
||||||
.replace(regex, ' ')
|
|
||||||
.split(' ').length
|
|
||||||
var sats = nwords * parseInt('{{ form_costpword }}')
|
|
||||||
if (sats === parseInt('{{ form_costpword }}')) {
|
|
||||||
return '0 Sats to pay'
|
return '0 Sats to pay'
|
||||||
} else {
|
} else {
|
||||||
|
var regex = /\s+/gi
|
||||||
|
var nwords = this.formDialog.data.text
|
||||||
|
.trim()
|
||||||
|
.replace(regex, ' ')
|
||||||
|
.split(' ').length
|
||||||
|
if (this.flatrate) {
|
||||||
|
var sats = parseInt('{{ form_amount }}')
|
||||||
|
} else {
|
||||||
|
var sats = nwords * parseInt('{{ form_amount }}')
|
||||||
|
}
|
||||||
this.formDialog.data.sats = sats
|
this.formDialog.data.sats = sats
|
||||||
return sats + ' Sats to pay'
|
return sats + ' Sats to pay'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
resetForm: function (e) {
|
resetForm: function (e) {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
|
|
@ -130,39 +145,9 @@
|
||||||
this.formDialog.data.email = ''
|
this.formDialog.data.email = ''
|
||||||
this.formDialog.data.text = ''
|
this.formDialog.data.text = ''
|
||||||
},
|
},
|
||||||
|
|
||||||
closeReceiveDialog: function () {
|
|
||||||
var checker = this.startPaymentNotifier
|
|
||||||
dismissMsg()
|
|
||||||
|
|
||||||
setTimeout(function () {}, 10000)
|
|
||||||
},
|
|
||||||
startPaymentNotifier() {
|
|
||||||
this.cancelListener()
|
|
||||||
|
|
||||||
this.cancelListener = LNbits.events.onInvoicePaid(
|
|
||||||
this.wallet,
|
|
||||||
payment => {
|
|
||||||
this.receive = {
|
|
||||||
show: false,
|
|
||||||
status: 'complete',
|
|
||||||
paymentReq: null
|
|
||||||
}
|
|
||||||
dismissMsg()
|
|
||||||
|
|
||||||
this.formDialog.data.name = ''
|
|
||||||
this.formDialog.data.email = ''
|
|
||||||
this.formDialog.data.text = ''
|
|
||||||
this.$q.notify({
|
|
||||||
type: 'positive',
|
|
||||||
message: 'Sent, thank you!',
|
|
||||||
icon: 'thumb_up'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
)
|
|
||||||
},
|
|
||||||
Invoice: function () {
|
Invoice: function () {
|
||||||
var self = this
|
var self = this
|
||||||
|
var dialog = this.formDialog
|
||||||
axios
|
axios
|
||||||
.post('/lnticket/api/v1/tickets/{{ form_id }}', {
|
.post('/lnticket/api/v1/tickets/{{ form_id }}', {
|
||||||
form: '{{ form_id }}',
|
form: '{{ form_id }}',
|
||||||
|
|
@ -175,16 +160,34 @@
|
||||||
self.paymentReq = response.data.payment_request
|
self.paymentReq = response.data.payment_request
|
||||||
self.paymentCheck = response.data.payment_hash
|
self.paymentCheck = response.data.payment_hash
|
||||||
|
|
||||||
dismissMsg = self.$q.notify({
|
dialog.dismissMsg = self.$q.notify({
|
||||||
timeout: 0,
|
timeout: 0,
|
||||||
message: 'Waiting for payment...'
|
message: 'Waiting for payment...'
|
||||||
})
|
})
|
||||||
|
|
||||||
self.receive = {
|
self.receive = {
|
||||||
show: true,
|
show: true,
|
||||||
status: 'pending',
|
status: 'pending',
|
||||||
paymentReq: self.paymentReq
|
paymentReq: self.paymentReq
|
||||||
}
|
}
|
||||||
|
dialog.paymentChecker = setInterval(function () {
|
||||||
|
axios
|
||||||
|
.get('/lnticket/api/v1/tickets/' + response.data.payment_hash)
|
||||||
|
.then(function (res) {
|
||||||
|
if (res.data.paid) {
|
||||||
|
clearInterval(dialog.paymentChecker)
|
||||||
|
dialog.dismissMsg()
|
||||||
|
self.receive.show = false
|
||||||
|
self.formDialog.data.name = ''
|
||||||
|
self.formDialog.data.email = ''
|
||||||
|
self.formDialog.data.text = ''
|
||||||
|
self.$q.notify({
|
||||||
|
type: 'positive',
|
||||||
|
message: 'Sats received, thanks!',
|
||||||
|
icon: 'thumb_up'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, 2000)
|
||||||
})
|
})
|
||||||
.catch(function (error) {
|
.catch(function (error) {
|
||||||
LNbits.utils.notifyApiError(error)
|
LNbits.utils.notifyApiError(error)
|
||||||
|
|
@ -193,7 +196,6 @@
|
||||||
},
|
},
|
||||||
created() {
|
created() {
|
||||||
this.wallet.inkey = '{{form_wallet}}'
|
this.wallet.inkey = '{{form_wallet}}'
|
||||||
this.startPaymentNotifier()
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
|
||||||
|
|
@ -206,13 +206,25 @@
|
||||||
type="textarea"
|
type="textarea"
|
||||||
label="Description "
|
label="Description "
|
||||||
></q-input>
|
></q-input>
|
||||||
<q-input
|
<div class="row">
|
||||||
filled
|
<div class="col-5">
|
||||||
dense
|
<q-toggle
|
||||||
v-model.number="formDialog.data.costpword"
|
:label="`${flatRate}`"
|
||||||
type="number"
|
color="primary"
|
||||||
label="Amount per word"
|
v-model="formDialog.data.flatrate"
|
||||||
></q-input>
|
></q-toggle>
|
||||||
|
</div>
|
||||||
|
<div class="col-7">
|
||||||
|
<q-input
|
||||||
|
filled
|
||||||
|
dense
|
||||||
|
v-model.number="formDialog.data.amount"
|
||||||
|
type="number"
|
||||||
|
label="Amount"
|
||||||
|
></q-input>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row q-mt-lg">
|
<div class="row q-mt-lg">
|
||||||
<q-btn
|
<q-btn
|
||||||
v-if="formDialog.data.id"
|
v-if="formDialog.data.id"
|
||||||
|
|
@ -226,7 +238,7 @@
|
||||||
v-else
|
v-else
|
||||||
unelevated
|
unelevated
|
||||||
color="primary"
|
color="primary"
|
||||||
:disable="formDialog.data.costpword == null || formDialog.data.costpword < 0 || formDialog.data.name == null"
|
:disable="formDialog.data.amount == null || formDialog.data.amount < 0 || formDialog.data.name == null"
|
||||||
type="submit"
|
type="submit"
|
||||||
>Create Form</q-btn
|
>Create Form</q-btn
|
||||||
>
|
>
|
||||||
|
|
@ -275,10 +287,16 @@
|
||||||
field: 'description'
|
field: 'description'
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'costpword',
|
name: 'flatrate',
|
||||||
align: 'left',
|
align: 'left',
|
||||||
label: 'Cost Per Word',
|
label: 'Flat Rate',
|
||||||
field: 'costpword'
|
field: 'flatrate'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'amount',
|
||||||
|
align: 'left',
|
||||||
|
label: 'Amount',
|
||||||
|
field: 'amount'
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
pagination: {
|
pagination: {
|
||||||
|
|
@ -299,11 +317,20 @@
|
||||||
},
|
},
|
||||||
formDialog: {
|
formDialog: {
|
||||||
show: false,
|
show: false,
|
||||||
data: {}
|
data: {flatrate: false}
|
||||||
},
|
},
|
||||||
cancelListener: () => {}
|
cancelListener: () => {}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
computed: {
|
||||||
|
flatRate: function () {
|
||||||
|
if (this.formDialog.data.flatrate) {
|
||||||
|
return 'Charge flat rate'
|
||||||
|
} else {
|
||||||
|
return 'Charge per word'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
getTickets: function () {
|
getTickets: function () {
|
||||||
var self = this
|
var self = this
|
||||||
|
|
@ -400,7 +427,8 @@
|
||||||
this.formDialog.data.wallet = link.wallet
|
this.formDialog.data.wallet = link.wallet
|
||||||
this.formDialog.data.name = link.name
|
this.formDialog.data.name = link.name
|
||||||
this.formDialog.data.description = link.description
|
this.formDialog.data.description = link.description
|
||||||
this.formDialog.data.costpword = link.costpword
|
this.formDialog.data.flatrate = link.flatrate
|
||||||
|
this.formDialog.data.amount = link.amount
|
||||||
this.formDialog.show = true
|
this.formDialog.show = true
|
||||||
},
|
},
|
||||||
updateForm: function (wallet, data) {
|
updateForm: function (wallet, data) {
|
||||||
|
|
|
||||||
|
|
@ -28,6 +28,7 @@ async def display(form_id):
|
||||||
form_id=form.id,
|
form_id=form.id,
|
||||||
form_name=form.name,
|
form_name=form.name,
|
||||||
form_desc=form.description,
|
form_desc=form.description,
|
||||||
form_costpword=form.costpword,
|
form_amount=form.amount,
|
||||||
|
form_flatrate=form.flatrate,
|
||||||
form_wallet=wallet.inkey,
|
form_wallet=wallet.inkey,
|
||||||
)
|
)
|
||||||
|
|
|
||||||
|
|
@ -47,7 +47,8 @@ async def api_forms():
|
||||||
"name": {"type": "string", "empty": False, "required": True},
|
"name": {"type": "string", "empty": False, "required": True},
|
||||||
"webhook": {"type": "string", "empty": False, "required": False},
|
"webhook": {"type": "string", "empty": False, "required": False},
|
||||||
"description": {"type": "string", "min": 0, "required": True},
|
"description": {"type": "string", "min": 0, "required": True},
|
||||||
"costpword": {"type": "integer", "min": 0, "required": True},
|
"amount": {"type": "integer", "min": 0, "required": True},
|
||||||
|
"flatrate": {"type": "integer", "required": True},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
async def api_form_create(form_id=None):
|
async def api_form_create(form_id=None):
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue