From 624a9ef60abd240a0e7214f01ab0187313e9c202 Mon Sep 17 00:00:00 2001 From: ben Date: Tue, 14 Feb 2023 14:45:47 +0000 Subject: [PATCH] Remove splitpayments --- lnbits/extensions/splitpayments/README.md | 34 --- lnbits/extensions/splitpayments/__init__.py | 35 ---- lnbits/extensions/splitpayments/config.json | 6 - lnbits/extensions/splitpayments/crud.py | 36 ---- lnbits/extensions/splitpayments/migrations.py | 99 --------- lnbits/extensions/splitpayments/models.py | 28 --- .../static/image/split-payments.png | Bin 9839 -> 0 bytes .../splitpayments/static/js/index.js | 195 ------------------ lnbits/extensions/splitpayments/tasks.py | 76 ------- .../templates/splitpayments/_api_docs.html | 97 --------- .../templates/splitpayments/index.html | 147 ------------- lnbits/extensions/splitpayments/views.py | 17 -- lnbits/extensions/splitpayments/views_api.py | 63 ------ 13 files changed, 833 deletions(-) delete mode 100644 lnbits/extensions/splitpayments/README.md delete mode 100644 lnbits/extensions/splitpayments/__init__.py delete mode 100644 lnbits/extensions/splitpayments/config.json delete mode 100644 lnbits/extensions/splitpayments/crud.py delete mode 100644 lnbits/extensions/splitpayments/migrations.py delete mode 100644 lnbits/extensions/splitpayments/models.py delete mode 100644 lnbits/extensions/splitpayments/static/image/split-payments.png delete mode 100644 lnbits/extensions/splitpayments/static/js/index.js delete mode 100644 lnbits/extensions/splitpayments/tasks.py delete mode 100644 lnbits/extensions/splitpayments/templates/splitpayments/_api_docs.html delete mode 100644 lnbits/extensions/splitpayments/templates/splitpayments/index.html delete mode 100644 lnbits/extensions/splitpayments/views.py delete mode 100644 lnbits/extensions/splitpayments/views_api.py diff --git a/lnbits/extensions/splitpayments/README.md b/lnbits/extensions/splitpayments/README.md deleted file mode 100644 index 8b0554cbe..000000000 --- a/lnbits/extensions/splitpayments/README.md +++ /dev/null @@ -1,34 +0,0 @@ -# Split Payments - -## Have payments split between multiple wallets - -LNbits Split Payments extension allows for distributing payments across multiple wallets. Set it and forget it. It will keep splitting your payments across wallets forever. - -## Usage - -1. After enabling the extension, choose the source wallet that will receive and distribute the Payments - -![choose wallet](https://i.imgur.com/nPQudqL.png) - -2. Add the wallet or wallets info to split payments to - -![split wallets](https://i.imgur.com/5hCNWpg.png) - get the wallet id, or an invoice key from a different wallet. It can be a completely different user as long as it's under the same LNbits instance/domain. You can get the wallet information on the API Info section on every wallet page\ - ![wallet info](https://i.imgur.com/betqflC.png) - set a wallet _Alias_ for your own identification\ - -- set how much, in percentage, this wallet will receive from every payment sent to the source wallets - -3. When done, click "SAVE TARGETS" to make the splits effective - -4. You can have several wallets to split to, as long as the sum of the percentages is under or equal to 100% - -5. When the source wallet receives a payment, the extension will automatically split the corresponding values to every wallet\ - - on receiving a 20 sats payment\ - ![get 20 sats payment](https://i.imgur.com/BKp0xvy.png) - - source wallet gets 18 sats\ - ![source wallet](https://i.imgur.com/GCxDZ5s.png) - - Ben's wallet (the wallet from the example) instantly, and feeless, gets the corresponding 10%, or 2 sats\ - ![ben wallet](https://i.imgur.com/MfsccNa.png) - -## Sponsored by - -[![](https://cdn.shopify.com/s/files/1/0826/9235/files/cryptograffiti_logo_clear_background.png?v=1504730421)](https://cryptograffiti.com/) diff --git a/lnbits/extensions/splitpayments/__init__.py b/lnbits/extensions/splitpayments/__init__.py deleted file mode 100644 index 5efb6335f..000000000 --- a/lnbits/extensions/splitpayments/__init__.py +++ /dev/null @@ -1,35 +0,0 @@ -import asyncio - -from fastapi import APIRouter -from fastapi.staticfiles import StaticFiles - -from lnbits.db import Database -from lnbits.helpers import template_renderer -from lnbits.tasks import catch_everything_and_restart - -db = Database("ext_splitpayments") - -splitpayments_static_files = [ - { - "path": "/splitpayments/static", - "app": StaticFiles(packages=[("lnbits", "extensions/splitpayments/static")]), - "name": "splitpayments_static", - } -] -splitpayments_ext: APIRouter = APIRouter( - prefix="/splitpayments", tags=["splitpayments"] -) - - -def splitpayments_renderer(): - return template_renderer(["lnbits/extensions/splitpayments/templates"]) - - -from .tasks import wait_for_paid_invoices -from .views import * # noqa: F401,F403 -from .views_api import * # noqa: F401,F403 - - -def splitpayments_start(): - loop = asyncio.get_event_loop() - loop.create_task(catch_everything_and_restart(wait_for_paid_invoices)) diff --git a/lnbits/extensions/splitpayments/config.json b/lnbits/extensions/splitpayments/config.json deleted file mode 100644 index 1e0c9671b..000000000 --- a/lnbits/extensions/splitpayments/config.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "name": "Split Payments", - "short_description": "Split incoming payments across wallets", - "tile": "/splitpayments/static/image/split-payments.png", - "contributors": ["fiatjaf", "cryptograffiti"] -} diff --git a/lnbits/extensions/splitpayments/crud.py b/lnbits/extensions/splitpayments/crud.py deleted file mode 100644 index 737e7bb9a..000000000 --- a/lnbits/extensions/splitpayments/crud.py +++ /dev/null @@ -1,36 +0,0 @@ -from typing import List - -from lnbits.helpers import urlsafe_short_hash - -from . import db -from .models import Target - - -async def get_targets(source_wallet: str) -> List[Target]: - rows = await db.fetchall( - "SELECT * FROM splitpayments.targets WHERE source = ?", (source_wallet,) - ) - return [Target(**row) for row in rows] - - -async def set_targets(source_wallet: str, targets: List[Target]): - async with db.connect() as conn: - await conn.execute( - "DELETE FROM splitpayments.targets WHERE source = ?", (source_wallet,) - ) - for target in targets: - await conn.execute( - """ - INSERT INTO splitpayments.targets - (id, source, wallet, percent, tag, alias) - VALUES (?, ?, ?, ?, ?, ?) - """, - ( - urlsafe_short_hash(), - source_wallet, - target.wallet, - target.percent, - target.tag, - target.alias, - ), - ) diff --git a/lnbits/extensions/splitpayments/migrations.py b/lnbits/extensions/splitpayments/migrations.py deleted file mode 100644 index eb72387e4..000000000 --- a/lnbits/extensions/splitpayments/migrations.py +++ /dev/null @@ -1,99 +0,0 @@ -from lnbits.helpers import urlsafe_short_hash - - -async def m001_initial(db): - """ - Initial split payment table. - """ - await db.execute( - """ - CREATE TABLE splitpayments.targets ( - wallet TEXT NOT NULL, - source TEXT NOT NULL, - percent INTEGER NOT NULL CHECK (percent >= 0 AND percent <= 100), - alias TEXT, - - UNIQUE (source, wallet) - ); - """ - ) - - -async def m002_float_percent(db): - """ - Add float percent and migrates the existing data. - """ - await db.execute("ALTER TABLE splitpayments.targets RENAME TO splitpayments_old") - await db.execute( - """ - CREATE TABLE splitpayments.targets ( - wallet TEXT NOT NULL, - source TEXT NOT NULL, - percent REAL NOT NULL CHECK (percent >= 0 AND percent <= 100), - alias TEXT, - - UNIQUE (source, wallet) - ); - """ - ) - - for row in [ - list(row) - for row in await db.fetchall("SELECT * FROM splitpayments.splitpayments_old") - ]: - await db.execute( - """ - INSERT INTO splitpayments.targets ( - wallet, - source, - percent, - alias - ) - VALUES (?, ?, ?, ?) - """, - (row[0], row[1], row[2], row[3]), - ) - - await db.execute("DROP TABLE splitpayments.splitpayments_old") - - -async def m003_add_id_and_tag(db): - """ - Add float percent and migrates the existing data. - """ - await db.execute("ALTER TABLE splitpayments.targets RENAME TO splitpayments_old") - await db.execute( - """ - CREATE TABLE splitpayments.targets ( - id TEXT PRIMARY KEY, - wallet TEXT NOT NULL, - source TEXT NOT NULL, - percent REAL NOT NULL CHECK (percent >= 0 AND percent <= 100), - tag TEXT NOT NULL, - alias TEXT, - - UNIQUE (source, wallet) - ); - """ - ) - - for row in [ - list(row) - for row in await db.fetchall("SELECT * FROM splitpayments.splitpayments_old") - ]: - await db.execute( - """ - INSERT INTO splitpayments.targets ( - id, - wallet, - source, - percent, - tag, - alias - ) - VALUES (?, ?, ?, ?, ?, ?) - """, - (urlsafe_short_hash(), row[0], row[1], row[2], "", row[3]), - ) - - await db.execute("DROP TABLE splitpayments.splitpayments_old") diff --git a/lnbits/extensions/splitpayments/models.py b/lnbits/extensions/splitpayments/models.py deleted file mode 100644 index 4f2bb0106..000000000 --- a/lnbits/extensions/splitpayments/models.py +++ /dev/null @@ -1,28 +0,0 @@ -from sqlite3 import Row -from typing import List, Optional - -from fastapi import Query -from pydantic import BaseModel - - -class Target(BaseModel): - wallet: str - source: str - percent: float - tag: str - alias: Optional[str] - - @classmethod - def from_row(cls, row: Row): - return cls(**dict(row)) - - -class TargetPutList(BaseModel): - wallet: str = Query(...) - alias: str = Query("") - percent: float = Query(..., ge=0, lt=100) - tag: str - - -class TargetPut(BaseModel): - __root__: List[TargetPutList] diff --git a/lnbits/extensions/splitpayments/static/image/split-payments.png b/lnbits/extensions/splitpayments/static/image/split-payments.png deleted file mode 100644 index 10b8e7f2370d7b325ff5b27964e18e79bbeeacb2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 9839 zcmeHMXIN9)whg_hARxWO(9u9bPayOvRYfTx5Rw3)1rj<)2WcW8qDT{@CtTpGDbB;OI-q|ZLCPq3Z>AC0u0KiE- zUCm3R&yeF6Ej8)AMH%@P0AM=lZ))y&3F!lL$K&j=E*PMvpF0MK@x|H$0KOw753t@m zFB7zn2xsakjV8-@PEEqxmCAeY7St&2J(R62Te+na!7ffS(SjibZunxiR^ZLaI7 zBNrswzTXL0^;pmU(cW8=O`PBJ3sye$b?x54>UUMNnO%Aod&>L|{=JrY!0qnI^gv!m z1Y_4A1wy4AH$m+`E;<|!ve_a$Jme8=YY$!Lu~kZ))yHivF&8LT?V*HLz77uS6!|Ro zsx=%bn&nnUW{PPWp7m3{>~91dp7`89sb7gcn&6&yk*b95Jd*TxeWg8KT zLg%t4n)eP?1FGB5sX zU5^3NisDQ9C6ynnb@ajV*!&M4Epd<9vw((+`Ge#BYbf_#;+ybd8j*PDjDfVVM@1KW zbeV9z(_vPq_-XCP7w6@g_faa|!(P>6+kFpHlI_ocEi18u9-@_ZK6U6A!8TvK@7?Jd zkuEvKy ziowU<-K!D?Ze~ec=apUJl*VR?J>zXlXT>116M|xSKEBXR-56tMI`@d3jmuM34}0O3 z#ndA%XX7HanxQECZF>6QYjo)mUHt2#%MTZ~Z(rLs7I`&siRfIwcH8aZv`z9T4eaJ) z+3I|2!Y!<1he(`ps@!n;^2Uvkka9*f5dtBj+#f-C_#(6a-Bf`SL!9N5?$Z&>^Dla( zN3~m{cO83YM(*j4)o;;qW+X2evEoDoOwBl!2fgPO7xOA;{e@KY(_r*MY2VJRm~TCi zkWZtbdiVZIVOX+Z;5kr4epx+Xbu-?eA&FgD3|F;UNYPkVPeEK2AJ9canlC>H%c3q9 z5te3el`r5kodGA6U7!|%7xVj+762?V(!};aNQ%?}r>u%Bna(ucp?EfL2A#>2j+58; zQY;)%Js{JA<;+*J2x1PFJ6&o)Nv=o$j05 zG_&WKZ;VMfy&6dyggBSl9j{teuDVrv0$6p%94?&+q_OWib<^=luVP(*a~-tWwHF(d zJ)3DD5OzhNjcHe#9&^zY>1bo&p_g#Fo&&k|fD`M+p?TXjuC_EeZTU}c%mu*6>&ZO;Ih~!KYL?qOf2kS*?2v81~ojguG zSix?wSCCx5%;y<*N;R)0ap~n;c3s94S#hqrd~jH%Z$KB~)QDiXMW39PAp%>X{`~Vv zi{5B1{BH1h!?$@uPlC+vKcymjQ8piT2b3%BtM;Z!Z4#m46m*LpDeO$tZEe9c(k|5N z#{ zkvYo~HQJs#Zf4Wm`s}iGDdiHn(Gn$XTfVT@W`@Wx)Dc1MR~yX%@e)ep~*Uh zJFmN%>fS^-6Dg~kjqRM3TBF{T=ebg{K79eex79@HjggN=`1fhG#cNTiye#2Hp&X=&o`-!Ft1HuQXrM6pjw*@wL!Ui{}))b2$omg+#o(<93R!ylNh4@NS$EHcq<3{$90e6%(a*V~<^6(l~(;HJU^mgXmkvzomJ3}D{su+I$Y3_1kMcEgj#-d8UGLYC%jDB0 zz2128c$VUL#+H65n}Xru)HztyD~Wh^1lwXO*;SgJrhXMbX)yV!P7Q2*%O*6vBBQZs zWpHXUOk$g6-CynA9>VvEbAFOPe&G8dS2YW7|Gx)JNe{{~lo?!xC%6vd?ob z9G9d2nKoTGKdkc<(?ZCpa=p;^i$yt9&sBmvPP=JF7j2q-kHu`s@UebYNU1M$E!%o! z1q~ch68E;nmd$c^ux@1K#a+1)c&4t{e)ZNz1ml>%-Aj^M9KlqCPZp*4aqZ>&HT*>U zr&3pk0P0USYH9PHKUx;B=AX2EVJWs{YsWx`L8dS=B!W?wEuHfKAXQlXJoOEX{ z5pmyL{6?bMVZQ5bEU11yncOyJM%=}tDPHMHALVhbMlOR3@~hZIQAGz zeubu$G2SPgieBt##KqTdgJAOqVgs3=%-%xdw`z++rtzBvUlljLQ9^b4t}nmzU(Ak$ zBTwhecG}0PSCsFYZKP9KxA}BeY*^=AB;Iimat_EF`f+g*?`&qkK@e;VG6&wuUw?PU zTQ`nH-$MKK1raWe5I9&eHQ7t(u8}X_ZO;+Ggf&*m{Y0bS7ojFwfX}B-IQcD(ia7U7aZxk#t5voguf< z2$hcT9ndssb`$?5|MKWo$sT(fOUY6X@AVu8MBh%{6Yob+OZ5h^Odm%ot7r-YKfSIW zWl8+t{5eD2J*0Lhyy~Kk6hRe3mNBzDW$34*pj8FDwk!E9)JPPgSO^XyBNSx-;_{$hvEHKOt1Ip^6ZKI$4VkN`j;)|} zHdo^dF2hXEL_;s|J0-1@^r`i+#n_Kg(4)y@oH;+=-`wMd*j`p{*%UHoEqD@#=0a(v zyv$gUi}YU8Gy5v@^q_TXwD(}F@dk4J6P)8R#__ z5A#QD5$k#eiPL84Gt5!-%DI)xbBFR#mFXpEgBhx1_cx+qH`yv+ou|1FB`*FSH3Kll zu@ps!X;GrJ-*Q{i7X63aHv`?tUGJ2&9>3yX*><)%yESd{{f5{;w?)V|VG+Z$!v@yo z_^J)xbD!13csmRlcXq#wemq!oKL4EkwW7fA&XNls|FlvJb)fM8x6iG`tweZMPTxnD z<^JzyQgB6cP^@8LJ!@WM7UsRZSO*7Vjj(i+T|%y0wjb$=VV63yXf1%Vj*nA=sPkT<( z?MYfD*LI{wYGn4Ky#PwF!%)@3*!~2~is+7(UA3?6)`sT$mz$a9F0mnyW>x|R0ry$_ z9;xM1J*Wpfnqj*o!10AEr)FXCa6mlhy&;YC>QL2(7#e#g-Vv%=%6%V>=rdMLi<)ls z^vvOQ138zMk50L~5$*RK`@Z5ubybwEfJHNs;<~E*+{Aj!wF2!2FlL>^xC+TF{Lq2Q z63f*h?k|fC003D8Rzt%?PebGPjRDejK*o(Egl>xoYozgIt)y5D#b^ceV*=e>SEdQm&^r>k`)|D3X5@Rc?onlNLZuh|rfHv> zFWcsaTXYOW7e+rSD!AN`xM@E7s?DN!z;lk^SD$cCPBL+TFKyNK)&(feZ{ducgH*DU@Gn2NvXVAxPEBG-8#U&J z25=P4RRW2|*R-nO?~kP6BYncp`zmt}bo_xGw_q6BkZ8J|30?0e_l!IwL^l zh9*D_93BIdk&uyqfVF(F-clfCdY}>>Z4bYssr?%S=^Fy%=;`SWmz4DJ@saS6mcZd1 zB%v@EOcEj`DJ2CaS%3+CZk|YAup2?(7~&U(CWe5*W8FQmI5*%iCejY)<%s}+Nb|to z^K*4KH2eeJjqn=_Bt9g4k?xXE35cYttK^?O2%cKrB#_?{`nMhgQ__ByaM zw7fBHo&tY{K%@TfclW}({458JlEk=RTuG(`QdH<)Lh9%ln*8B$Oo0Q|)%~XziR`~v zdSdPWMb=+(JD&Mj&Yu$@x&MLt7wg|^|7lFJGBkv1;!s}4>FH@AK*#anXdDWQhW|XY zmz76CWKkF}5@s(2mcd9vz;;ql7+4yKl!IcBFj-ls{hy%p+z6gXHx%X=iUcl!CE>_Q zp-Fa9Xs{gyCJUCamqmeL5TqSg4uX-9!=R*L7#QqN5XN{cNtH;KKWB9eg+7Kt$=b`o zVDezJy|f)zMjmDdRv;w;mXg7s(bCdL7!)n{6AF!jYvb^)NK!kou1E)rq`RBL&xvEg z;c6y&2#}NnUS1X^Cj*1Z%R_$! z`iXZHgC~%bc#H{!NJ#&}J#Gt}R1Aq&VR{U?8zl`zt_3hUZaKZkZ0s?=^7LG*y3PM17W6(bXk^Fv5p&XHJ z4j9t<{!LK7k7NH$v+ShoAaY2w92f(WLxN?b(J-)r9Lf$1k(ZZ|MN7%aDHn}qs5BfR2Zzf4 z|CNYBDL@qzNDYKQNsC2B3MmU#usdFX3ih&g@-j$y1^Zu1?mz9KG${fYB5ewV!DUHZ zl=*9kN|MLt=HGRzB>BIT{mbA_Cv#Hwe~pnY5~OR6P1{zKP4bp0&`{+97S>iUPSzs11cGX6(h|7Ud3|Le9D<3{=|>qELJ74kVfd-bu9=000YbMiwy7}gM&0kJ2tsB2S>AAY;Am;@ z-ezD{@AhK#EvBf|Qqe@wF?gtGv!;Itq3Lvg+H;6cbi|2yVVxMLFpsCw6%E8ileaId#Sr)_f&rrC*=7fk3Q|)~l5mTuWqcJZQN# zVcR};?~6ne+AW!P_}Px%bOY5HTU(CWicQ{}9jT$3(m8S&YO;4)j{yK8IlICpz==Ha zSHzqf01C$#HIw{%&+UbrDgoX4lLupENY_nn__?g7KN^25O>P#g)#W`;TH)d5EjQZx6tB!IrQyM`xQxS#=Xv07%uHRNtASx2>Tpjlfj_ zqApBc+L>dpy-#_YJ4iXZ5I6K`eTAusiH=WxKgKFSGDxXq)S>)!6z(=4?!21pRPb;> z>g_#A4oPOK~Qzx!^g*Q@d?TDiBx$_@D_x2`y(fEu#Kl?^AWkrpVX96* zHB-o4D6pFH@f}KVH+hU5M!mSp+E_7x{6x^^`2Y!aD`uw?6<1g&yn4bPweL2>=uq z<56d@2c53N>yQ>M#x?sRwf?jiJd3aHm8QmF6)(1j;*%`fofW~m*h#X<7q8p>P9^tb zWWHof?D<4-DY*%2bWeDf!oC6gaqJ950)?Wibx&lrxS^;0)sVEilQ%C_i!R%IS9!)7 zG)oV;LYJ$`tU%o=n{YMj%cgaluSAc1N5|vG)vWO!~O^Rn1xmV diff --git a/lnbits/extensions/splitpayments/static/js/index.js b/lnbits/extensions/splitpayments/static/js/index.js deleted file mode 100644 index f5f162769..000000000 --- a/lnbits/extensions/splitpayments/static/js/index.js +++ /dev/null @@ -1,195 +0,0 @@ -/* globals Quasar, Vue, _, VueQrcode, windowMixin, LNbits, LOCALE */ - -Vue.component(VueQrcode.name, VueQrcode) - -function hashTargets(targets) { - return targets - .filter(isTargetComplete) - .map(({wallet, percent, alias}) => `${wallet}${percent}${alias}`) - .join('') -} - -function isTargetComplete(target) { - return ( - target.wallet && - target.wallet.trim() !== '' && - (target.percent > 0 || target.tag != '') - ) -} - -new Vue({ - el: '#vue', - mixins: [windowMixin], - data() { - return { - selectedWallet: null, - currentHash: '', // a string that must match if the edit data is unchanged - targets: [ - { - method: 'split' - } - ] - } - }, - computed: { - isDirty() { - return hashTargets(this.targets) !== this.currentHash - } - }, - methods: { - clearTargets() { - this.targets = [{}] - this.$q.notify({ - message: - 'Cleared the form, but not saved. You must click to save manually.', - timeout: 500 - }) - }, - clearTarget(index) { - this.targets.splice(index, 1) - console.log(this.targets) - this.$q.notify({ - message: 'Removed item. You must click to save manually.', - timeout: 500 - }) - }, - getTargets() { - LNbits.api - .request( - 'GET', - '/splitpayments/api/v1/targets', - this.selectedWallet.adminkey - ) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - .then(response => { - this.currentHash = hashTargets(response.data) - this.targets = response.data.concat({}) - for (let i = 0; i < this.targets.length; i++) { - if (this.targets[i].tag.length > 0) { - this.targets[i].method = 'tag' - } else if (this.targets[i].percent.length > 0) { - this.targets[i].method = 'split' - } else { - this.targets[i].method = '' - } - } - }) - }, - changedWallet(wallet) { - this.selectedWallet = wallet - this.getTargets() - }, - clearChanged(index) { - if (this.targets[index].method == 'split') { - this.targets[index].tag = null - this.targets[index].method = 'split' - } else { - this.targets[index].percent = null - this.targets[index].method = 'tag' - } - }, - targetChanged(index) { - // fix percent min and max range - if (this.targets[index].percent) { - if (this.targets[index].percent > 100) this.targets[index].percent = 100 - if (this.targets[index].percent < 0) this.targets[index].percent = 0 - this.targets[index].tag = '' - } - - // not percentage - if (!this.targets[index].percent) { - this.targets[index].percent = 0 - } - - // remove empty lines (except last) - if (this.targets.length >= 2) { - for (let i = this.targets.length - 2; i >= 0; i--) { - let target = this.targets[i] - if ( - (!target.wallet || target.wallet.trim() === '') && - (!target.alias || target.alias.trim() === '') && - (!target.tag || target.tag.trim() === '') && - !target.percent - ) { - this.targets.splice(i, 1) - } - } - } - - // add a line at the end if the last one is filled - let last = this.targets[this.targets.length - 1] - if (last.wallet && last.wallet.trim() !== '') { - this.targets.push({}) - } - - // sum of all percents - let currentTotal = this.targets.reduce( - (acc, target) => acc + (target.percent || 0), - 0 - ) - - // remove last (unfilled) line if the percent is already 100 - if (currentTotal >= 100) { - let last = this.targets[this.targets.length - 1] - if ( - (!last.wallet || last.wallet.trim() === '') && - (!last.alias || last.alias.trim() === '') && - !last.percent - ) { - this.targets = this.targets.slice(0, -1) - } - } - - // adjust percents of other lines (not this one) - if (currentTotal > 100 && isPercent) { - let diff = (currentTotal - 100) / (100 - this.targets[index].percent) - this.targets.forEach((target, t) => { - if (t !== index) target.percent -= +(diff * target.percent).toFixed(2) - }) - } - // overwrite so changes appear - this.targets = this.targets - }, - saveTargets() { - for (let i = 0; i < this.targets.length; i++) { - if (this.targets[i].tag != '') { - this.targets[i].percent = 0 - } else { - this.targets[i].tag = '' - } - } - LNbits.api - .request( - 'PUT', - '/splitpayments/api/v1/targets', - this.selectedWallet.adminkey, - { - targets: this.targets - .filter(isTargetComplete) - .map(({wallet, percent, tag, alias}) => ({ - wallet, - percent, - tag, - alias - })) - } - ) - .then(response => { - this.$q.notify({ - message: 'Split payments targets set.', - timeout: 700 - }) - this.getTargets() - }) - .catch(err => { - LNbits.utils.notifyApiError(err) - }) - } - }, - created() { - this.selectedWallet = this.g.user.wallets[0] - this.getTargets() - } -}) diff --git a/lnbits/extensions/splitpayments/tasks.py b/lnbits/extensions/splitpayments/tasks.py deleted file mode 100644 index 59aa8e051..000000000 --- a/lnbits/extensions/splitpayments/tasks.py +++ /dev/null @@ -1,76 +0,0 @@ -import asyncio - -from loguru import logger - -from lnbits.core.models import Payment -from lnbits.core.services import create_invoice, pay_invoice -from lnbits.helpers import get_current_extension_name -from lnbits.tasks import register_invoice_listener - -from .crud import get_targets - - -async def wait_for_paid_invoices(): - invoice_queue = asyncio.Queue() - register_invoice_listener(invoice_queue, get_current_extension_name()) - - while True: - payment = await invoice_queue.get() - await on_invoice_paid(payment) - - -async def on_invoice_paid(payment: Payment) -> None: - - if payment.extra.get("tag") == "splitpayments" or payment.extra.get("splitted"): - # already a splitted payment, ignore - return - - targets = await get_targets(payment.wallet_id) - - if not targets: - return - - # validate target percentages - total_percent = sum([target.percent for target in targets]) - - if total_percent > 100: - logger.error("splitpayment: total percent adds up to more than 100%") - return - - logger.trace(f"splitpayments: performing split payments to {len(targets)} targets") - - if payment.extra.get("amount"): - amount_to_split = (payment.extra.get("amount") or 0) * 1000 - else: - amount_to_split = payment.amount - - if not amount_to_split: - logger.error("splitpayments: no amount to split") - return - - for target in targets: - tagged = target.tag in payment.extra - - if tagged or target.percent > 0: - - if tagged: - memo = f"Pushed tagged payment to {target.alias}" - amount_msat = int(amount_to_split) - else: - amount_msat = int(amount_to_split * target.percent / 100) - memo = f"Split payment: {target.percent}% for {target.alias or target.wallet}" - - payment_hash, payment_request = await create_invoice( - wallet_id=target.wallet, - amount=int(amount_msat / 1000), - internal=True, - memo=memo, - ) - - extra = {**payment.extra, "tag": "splitpayments", "splitted": True} - - await pay_invoice( - payment_request=payment_request, - wallet_id=payment.wallet_id, - extra=extra, - ) diff --git a/lnbits/extensions/splitpayments/templates/splitpayments/_api_docs.html b/lnbits/extensions/splitpayments/templates/splitpayments/_api_docs.html deleted file mode 100644 index 4b5ed979c..000000000 --- a/lnbits/extensions/splitpayments/templates/splitpayments/_api_docs.html +++ /dev/null @@ -1,97 +0,0 @@ - - - -

