kopia lustrzana https://github.com/vitorpamplona/amethyst
225 wiersze
8.8 KiB
Kotlin
225 wiersze
8.8 KiB
Kotlin
/**
|
|
* Copyright (c) 2024 Vitor Pamplona
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
* this software and associated documentation files (the "Software"), to deal in
|
|
* the Software without restriction, including without limitation the rights to use,
|
|
* copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the
|
|
* Software, and to permit persons to whom the Software is furnished to do so,
|
|
* subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
* copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
package com.vitorpamplona.amethyst.service
|
|
|
|
import android.content.Context
|
|
import androidx.compose.runtime.Immutable
|
|
import com.fasterxml.jackson.databind.JsonNode
|
|
import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
|
|
import com.vitorpamplona.amethyst.R
|
|
import com.vitorpamplona.amethyst.service.lnurl.LightningAddressResolver
|
|
import com.vitorpamplona.amethyst.ui.components.GenericLoadable
|
|
import com.vitorpamplona.quartz.events.Event
|
|
import okhttp3.MediaType.Companion.toMediaType
|
|
import okhttp3.Request
|
|
import okhttp3.RequestBody.Companion.toRequestBody
|
|
import java.util.Base64
|
|
import kotlin.coroutines.cancellation.CancellationException
|
|
|
|
@Immutable
|
|
data class CashuToken(
|
|
val token: String,
|
|
val mint: String,
|
|
val totalAmount: Long,
|
|
val proofs: JsonNode,
|
|
)
|
|
|
|
class CashuProcessor {
|
|
fun parse(cashuToken: String): GenericLoadable<CashuToken> {
|
|
checkNotInMainThread()
|
|
|
|
try {
|
|
val base64token = cashuToken.replace("cashuA", "")
|
|
val cashu = jacksonObjectMapper().readTree(String(Base64.getDecoder().decode(base64token)))
|
|
val token = cashu.get("token").get(0)
|
|
val proofs = token.get("proofs")
|
|
val mint = token.get("mint").asText()
|
|
|
|
var totalAmount = 0L
|
|
for (proof in proofs) {
|
|
totalAmount += proof.get("amount").asLong()
|
|
}
|
|
|
|
return GenericLoadable.Loaded(CashuToken(cashuToken, mint, totalAmount, proofs))
|
|
} catch (e: Exception) {
|
|
if (e is CancellationException) throw e
|
|
return GenericLoadable.Error<CashuToken>("Could not parse this cashu token")
|
|
}
|
|
}
|
|
|
|
suspend fun melt(
|
|
token: CashuToken,
|
|
lud16: String,
|
|
onSuccess: (String, String) -> Unit,
|
|
onError: (String, String) -> Unit,
|
|
context: Context,
|
|
) {
|
|
checkNotInMainThread()
|
|
|
|
runCatching {
|
|
LightningAddressResolver()
|
|
.lnAddressInvoice(
|
|
lnaddress = lud16,
|
|
// Make invoice and leave room for fees
|
|
milliSats = token.totalAmount * 1000,
|
|
message = "Calculate Fees for Cashu",
|
|
onSuccess = { baseInvoice ->
|
|
feeCalculator(
|
|
token.mint,
|
|
baseInvoice,
|
|
onSuccess = { fees ->
|
|
LightningAddressResolver()
|
|
.lnAddressInvoice(
|
|
lnaddress = lud16,
|
|
// Make invoice and leave room for fees
|
|
milliSats = (token.totalAmount - fees) * 1000,
|
|
message = "Redeem Cashu",
|
|
onSuccess = { invoice ->
|
|
meltInvoice(token, invoice, fees, onSuccess, onError, context)
|
|
},
|
|
onProgress = {},
|
|
onError = onError,
|
|
context = context,
|
|
)
|
|
},
|
|
onError = onError,
|
|
context,
|
|
)
|
|
},
|
|
onProgress = {},
|
|
onError = onError,
|
|
context = context,
|
|
)
|
|
}
|
|
}
|
|
|
|
fun feeCalculator(
|
|
mintAddress: String,
|
|
invoice: String,
|
|
onSuccess: (Int) -> Unit,
|
|
onError: (String, String) -> Unit,
|
|
context: Context,
|
|
) {
|
|
checkNotInMainThread()
|
|
|
|
try {
|
|
val client = HttpClientManager.getHttpClient()
|
|
val url = "$mintAddress/checkfees" // Melt cashu tokens at Mint
|
|
|
|
val factory = Event.mapper.nodeFactory
|
|
|
|
val jsonObject = factory.objectNode()
|
|
jsonObject.put("pr", invoice)
|
|
|
|
val mediaType = "application/json; charset=utf-8".toMediaType()
|
|
val requestBody = jsonObject.toString().toRequestBody(mediaType)
|
|
val request = Request.Builder().url(url).post(requestBody).build()
|
|
|
|
client.newCall(request).execute().use {
|
|
val body = it.body.string()
|
|
val tree = jacksonObjectMapper().readTree(body)
|
|
|
|
val feeCost = tree?.get("fee")?.asInt()
|
|
|
|
if (feeCost != null) {
|
|
onSuccess(
|
|
feeCost,
|
|
)
|
|
} else {
|
|
val msg = tree?.get("detail")?.asText()?.split('.')?.getOrNull(0)?.ifBlank { null }
|
|
onError(
|
|
context.getString(R.string.cashu_failed_redemption),
|
|
if (msg != null) {
|
|
context.getString(R.string.cashu_failed_redemption_explainer_error_msg, msg)
|
|
} else {
|
|
context.getString(R.string.cashu_failed_redemption_explainer_error_msg)
|
|
},
|
|
)
|
|
}
|
|
}
|
|
} catch (e: Exception) {
|
|
if (e is CancellationException) throw e
|
|
onError(
|
|
context.getString(R.string.cashu_successful_redemption),
|
|
context.getString(R.string.cashu_failed_redemption_explainer_error_msg, e.message),
|
|
)
|
|
}
|
|
}
|
|
|
|
private fun meltInvoice(
|
|
token: CashuToken,
|
|
invoice: String,
|
|
fees: Int,
|
|
onSuccess: (String, String) -> Unit,
|
|
onError: (String, String) -> Unit,
|
|
context: Context,
|
|
) {
|
|
try {
|
|
val client = HttpClientManager.getHttpClient()
|
|
val url = token.mint + "/melt" // Melt cashu tokens at Mint
|
|
|
|
val factory = Event.mapper.nodeFactory
|
|
|
|
val jsonObject = factory.objectNode()
|
|
jsonObject.put("proofs", token.proofs)
|
|
jsonObject.put("pr", invoice)
|
|
|
|
val mediaType = "application/json; charset=utf-8".toMediaType()
|
|
val requestBody = jsonObject.toString().toRequestBody(mediaType)
|
|
val request = Request.Builder().url(url).post(requestBody).build()
|
|
|
|
client.newCall(request).execute().use {
|
|
val body = it.body.string()
|
|
val tree = jacksonObjectMapper().readTree(body)
|
|
|
|
val successful = tree?.get("paid")?.asText() == "true"
|
|
|
|
if (successful) {
|
|
onSuccess(
|
|
context.getString(R.string.cashu_successful_redemption),
|
|
context.getString(
|
|
R.string.cashu_successful_redemption_explainer,
|
|
token.totalAmount.toString(),
|
|
fees.toString(),
|
|
),
|
|
)
|
|
} else {
|
|
val msg = tree?.get("detail")?.asText()?.split('.')?.getOrNull(0)?.ifBlank { null }
|
|
onError(
|
|
context.getString(R.string.cashu_failed_redemption),
|
|
if (msg != null) {
|
|
context.getString(R.string.cashu_failed_redemption_explainer_error_msg, msg)
|
|
} else {
|
|
context.getString(R.string.cashu_failed_redemption_explainer_error_msg)
|
|
},
|
|
)
|
|
}
|
|
}
|
|
} catch (e: Exception) {
|
|
if (e is CancellationException) throw e
|
|
onError(
|
|
context.getString(R.string.cashu_successful_redemption),
|
|
context.getString(R.string.cashu_failed_redemption_explainer_error_msg, e.message),
|
|
)
|
|
}
|
|
}
|
|
}
|