- Add some wallets to the list of "Target Wallets", each with an - associated percent. After saving, every time any payment - arrives at the "Source Wallet" that payment will be split with the - target wallets according to their percent. -

-

This is valid for every payment, doesn't matter how it was created.

-

Target wallets can be any wallet from this same LNbits instance.

-

- To remove a wallet from the targets list, just erase its fields and - save. To remove all, click "Clear" then save. -

-
-
-
- - - - - - - GET - /splitpayments/api/v1/targets -
Headers
- {"X-Api-Key": <admin_key>}
-
Body (application/json)
-
- Returns 200 OK (application/json) -
- [{"wallet": <wallet id>, "alias": <chosen name for this - wallet>, "percent": <number between 1 and 100>}, ...] -
Curl example
- curl -X GET {{ request.base_url }}splitpayments/api/v1/targets -H - "X-Api-Key: {{ user.wallets[0].inkey }}" - -
-
-
- - - - PUT - /splitpayments/api/v1/targets -
Headers
- {"X-Api-Key": <admin_key>}
-
Body (application/json)
-
- Returns 200 OK (application/json) -
-
Curl example
- curl -X PUT {{ request.base_url }}splitpayments/api/v1/targets -H - "X-Api-Key: {{ user.wallets[0].adminkey }}" -H 'Content-Type: - application/json' -d '{"targets": [{"wallet": <wallet id or invoice - key>, "alias": <name to identify this>, "percent": <number - between 1 and 100>}, ...]}' - -
-
-
-
diff --git a/lnbits/extensions/splitpayments/templates/splitpayments/index.html b/lnbits/extensions/splitpayments/templates/splitpayments/index.html deleted file mode 100644 index b105bf2c4..000000000 --- a/lnbits/extensions/splitpayments/templates/splitpayments/index.html +++ /dev/null @@ -1,147 +0,0 @@ -{% extends "base.html" %} {% from "macros.jinja" import window_vars with context -%} {% block page %} -
-
- - - - - - - - - - - -
-
Target Wallets
-
- - -
- - - - - - - - - - - - Add more - - -
-
-
- - Clear - -
- -
- - Save Targets - -
-
-
-
-
-
- -
- - -
- {{SITE_TITLE}} SplitPayments extension -
-
- - - {% include "splitpayments/_api_docs.html" %} - -
-
-
-{% endblock %} {% block scripts %} {{ window_vars(user) }} - -{% endblock %} diff --git a/lnbits/extensions/splitpayments/views.py b/lnbits/extensions/splitpayments/views.py deleted file mode 100644 index c33fd4241..000000000 --- a/lnbits/extensions/splitpayments/views.py +++ /dev/null @@ -1,17 +0,0 @@ -from fastapi import Depends, Request -from fastapi.templating import Jinja2Templates -from starlette.responses import HTMLResponse - -from lnbits.core.models import User -from lnbits.decorators import check_user_exists - -from . import splitpayments_ext, splitpayments_renderer - -templates = Jinja2Templates(directory="templates") - - -@splitpayments_ext.get("/", response_class=HTMLResponse) -async def index(request: Request, user: User = Depends(check_user_exists)): - return splitpayments_renderer().TemplateResponse( - "splitpayments/index.html", {"request": request, "user": user.dict()} - ) diff --git a/lnbits/extensions/splitpayments/views_api.py b/lnbits/extensions/splitpayments/views_api.py deleted file mode 100644 index f83f2b4f5..000000000 --- a/lnbits/extensions/splitpayments/views_api.py +++ /dev/null @@ -1,63 +0,0 @@ -from http import HTTPStatus - -from fastapi import Depends, Request -from starlette.exceptions import HTTPException - -from lnbits.core.crud import get_wallet, get_wallet_for_key -from lnbits.decorators import WalletTypeInfo, require_admin_key - -from . import splitpayments_ext -from .crud import get_targets, set_targets -from .models import Target, TargetPut - - -@splitpayments_ext.get("/api/v1/targets") -async def api_targets_get(wallet: WalletTypeInfo = Depends(require_admin_key)): - targets = await get_targets(wallet.wallet.id) - return [target.dict() for target in targets] or [] - - -@splitpayments_ext.put("/api/v1/targets") -async def api_targets_set( - req: Request, wal: WalletTypeInfo = Depends(require_admin_key) -): - body = await req.json() - targets = [] - data = TargetPut.parse_obj(body["targets"]) - for entry in data.__root__: - wallet = await get_wallet(entry.wallet) - if not wallet: - wallet = await get_wallet_for_key(entry.wallet, "invoice") - if not wallet: - raise HTTPException( - status_code=HTTPStatus.BAD_REQUEST, - detail=f"Invalid wallet '{entry.wallet}'.", - ) - - if wallet.id == wal.wallet.id: - raise HTTPException( - status_code=HTTPStatus.BAD_REQUEST, detail="Can't split to itself." - ) - - if entry.percent < 0: - raise HTTPException( - status_code=HTTPStatus.BAD_REQUEST, - detail=f"Invalid percent '{entry.percent}'.", - ) - - targets.append( - Target( - wallet=wallet.id, - source=wal.wallet.id, - tag=entry.tag, - percent=entry.percent, - alias=entry.alias, - ) - ) - percent_sum = sum([target.percent for target in targets]) - if percent_sum > 100: - raise HTTPException( - status_code=HTTPStatus.BAD_REQUEST, detail="Splitting over 100%." - ) - await set_targets(wal.wallet.id, targets) - return ""