From 50efa54a81277c798dba37b9dd36ac4e4d73a1c1 Mon Sep 17 00:00:00 2001 From: koalasat Date: Mon, 16 Sep 2024 12:11:56 +0200 Subject: [PATCH 01/85] Add extra tags to nostr --- api/nostr.py | 21 +++++++++++++++++---- 1 file changed, 17 insertions(+), 4 deletions(-) diff --git a/api/nostr.py b/api/nostr.py index 2792aa7d..6c2654b2 100644 --- a/api/nostr.py +++ b/api/nostr.py @@ -29,10 +29,13 @@ class Nostr: await client.connect() robot_name = await self.get_robot_name(order) + robot_hash_id = await self.get_robot_hash_id(order) currency = await self.get_robot_currency(order) event = EventBuilder( - Kind(38383), "", self.generate_tags(order, robot_name, currency) + Kind(38383), + "", + self.generate_tags(order, robot_name, robot_hash_id, currency), ).to_event(keys) await client.send_event(event) print(f"Nostr event sent: {event.as_json()}") @@ -41,18 +44,22 @@ class Nostr: def get_robot_name(self, order): return order.maker.username + @sync_to_async + def get_robot_hash_id(self, order): + return order.maker.robot.hash_id + @sync_to_async def get_robot_currency(self, order): return str(order.currency) - def generate_tags(self, order, robot_name, currency): + def generate_tags(self, order, robot_name, robot_hash_id, currency): hashed_id = hashlib.md5( f"{config("COORDINATOR_ALIAS", cast=str)}{order.id}".encode("utf-8") ).hexdigest() tags = [ Tag.parse(["d", str(uuid.UUID(hashed_id))]), - Tag.parse(["name", robot_name]), + Tag.parse(["name", robot_name, robot_hash_id]), Tag.parse(["k", "sell" if order.type == Order.Types.SELL else "buy"]), Tag.parse(["f", currency]), Tag.parse(["s", self.get_status_tag(order)]), @@ -73,7 +80,13 @@ class Nostr: f"http://{config("HOST_NAME")}/order/{config("COORDINATOR_ALIAS", cast=str).lower()}/{order.id}", ] ), - Tag.parse(["expiration", str(int(order.expires_at.timestamp()))]), + Tag.parse( + [ + "expiration", + str(int(order.expires_at.timestamp())), + str(order.escrow_duration), + ] + ), Tag.parse(["y", "robosats", config("COORDINATOR_ALIAS", cast=str).lower()]), Tag.parse(["n", str(config("NETWORK"))]), Tag.parse(["layer"] + self.get_layer_tag(order)), From a642ecb6f594e9a1253def84481ad07c43840688 Mon Sep 17 00:00:00 2001 From: koalasat Date: Mon, 16 Sep 2024 15:28:39 +0200 Subject: [PATCH 02/85] Fix trade summary sats --- frontend/src/components/OrderDetails/index.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/frontend/src/components/OrderDetails/index.tsx b/frontend/src/components/OrderDetails/index.tsx index a368df36..19a9e674 100644 --- a/frontend/src/components/OrderDetails/index.tsx +++ b/frontend/src/components/OrderDetails/index.tsx @@ -181,7 +181,7 @@ const OrderDetails = ({ const rate = Number(order.max_amount ?? order.amount) / btc_now; if (isBuyer) { - if (order.amount > 0) { + if (order.amount && order.amount > 0) { sats = computeSats({ amount: order.amount, fee: -tradeFee, @@ -211,7 +211,7 @@ const OrderDetails = ({ amount: sats, }); } else { - if (order.amount > 0) { + if (order.amount && order.amount > 0) { sats = computeSats({ amount: order.amount, fee: tradeFee, From 4f2ec020a9be5c16d7da0ddc28a6d637d0f619a7 Mon Sep 17 00:00:00 2001 From: koalasat Date: Mon, 26 Aug 2024 15:49:03 +0200 Subject: [PATCH 03/85] Store status notifications --- api/admin.py | 6 -- api/lightning/cln.py | 8 +- api/lightning/lnd.py | 7 +- api/logics.py | 72 ++++++------- api/management/commands/follow_invoices.py | 6 +- api/management/commands/telegram_watcher.py | 7 +- api/models/order.py | 5 +- api/notifications.py | 17 ++- api/tasks.py | 108 +++++++++++--------- chat/consumers.py | 5 +- chat/views.py | 6 +- tests/test_trade_pipeline.py | 54 +++++++++- tests/utils/node.py | 3 +- tests/utils/trade.py | 36 ++++--- 14 files changed, 189 insertions(+), 151 deletions(-) diff --git a/api/admin.py b/api/admin.py index 0ca8e536..e9957a1d 100644 --- a/api/admin.py +++ b/api/admin.py @@ -11,7 +11,6 @@ from rest_framework.authtoken.models import TokenProxy from api.logics import Logics from api.models import Currency, LNPayment, MarketTick, OnchainPayment, Order, Robot from api.utils import objects_to_hyperlinks -from api.tasks import send_notification admin.site.unregister(Group) admin.site.unregister(User) @@ -164,9 +163,6 @@ class OrderAdmin(AdminChangeLinksMixin, admin.ModelAdmin): f"Order {order.id} successfully closed", messages.SUCCESS, ) - send_notification.delay( - order_id=order.id, message="coordinator_cancelled" - ) else: self.message_user( request, @@ -210,7 +206,6 @@ class OrderAdmin(AdminChangeLinksMixin, admin.ModelAdmin): f"Dispute of order {order.id} solved successfully on favor of the maker", messages.SUCCESS, ) - send_notification.delay(order_id=order.id, message="dispute_closed") else: self.message_user( @@ -249,7 +244,6 @@ class OrderAdmin(AdminChangeLinksMixin, admin.ModelAdmin): f"Dispute of order {order.id} solved successfully on favor of the taker", messages.SUCCESS, ) - send_notification.delay(order_id=order.id, message="dispute_closed") else: self.message_user( diff --git a/api/lightning/cln.py b/api/lightning/cln.py index 82488417..0c0acba5 100755 --- a/api/lightning/cln.py +++ b/api/lightning/cln.py @@ -617,10 +617,10 @@ class CLNNode: ] ) - order.update_status(Order.Status.FAI) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.FAI) ) + order.update_status(Order.Status.FAI) order.save(update_fields=["expires_at"]) print( @@ -646,10 +646,10 @@ class CLNNode: ) lnpayment.preimage = response.payment_preimage.hex() lnpayment.save(update_fields=["status", "fee", "preimage"]) - order.update_status(Order.Status.SUC) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.SUC) ) + order.update_status(Order.Status.SUC) order.save(update_fields=["expires_at"]) order.log( @@ -697,10 +697,10 @@ class CLNNode: ] ) - order.update_status(Order.Status.FAI) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.FAI) ) + order.update_status(Order.Status.FAI) order.save(update_fields=["expires_at"]) print( @@ -737,10 +737,10 @@ class CLNNode: "in_flight", ] ) - order.update_status(Order.Status.FAI) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.FAI) ) + order.update_status(Order.Status.FAI) order.save(update_fields=["expires_at"]) order.log( diff --git a/api/lightning/lnd.py b/api/lightning/lnd.py index f914857e..96562a68 100644 --- a/api/lightning/lnd.py +++ b/api/lightning/lnd.py @@ -594,11 +594,10 @@ class LNDNode: ] ) - order.update_status(Order.Status.FAI) - order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.FAI) ) + order.update_status(Order.Status.FAI) order.save(update_fields=["expires_at"]) str_failure_reason = cls.payment_failure_context[ @@ -625,10 +624,10 @@ class LNDNode: lnpayment.preimage = response.payment_preimage lnpayment.save(update_fields=["status", "fee", "preimage"]) - order.update_status(Order.Status.SUC) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.SUC) ) + order.update_status(Order.Status.SUC) order.save(update_fields=["expires_at"]) order.log( @@ -671,10 +670,10 @@ class LNDNode: update_fields=["status", "last_routing_time", "in_flight"] ) - order.update_status(Order.Status.FAI) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.FAI) ) + order.update_status(Order.Status.FAI) order.save(update_fields=["expires_at"]) order.log( diff --git a/api/logics.py b/api/logics.py index 8bec242a..af5fe817 100644 --- a/api/logics.py +++ b/api/logics.py @@ -8,7 +8,7 @@ from django.utils import timezone from api.lightning.node import LNNode from api.models import Currency, LNPayment, MarketTick, OnchainPayment, Order -from api.tasks import send_devfund_donation, send_notification, nostr_send_order_event +from api.tasks import send_devfund_donation, send_status_notification, send_notification, nostr_send_order_event from api.utils import get_minning_fee, validate_onchain_address, location_country from chat.models import Message @@ -180,10 +180,10 @@ class Logics: if order.has_range: order.amount = amount order.taker = user - order.update_status(Order.Status.TAK) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.TAK) ) + order.update_status(Order.Status.TAK) order.save(update_fields=["amount", "taker", "expires_at"]) order.log( f"Taken by Robot({user.robot.id},{user.username}) for {order.amount} fiat units" @@ -267,9 +267,9 @@ class Logics: return False elif order.status == Order.Status.WFB: - order.update_status(Order.Status.EXP) order.expiry_reason = Order.ExpiryReasons.NMBOND cls.cancel_bond(order.maker_bond) + order.update_status(Order.Status.EXP) order.save(update_fields=["expiry_reason"]) order.log("Order expired while waiting for maker bond") @@ -279,10 +279,9 @@ class Logics: elif order.status in [Order.Status.PUB, Order.Status.PAU]: cls.return_bond(order.maker_bond) - order.update_status(Order.Status.EXP) order.expiry_reason = Order.ExpiryReasons.NTAKEN order.save(update_fields=["expiry_reason"]) - send_notification.delay(order_id=order.id, message="order_expired_untaken") + order.update_status(Order.Status.EXP) order.log("Order expired while public or paused") order.log("Maker bond was unlocked") @@ -307,8 +306,8 @@ class Logics: cls.settle_bond(order.maker_bond) cls.settle_bond(order.taker_bond) cls.cancel_escrow(order) - order.update_status(Order.Status.EXP) order.expiry_reason = Order.ExpiryReasons.NESINV + order.update_status(Order.Status.EXP) order.save(update_fields=["expiry_reason"]) order.log( @@ -330,8 +329,8 @@ class Logics: cls.cancel_escrow(order) except Exception: pass - order.update_status(Order.Status.EXP) order.expiry_reason = Order.ExpiryReasons.NESCRO + order.update_status(Order.Status.EXP) order.save(update_fields=["expiry_reason"]) # Reward taker with part of the maker bond cls.add_slashed_rewards(order, order.maker_bond, order.taker_bond) @@ -352,7 +351,9 @@ class Logics: pass taker_bond = order.taker_bond cls.publish_order(order) - send_notification.delay(order_id=order.id, message="order_published") + send_status_notification.delay( + order_id=order.id, status=Order.Status.PUB + ) # Reward maker with part of the taker bond cls.add_slashed_rewards(order, taker_bond, order.maker_bond) @@ -371,8 +372,8 @@ class Logics: cls.settle_bond(order.maker_bond) cls.return_bond(order.taker_bond) cls.return_escrow(order) - order.update_status(Order.Status.EXP) order.expiry_reason = Order.ExpiryReasons.NINVOI + order.update_status(Order.Status.EXP) order.save(update_fields=["expiry_reason"]) # Reward taker with part of the maker bond cls.add_slashed_rewards(order, order.maker_bond, order.taker_bond) @@ -389,7 +390,9 @@ class Logics: cls.return_escrow(order) taker_bond = order.taker_bond cls.publish_order(order) - send_notification.delay(order_id=order.id, message="order_published") + send_status_notification.delay( + order_id=order.id, status=Order.Status.PUB + ) # Reward maker with part of the taker bond cls.add_slashed_rewards(order, taker_bond, order.maker_bond) @@ -498,7 +501,6 @@ class Logics: seconds=order.t_to_expire(Order.Status.DIS) ) order.save(update_fields=["is_disputed", "expires_at"]) - send_notification.delay(order_id=order.id, message="dispute_opened") return True @@ -530,10 +532,10 @@ class Logics: cls.settle_bond(order.taker_bond) order.is_disputed = True - order.update_status(Order.Status.DIS) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.DIS) ) + order.update_status(Order.Status.DIS) order.save(update_fields=["is_disputed", "expires_at"]) # User could be None if a dispute is open automatically due to time expiration. @@ -548,7 +550,6 @@ class Logics: ).append(str(order.id)) robot.save(update_fields=["num_disputes", "orders_disputes_started"]) - send_notification.delay(order_id=order.id, message="dispute_opened") order.log( f"Dispute was opened {f'by Robot({user.robot.id},{user.username})' if user else ''}" ) @@ -587,10 +588,10 @@ class Logics: None, "", ]: - order.update_status(Order.Status.WFR) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.WFR) ) + order.update_status(Order.Status.WFR) order.save(update_fields=["status", "expires_at"]) order.log( @@ -944,11 +945,10 @@ class Logics: def move_state_updated_payout_method(cls, order): # If the order status is 'Waiting for invoice'. Move forward to 'chat' if order.status == Order.Status.WFI: - order.update_status(Order.Status.CHA) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.CHA) ) - send_notification.delay(order_id=order.id, message="fiat_exchange_starts") + order.update_status(Order.Status.CHA) # If the order status is 'Waiting for both'. Move forward to 'waiting for escrow' elif order.status == Order.Status.WF2: @@ -958,22 +958,19 @@ class Logics: # If the escrow is locked move to Chat. elif order.trade_escrow.status == LNPayment.Status.LOCKED: - order.update_status(Order.Status.CHA) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.CHA) ) - send_notification.delay( - order_id=order.id, message="fiat_exchange_starts" - ) + order.update_status(Order.Status.CHA) else: order.update_status(Order.Status.WFE) # If the order status is 'Failed Routing'. Retry payment. elif order.status == Order.Status.FAI: if LNNode.double_check_htlc_is_settled(order.trade_escrow.payment_hash): - order.update_status(Order.Status.PAY) order.payout.status = LNPayment.Status.FLIGHT order.payout.routing_attempts = 0 + order.update_status(Order.Status.PAY) order.payout.save(update_fields=["status", "routing_attempts"]) order.save(update_fields=["expires_at"]) @@ -1034,10 +1031,6 @@ class Logics: # Return the maker bond (Maker gets returned the bond for cancelling public order) if cls.return_bond(order.maker_bond): order.update_status(Order.Status.UCA) - send_notification.delay( - order_id=order.id, message="public_order_cancelled" - ) - order.log("Order cancelled by maker while public or paused") order.log("Maker bond was unlocked") @@ -1054,9 +1047,6 @@ class Logics: if cls.return_bond(order.maker_bond): cls.cancel_bond(order.taker_bond) order.update_status(Order.Status.UCA) - send_notification.delay( - order_id=order.id, message="public_order_cancelled" - ) order.log("Order cancelled by maker before the taker locked the bond") order.log("Maker bond was unlocked") @@ -1123,7 +1113,9 @@ class Logics: if valid: taker_bond = order.taker_bond cls.publish_order(order) - send_notification.delay(order_id=order.id, message="order_published") + send_status_notification.delay( + order_id=order.id, status=Order.Status.PUB + ) # Reward maker with part of the taker bond cls.add_slashed_rewards(order, taker_bond, order.maker_bond) @@ -1197,7 +1189,6 @@ class Logics: cls.return_bond(order.taker_bond) cls.return_escrow(order) order.update_status(Order.Status.CCA) - send_notification.delay(order_id=order.id, message="collaborative_cancelled") nostr_send_order_event.delay(order_id=order.id) @@ -1210,7 +1201,6 @@ class Logics: @classmethod def publish_order(cls, order): - order.status = Order.Status.PUB order.expires_at = order.created_at + timedelta( seconds=order.t_to_expire(Order.Status.PUB) ) @@ -1226,6 +1216,7 @@ class Logics: order.payout = None order.payout_tx = None + order.update_status(Order.Status.PUB) order.save() # update all fields nostr_send_order_event.delay(order_id=order.id) @@ -1344,7 +1335,6 @@ class Logics: order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.WF2) ) - order.status = Order.Status.WF2 order.save( update_fields=[ "status", @@ -1371,10 +1361,11 @@ class Logics: ) except Exception: pass - send_notification.delay(order_id=order.id, message="order_taken_confirmed") + + order.update_status(Order.Status.WF2) nostr_send_order_event.delay(order_id=order.id) - + order.log( f"Contract formalized. Maker: Robot({order.maker.robot.id},{order.maker}). Taker: Robot({order.taker.robot.id},{order.taker}). API median price {order.currency.exchange_rate} {dict(Currency.currency_choices)[order.currency.currency]}/BTC. Premium is {order.premium}%. Contract size {order.last_satoshis} Sats" ) @@ -1471,12 +1462,11 @@ class Logics: order.update_status(Order.Status.WFI) # If status is 'Waiting for invoice' move to Chat elif order.status == Order.Status.WFE: - order.update_status(Order.Status.CHA) order.expires_at = timezone.now() + timedelta( seconds=order.t_to_expire(Order.Status.CHA) ) + order.update_status(Order.Status.CHA) order.save(update_fields=["expires_at"]) - send_notification.delay(order_id=order.id, message="fiat_exchange_starts") @classmethod def gen_escrow_hold_invoice(cls, order, user): @@ -1644,11 +1634,10 @@ class Logics: order.payout.status = LNPayment.Status.FLIGHT order.payout.save(update_fields=["status"]) - order.update_status(Order.Status.PAY) order.contract_finalization_time = timezone.now() + order.update_status(Order.Status.PAY) order.save(update_fields=["contract_finalization_time"]) - send_notification.delay(order_id=order.id, message="trade_successful") order.log("Paying buyer invoice") return True @@ -1661,11 +1650,10 @@ class Logics: order.payout_tx.status = OnchainPayment.Status.QUEUE order.payout_tx.save(update_fields=["status"]) - order.update_status(Order.Status.SUC) order.contract_finalization_time = timezone.now() + order.update_status(Order.Status.SUC) order.save(update_fields=["contract_finalization_time"]) - send_notification.delay(order_id=order.id, message="trade_successful") order.log("Paying buyer onchain address") return True @@ -1679,8 +1667,8 @@ class Logics: if order.status == Order.Status.CHA or order.status == Order.Status.FSE: # If buyer mark fiat sent if cls.is_buyer(order, user): - order.update_status(Order.Status.FSE) order.is_fiat_sent = True + order.update_status(Order.Status.FSE) order.save(update_fields=["is_fiat_sent"]) order.log("Buyer confirmed 'fiat sent'") @@ -1744,9 +1732,9 @@ class Logics: return False, { "bad_request": "Only orders in Chat and with fiat sent confirmed can be reverted." } - order.update_status(Order.Status.CHA) order.is_fiat_sent = False order.reverted_fiat_sent = True + order.update_status(Order.Status.CHA) order.save(update_fields=["is_fiat_sent", "reverted_fiat_sent"]) order.log( diff --git a/api/management/commands/follow_invoices.py b/api/management/commands/follow_invoices.py index 8b4b98c1..93b1f34c 100644 --- a/api/management/commands/follow_invoices.py +++ b/api/management/commands/follow_invoices.py @@ -8,7 +8,7 @@ from django.utils import timezone from api.lightning.node import LNNode from api.logics import Logics from api.models import LNPayment, OnchainPayment, Order -from api.tasks import follow_send_payment, send_notification +from api.tasks import follow_send_payment, send_status_notification def is_same_status(a: LNPayment.Status, b: LNPayment.Status) -> bool: @@ -229,8 +229,8 @@ class Command(BaseCommand): if hasattr(lnpayment, "order_made"): lnpayment.order_made.log("Maker bond locked") Logics.publish_order(lnpayment.order_made) - send_notification.delay( - order_id=lnpayment.order_made.id, message="order_published" + send_status_notification.delay( + order_id=lnpayment.order_made.id, status=Order.Status.PUB ) return diff --git a/api/management/commands/telegram_watcher.py b/api/management/commands/telegram_watcher.py index d626b756..5f57dcae 100644 --- a/api/management/commands/telegram_watcher.py +++ b/api/management/commands/telegram_watcher.py @@ -6,8 +6,8 @@ from django.core.management.base import BaseCommand from django.db import transaction from api.models import Robot -from api.notifications import Notifications from api.utils import get_session +from api.tasks import send_telegram_notification class Command(BaseCommand): @@ -17,7 +17,6 @@ class Command(BaseCommand): bot_token = config("TELEGRAM_TOKEN") updates_url = f"https://api.telegram.org/bot{bot_token}/getUpdates" session = get_session() - notifications = Notifications() def handle(self, *args, **options): offset = 0 @@ -49,7 +48,7 @@ class Command(BaseCommand): continue parts = message.split(" ") if len(parts) < 2: - self.notifications.send_telegram_message( + send_telegram_notification.delay( result["message"]["from"]["id"], 'You must enable the notifications bot using the RoboSats client. Click on your "Robot robot" -> "Enable Telegram" and follow the link or scan the QR code.', ) @@ -57,7 +56,7 @@ class Command(BaseCommand): token = parts[-1] robot = Robot.objects.filter(telegram_token=token).first() if not robot: - self.notifications.send_telegram_message( + send_telegram_notification.delay( result["message"]["from"]["id"], f'Wops, invalid token! There is no Robot with telegram chat token "{token}"', ) diff --git a/api/models/order.py b/api/models/order.py index 483e6fa4..abd34a2e 100644 --- a/api/models/order.py +++ b/api/models/order.py @@ -10,7 +10,7 @@ from django.db import models from django.db.models.signals import pre_delete from django.dispatch import receiver from django.utils import timezone -from api.tasks import send_notification +from api.tasks import send_status_notification if config("TESTING", cast=bool, default=False): import random @@ -350,8 +350,7 @@ class Order(models.Model): self.log( f"Order state went from {old_status}: {Order.Status(old_status).label} to {new_status}: {Order.Status(new_status).label}" ) - if new_status == Order.Status.FAI: - send_notification.delay(order_id=self.id, message="lightning_failed") + send_status_notification.delay(order_id=self.id, status=self.status) @receiver(pre_delete, sender=Order) diff --git a/api/notifications.py b/api/notifications.py index 57ec4a42..7792d6a0 100644 --- a/api/notifications.py +++ b/api/notifications.py @@ -57,6 +57,18 @@ class Notifications: except Exception: pass + def status_change(self, order): + Notification.objects.create( + title="", description="", robot=order.maker.robot, order=order + ) + + if order.taker: + Notification.objects.create( + title="", description="", robot=order.taker.robot, order=order + ) + + return + def welcome(self, user): """User enabled Telegram Notifications""" lang = user.robot.telegram_lang_code @@ -215,11 +227,6 @@ class Notifications: return - def coordinator_cancelled(self, order): - title = f"🛠️ Your order with ID {order.id} has been cancelled by the coordinator {config('COORDINATOR_ALIAS', cast=str, default='NoAlias')} for the upcoming maintenance stop." - self.send_message(order, order.maker.robot, title) - return - def dispute_closed(self, order): lang = order.maker.robot.telegram_lang_code if order.status == Order.Status.MLD: diff --git a/api/tasks.py b/api/tasks.py index 416a5847..fffe0acf 100644 --- a/api/tasks.py +++ b/api/tasks.py @@ -251,7 +251,6 @@ def cache_market(): print("SOFT LIMIT REACHED. Could not fetch current external market prices.") return - @shared_task(name="", ignore_result=True, time_limit=120) def nostr_send_order_event(order_id=None): if order_id: @@ -265,63 +264,70 @@ def nostr_send_order_event(order_id=None): return - -@shared_task(name="send_notification", ignore_result=True, time_limit=120) -def send_notification(order_id=None, chat_message_id=None, message=None): - if order_id: +@shared_task(name="send_chat_notification", ignore_result=True, time_limit=120) +def send_chat_notification(message_id=None, order_id=None): + if message_id and order_id: from api.models import Order + from chat.models import Message + from api.notifications import Notifications order = Order.objects.get(id=order_id) - elif chat_message_id: - from chat.models import Message + chat_message = Message.objects.get(id=message_id) + notifications = Notifications() - chat_message = Message.objects.get(id=chat_message_id) - order = chat_message.order - - from api.notifications import Notifications - - notifications = Notifications() - - if message == "welcome": - notifications.welcome(order) - - elif message == "order_expired_untaken": - notifications.order_expired_untaken(order) - - elif message == "trade_successful": - notifications.trade_successful(order) - - elif message == "public_order_cancelled": - notifications.public_order_cancelled(order) - - elif message == "taker_expired_b4bond": - notifications.taker_expired_b4bond(order) - - elif message == "order_published": - notifications.order_published(order) - - elif message == "order_taken_confirmed": - notifications.order_taken_confirmed(order) - - elif message == "fiat_exchange_starts": - notifications.fiat_exchange_starts(order) - - elif message == "dispute_opened": - notifications.dispute_opened(order) - - elif message == "collaborative_cancelled": - notifications.collaborative_cancelled(order) - - elif message == "new_chat_message": notifications.new_chat_message(order, chat_message) - elif message == "coordinator_cancelled": - notifications.coordinator_cancelled(order) - elif message == "dispute_closed": - notifications.dispute_closed(order) +@shared_task(name="send_telegram_notification", ignore_result=True, time_limit=120) +def send_telegram_notification(chat_id=None, message=None): + if chat_id: + from api.notifications import Notifications - elif message == "lightning_failed": - notifications.lightning_failed(order) + notifications = Notifications() + + notifications.send_telegram_message(chat_id, message) + + +@shared_task(name="send_status_notification", ignore_result=True, time_limit=120) +def send_status_notification(order_id=None, status=None): + if order_id: + from api.models import Order + from api.notifications import Notifications + + order = Order.objects.get(id=order_id) + notifications = Notifications() + + if status == Order.Status.EXP: + notifications.order_expired_untaken(order) + + elif status == Order.Status.PAY or status == Order.Status.SUC: + notifications.trade_successful(order) + + elif status == Order.Status.UCA: + notifications.public_order_cancelled(order) + + elif status == Order.Status.PUB: + notifications.order_published(order) + + elif status == Order.Status.WF2: + notifications.order_taken_confirmed(order) + + elif status == Order.Status.CHA: + notifications.fiat_exchange_starts(order) + + elif status == Order.Status.DIS: + notifications.dispute_opened(order) + + elif status == Order.Status.CCA: + notifications.collaborative_cancelled(order) + + elif status == Order.Status.TLD or status == Order.Status.MLD: + notifications.dispute_closed(order) + + elif status == Order.Status.FAI: + notifications.lightning_failed(order) + + else: + notifications.status_change(order) return diff --git a/chat/consumers.py b/chat/consumers.py index c2443e66..1c32ac87 100644 --- a/chat/consumers.py +++ b/chat/consumers.py @@ -4,7 +4,7 @@ from channels.db import database_sync_to_async from channels.generic.websocket import AsyncWebsocketConsumer from api.models import Order -from api.tasks import send_notification +from api.tasks import send_chat_notification from chat.models import ChatRoom, Message @@ -85,7 +85,8 @@ class ChatRoomConsumer(AsyncWebsocketConsumer): ) # send Telegram notification for new message (if conditions apply) - send_notification.delay(chat_message_id=msg_obj.id, message="new_chat_message") + send_chat_notification.delay(message_id=msg_obj.id, order_id=order.id) + return msg_obj @database_sync_to_async diff --git a/chat/views.py b/chat/views.py index ab288169..fc15f73d 100644 --- a/chat/views.py +++ b/chat/views.py @@ -11,7 +11,7 @@ from rest_framework.permissions import IsAuthenticated from rest_framework.response import Response from api.models import Order -from api.tasks import send_notification +from api.tasks import send_chat_notification from chat.models import ChatRoom, Message from chat.serializers import ChatSerializer, InMessageSerializer, PostMessageSerializer @@ -179,9 +179,7 @@ class ChatView(viewsets.ViewSet): ) # send Telegram notification for new message (if conditions apply) - send_notification.delay( - chat_message_id=new_message.id, message="new_chat_message" - ) + send_chat_notification.delay(message_id=new_message.id, order_id=order.id) # Send websocket message if chatroom.maker == request.user: diff --git a/tests/test_trade_pipeline.py b/tests/test_trade_pipeline.py index c226b5c0..35ff1fd2 100644 --- a/tests/test_trade_pipeline.py +++ b/tests/test_trade_pipeline.py @@ -560,6 +560,21 @@ class TradeTest(BaseAPITestCase): trade.publish_order() trade.take_order() trade.lock_taker_bond() + + maker_nick = read_file(f"tests/robots/{trade.maker_index}/nickname") + taker_nick = read_file(f"tests/robots/{trade.taker_index}/nickname") + + maker_headers = trade.get_robot_auth(trade.maker_index) + response = self.client.get(reverse("notifications"), **maker_headers) + self.assertResponse(response) + notifications_data = list(response.json()) + self.assertEqual(notifications_data[0]["order_id"], trade.order_id) + # Does not receive notification because user is online + self.assertEqual( + notifications_data[0]["title"], + f"✅ Hey {maker_nick}, your order was taken by {taker_nick}!🥳", + ) + trade.lock_escrow(trade.taker_index) trade.submit_payout_invoice(trade.maker_index) @@ -571,7 +586,6 @@ class TradeTest(BaseAPITestCase): self.assertEqual(data["status_message"], Order.Status(Order.Status.CHA).label) self.assertFalse(data["is_fiat_sent"]) - maker_headers = trade.get_robot_auth(trade.maker_index) maker_nick = read_file(f"tests/robots/{trade.maker_index}/nickname") response = self.client.get(reverse("notifications"), **maker_headers) self.assertResponse(response) @@ -845,6 +859,16 @@ class TradeTest(BaseAPITestCase): self.assert_order_logs(data["id"]) + maker_headers = trade.get_robot_auth(trade.maker_index) + response = self.client.get(reverse("notifications"), **maker_headers) + self.assertResponse(response) + notifications_data = list(response.json()) + self.assertEqual(notifications_data[0]["order_id"], trade.order_id) + self.assertEqual( + notifications_data[0]["title"], + f"😪 Hey {data['maker_nick']}, your order with ID {str(trade.order_id)} has expired without a taker.", + ) + def test_public_order_expires(self): """ Tests the expiration of a public order @@ -917,6 +941,16 @@ class TradeTest(BaseAPITestCase): self.assert_order_logs(data["id"]) + maker_headers = trade.get_robot_auth(trade.maker_index) + response = self.client.get(reverse("notifications"), **maker_headers) + self.assertResponse(response) + notifications_data = list(response.json()) + self.assertEqual(notifications_data[0]["order_id"], trade.order_id) + self.assertEqual( + notifications_data[0]["title"], + f"😪 Hey {data['maker_nick']}, your order with ID {str(trade.order_id)} has expired without a taker.", + ) + def test_escrow_locked_expires(self): """ Tests the expiration of a public order @@ -953,6 +987,16 @@ class TradeTest(BaseAPITestCase): self.assert_order_logs(data["id"]) + maker_headers = trade.get_robot_auth(trade.maker_index) + response = self.client.get(reverse("notifications"), **maker_headers) + self.assertResponse(response) + notifications_data = list(response.json()) + self.assertEqual(notifications_data[0]["order_id"], trade.order_id) + self.assertEqual( + notifications_data[0]["title"], + f"😪 Hey {data['maker_nick']}, your order with ID {str(trade.order_id)} has expired without a taker.", + ) + def test_chat(self): """ Tests the chatting REST functionality @@ -1090,7 +1134,7 @@ class TradeTest(BaseAPITestCase): self.assertEqual(notifications_data[0]["order_id"], trade.order_id) self.assertEqual( notifications_data[0]["title"], - f"⚖️ Hey {data['maker_nick']}, a dispute has been opened on your order with ID {str(trade.order_id)}.", + f"⚖️ Hey {data['maker_nick']}, you lost the dispute on your order with ID {str(trade.order_id)}.", ) taker_headers = trade.get_robot_auth(trade.taker_index) response = self.client.get(reverse("notifications"), **taker_headers) @@ -1099,7 +1143,7 @@ class TradeTest(BaseAPITestCase): self.assertEqual(notifications_data[0]["order_id"], trade.order_id) self.assertEqual( notifications_data[0]["title"], - f"⚖️ Hey {data['taker_nick']}, a dispute has been opened on your order with ID {str(trade.order_id)}.", + f"⚖️ Hey {data['taker_nick']}, you won the dispute on your order with ID {str(trade.order_id)}.", ) def test_order_expires_after_only_maker_messaged(self): @@ -1151,7 +1195,7 @@ class TradeTest(BaseAPITestCase): self.assertEqual(notifications_data[0]["order_id"], trade.order_id) self.assertEqual( notifications_data[0]["title"], - f"⚖️ Hey {data['maker_nick']}, a dispute has been opened on your order with ID {str(trade.order_id)}.", + f"⚖️ Hey {data['maker_nick']}, you won the dispute on your order with ID {str(trade.order_id)}.", ) taker_headers = trade.get_robot_auth(trade.taker_index) response = self.client.get(reverse("notifications"), **taker_headers) @@ -1160,7 +1204,7 @@ class TradeTest(BaseAPITestCase): self.assertEqual(notifications_data[0]["order_id"], trade.order_id) self.assertEqual( notifications_data[0]["title"], - f"⚖️ Hey {data['taker_nick']}, a dispute has been opened on your order with ID {str(trade.order_id)}.", + f"⚖️ Hey {data['taker_nick']}, you lost the dispute on your order with ID {str(trade.order_id)}.", ) # def test_dispute_closed_maker_wins(self): diff --git a/tests/utils/node.py b/tests/utils/node.py index d5cb1289..002de9bc 100644 --- a/tests/utils/node.py +++ b/tests/utils/node.py @@ -269,7 +269,7 @@ def pay_invoice(node_name, invoice): node = get_node(node_name) data = {"payment_request": invoice} try: - response = requests.post( + requests.post( f'http://localhost:{node["port"]}/v1/channels/transactions', json=data, headers=node["headers"], @@ -277,7 +277,6 @@ def pay_invoice(node_name, invoice): # 0.4s is enough for LND to CLN hodl ACCEPT timeout=0.2 if LNVENDOR == "LND" else 1, ) - print(response.json()) except ReadTimeout: # Request to pay hodl invoice has timed out: that's good! return diff --git a/tests/utils/trade.py b/tests/utils/trade.py index 39bcda0a..f0258d67 100644 --- a/tests/utils/trade.py +++ b/tests/utils/trade.py @@ -5,7 +5,11 @@ from django.urls import reverse from api.management.commands.clean_orders import Command as CleanOrders from api.management.commands.follow_invoices import Command as FollowInvoices from api.models import Order -from api.tasks import follow_send_payment, send_notification +from api.tasks import ( + follow_send_payment, + send_status_notification, + send_chat_notification, +) from tests.utils.node import ( add_invoice, create_address, @@ -111,7 +115,7 @@ class Trade: headers = self.get_robot_auth(robot_index, first_encounter) self.response = self.client.get(path + params, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def cancel_order(self, robot_index=1): path = reverse("order") params = f"?order_id={self.order_id}" @@ -119,14 +123,14 @@ class Trade: body = {"action": "cancel"} self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_chat_notification.delay", send_chat_notification) def send_chat_message(self, message, robot_index=1): path = reverse("chat") headers = self.get_robot_auth(robot_index) body = {"PGP_message": message, "order_id": self.order_id, "offset": 0} self.response = self.client.post(path, data=body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def pause_order(self, robot_index=1): path = reverse("order") params = f"?order_id={self.order_id}" @@ -134,13 +138,13 @@ class Trade: body = {"action": "pause"} self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def follow_hold_invoices(self): # A background thread checks every 5 second the status of invoices. We invoke directly during test. follower = FollowInvoices() follower.follow_hold_invoices() - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def clean_orders(self): # A background thread checks every 5 second order expirations. We invoke directly during test. cleaner = CleanOrders() @@ -155,7 +159,7 @@ class Trade: generate_blocks(create_address("robot"), 1) wait_nodes_sync() - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def publish_order(self): # Maker's first order fetch. Should trigger maker bond hold invoice generation. self.get_order() @@ -170,7 +174,7 @@ class Trade: # Get order self.get_order() - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def take_order(self): path = reverse("order") params = f"?order_id={self.order_id}" @@ -178,7 +182,7 @@ class Trade: body = {"action": "take", "amount": self.take_amount} self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def lock_taker_bond(self): # Takers's first order fetch. Should trigger maker bond hold invoice generation. self.get_order(self.taker_index) @@ -193,7 +197,7 @@ class Trade: # Get order self.get_order(self.taker_index) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def lock_escrow(self, robot_index): # Takers's order fetch. Should trigger trade escrow bond hold invoice generation. self.get_order(robot_index) @@ -208,7 +212,7 @@ class Trade: # Get order self.get_order() - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def submit_payout_address(self, robot_index=1): path = reverse("order") params = f"?order_id={self.order_id}" @@ -227,7 +231,7 @@ class Trade: } self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def submit_payout_invoice(self, robot_index=1, routing_budget=0): path = reverse("order") params = f"?order_id={self.order_id}" @@ -249,7 +253,7 @@ class Trade: self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def confirm_fiat(self, robot_index=1): path = reverse("order") params = f"?order_id={self.order_id}" @@ -257,7 +261,7 @@ class Trade: body = {"action": "confirm"} self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def undo_confirm_sent(self, robot_index=1): path = reverse("order") params = f"?order_id={self.order_id}" @@ -265,14 +269,14 @@ class Trade: body = {"action": "undo_confirm"} self.response = self.client.post(path + params, body, **headers) - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def expire_order(self): # Change order expiry to now order = Order.objects.get(id=self.order_id) order.expires_at = datetime.now() order.save() - @patch("api.tasks.send_notification.delay", send_notification) + @patch("api.tasks.send_status_notification.delay", send_status_notification) def change_order_status(self, status): # Change order expiry to now order = Order.objects.get(id=self.order_id) From 0a33147a4f8301694cd86034eb7e6fcac7dfff31 Mon Sep 17 00:00:00 2001 From: koalasat Date: Mon, 16 Sep 2024 16:05:21 +0200 Subject: [PATCH 04/85] Fix Rebase --- api/logics.py | 8 ++++++-- docs/assets/schemas/api-latest.yaml | 2 +- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/api/logics.py b/api/logics.py index af5fe817..c5055fbd 100644 --- a/api/logics.py +++ b/api/logics.py @@ -8,7 +8,11 @@ from django.utils import timezone from api.lightning.node import LNNode from api.models import Currency, LNPayment, MarketTick, OnchainPayment, Order -from api.tasks import send_devfund_donation, send_status_notification, send_notification, nostr_send_order_event +from api.tasks import ( + send_devfund_donation, + send_status_notification, + nostr_send_order_event, +) from api.utils import get_minning_fee, validate_onchain_address, location_country from chat.models import Message @@ -1365,7 +1369,7 @@ class Logics: order.update_status(Order.Status.WF2) nostr_send_order_event.delay(order_id=order.id) - + order.log( f"Contract formalized. Maker: Robot({order.maker.robot.id},{order.maker}). Taker: Robot({order.taker.robot.id},{order.taker}). API median price {order.currency.exchange_rate} {dict(Currency.currency_choices)[order.currency.currency]}/BTC. Premium is {order.premium}%. Contract size {order.last_satoshis} Sats" ) diff --git a/docs/assets/schemas/api-latest.yaml b/docs/assets/schemas/api-latest.yaml index 04a0577f..2c8e044b 100644 --- a/docs/assets/schemas/api-latest.yaml +++ b/docs/assets/schemas/api-latest.yaml @@ -1,7 +1,7 @@ openapi: 3.0.3 info: title: RoboSats REST API - version: 0.6.3 + version: 0.7.0 x-logo: url: https://raw.githubusercontent.com/Reckless-Satoshi/robosats/main/frontend/static/assets/images/robosats-0.1.1-banner.png backgroundColor: '#FFFFFF' From bfda0e93be7d9b80e8d6abd65fd6fd3e47380921 Mon Sep 17 00:00:00 2001 From: koalasat Date: Tue, 17 Sep 2024 15:14:43 +0200 Subject: [PATCH 05/85] Fix files cache problem and static files management --- .gitignore | 15 +- desktopApp/index.html | 63 -- desktopApp/static | 1 - frontend/package-lock.json | 620 ++++++++++++++++++ frontend/package.json | 1 + frontend/src/App.tsx | 7 +- frontend/src/basic/Main.tsx | 5 +- frontend/src/basic/RobotPage/RobotProfile.tsx | 4 +- frontend/src/basic/RobotPage/index.tsx | 6 +- frontend/src/components/RobotAvatar/index.tsx | 4 +- .../src/components/SettingsForm/index.tsx | 4 +- .../TradeBox/Forms/LightningPayout.tsx | 6 +- .../src/components/TradeBox/TradeSummary.tsx | 6 +- frontend/src/contexts/AppContext.tsx | 28 +- frontend/src/contexts/FederationContext.tsx | 4 +- frontend/src/contexts/GarageContext.tsx | 6 +- frontend/src/models/Settings.model.ts | 4 +- frontend/src/services/Native/index.d.ts | 1 - frontend/templates/frontend/basic.html | 16 +- frontend/templates/frontend/pro.html | 20 +- frontend/webpack.config.ts | 100 ++- mobile/.gitignore | 3 +- mobile/html/Web.bundle/index.html | 55 -- nodeapp/basic.html | 63 -- nodeapp/docker-compose.yml | 14 +- nodeapp/nginx.conf | 8 - nodeapp/pro.html | 65 -- web/basic.html | 63 -- web/docker-compose.yml | 20 + web/pro.html | 65 -- 30 files changed, 810 insertions(+), 467 deletions(-) delete mode 100644 desktopApp/index.html delete mode 120000 desktopApp/static delete mode 100644 mobile/html/Web.bundle/index.html delete mode 100644 nodeapp/basic.html delete mode 100644 nodeapp/pro.html delete mode 100644 web/basic.html create mode 100644 web/docker-compose.yml delete mode 100644 web/pro.html diff --git a/.gitignore b/.gitignore index 40a4ba98..5b3d8cc7 100755 --- a/.gitignore +++ b/.gitignore @@ -646,10 +646,17 @@ docs/.jekyll-cache* docs/_site* node -# mobile frontend statics -mobile/html/Web.bundle/js* -mobile/html/Web.bundle/css* -mobile/html/Web.bundle/assets* +# frontend statics +mobile/html/Web.bundle/static/* +# mobile/html/Web.bundle/index.html +desktopApp/static/* +# desktopApp/index.html +web/static/* +# web/basic.html +# web/pro.html +nodeapp/static/* +# nodeapp/basic.html +# nodeapp/pro.html # Protocol Buffers api/lightning/*.proto diff --git a/desktopApp/index.html b/desktopApp/index.html deleted file mode 100644 index 7045672b..00000000 --- a/desktopApp/index.html +++ /dev/null @@ -1,63 +0,0 @@ - - - - - - - - - - - - - RoboSats - Simple and Private Bitcoin Exchange - - - - - - - - -
-
-
-
-
-
-
-
    -
  • -
    Looking for robot parts ...
    -
  • -
  • -
    Adding layers to the onion ...
    -
  • -
  • -
    Winning at game theory ...
    -
  • -
  • -
    Moving Sats at light speed ...
    -
  • -
  • -
    Hiding in 2^256 bits of entropy...
    -
  • -
-
-
-
-
-
- - - - diff --git a/desktopApp/static b/desktopApp/static deleted file mode 120000 index 9653c34d..00000000 --- a/desktopApp/static +++ /dev/null @@ -1 +0,0 @@ -../frontend/static \ No newline at end of file diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 5187a5bf..9b256585 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -78,6 +78,7 @@ "eslint-plugin-promise": "^6.1.1", "eslint-plugin-react": "^7.34.0", "eslint-plugin-react-hooks": "^4.6.0", + "html-webpack-plugin": "^5.6.0", "jest": "^29.6.1", "prettier": "^3.3.3", "ts-node": "^10.9.2", @@ -4462,6 +4463,13 @@ "@types/node": "*" } }, + "node_modules/@types/html-minifier-terser": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/@types/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz", + "integrity": "sha512-oh/6byDPnL1zeNXFrDXFLyZjkr1MsBG667IM792caf1L2UPOOMf65NFzjUH/ltyfwjAGfs1rsX1eftK0jC/KIg==", + "dev": true, + "license": "MIT" + }, "node_modules/@types/istanbul-lib-coverage": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", @@ -6003,6 +6011,17 @@ "node": ">=6" } }, + "node_modules/camel-case": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.2.tgz", + "integrity": "sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "pascal-case": "^3.1.2", + "tslib": "^2.0.3" + } + }, "node_modules/camelcase": { "version": "5.3.1", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", @@ -6203,6 +6222,29 @@ "integrity": "sha512-0TNiGstbQmCFwt4akjjBg5pLRTSyj/PkWQ1ZoO2zntmg9yLqSRxwEa4iCfQLGjqhiqBfOJa7W/E8wfGrTDmlZQ==", "dev": true }, + "node_modules/clean-css": { + "version": "5.3.3", + "resolved": "https://registry.npmjs.org/clean-css/-/clean-css-5.3.3.tgz", + "integrity": "sha512-D5J+kHaVb/wKSFcyyV75uCn8fiY4sV38XJoe4CUyGQ+mOU/fMVYUdH1hJC+CJQ5uY3EnW27SbJYS4X8BiLrAFg==", + "dev": true, + "license": "MIT", + "dependencies": { + "source-map": "~0.6.0" + }, + "engines": { + "node": ">= 10.0" + } + }, + "node_modules/clean-css/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/cliui": { "version": "8.0.1", "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", @@ -6929,6 +6971,16 @@ "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.6.3.tgz", "integrity": "sha512-7ZgogeTnjuHbo+ct10G9Ffp0mif17idi0IyWNVA/wcwcm7NPOD/WEHVP3n7n3MhXqxoIYm8d6MuZohYWIZ4T3w==" }, + "node_modules/dom-converter": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/dom-converter/-/dom-converter-0.2.0.tgz", + "integrity": "sha512-gd3ypIPfOMr9h5jIKq8E3sHOTCjeirnl0WK5ZdS1AW0Odt0b1PaWaHdJ4Qk4klv+YB9aJBS7mESXjFoDQPu6DA==", + "dev": true, + "license": "MIT", + "dependencies": { + "utila": "~0.4" + } + }, "node_modules/dom-helpers": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/dom-helpers/-/dom-helpers-5.2.1.tgz", @@ -6989,6 +7041,17 @@ "url": "https://github.com/fb55/domutils?sponsor=1" } }, + "node_modules/dot-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.4.tgz", + "integrity": "sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==", + "dev": true, + "license": "MIT", + "dependencies": { + "no-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, "node_modules/electron-to-chromium": { "version": "1.4.461", "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.461.tgz", @@ -8741,6 +8804,38 @@ "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", "dev": true }, + "node_modules/html-minifier-terser": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/html-minifier-terser/-/html-minifier-terser-6.1.0.tgz", + "integrity": "sha512-YXxSlJBZTP7RS3tWnQw74ooKa6L9b9i9QYXY21eUEvhZ3u9XLfv6OnFsQq6RxkhHygsaUMvYsZRV5rU/OVNZxw==", + "dev": true, + "license": "MIT", + "dependencies": { + "camel-case": "^4.1.2", + "clean-css": "^5.2.2", + "commander": "^8.3.0", + "he": "^1.2.0", + "param-case": "^3.0.4", + "relateurl": "^0.2.7", + "terser": "^5.10.0" + }, + "bin": { + "html-minifier-terser": "cli.js" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/html-minifier-terser/node_modules/commander": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", + "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 12" + } + }, "node_modules/html-parse-stringify": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/html-parse-stringify/-/html-parse-stringify-3.0.1.tgz", @@ -8749,6 +8844,39 @@ "void-elements": "3.1.0" } }, + "node_modules/html-webpack-plugin": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/html-webpack-plugin/-/html-webpack-plugin-5.6.0.tgz", + "integrity": "sha512-iwaY4wzbe48AfKLZ/Cc8k0L+FKG6oSNRaZ8x5A/T/IVDGyXcbHncM9TdDa93wn0FsSm82FhTKW7f3vS61thXAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/html-minifier-terser": "^6.0.0", + "html-minifier-terser": "^6.0.2", + "lodash": "^4.17.21", + "pretty-error": "^4.0.0", + "tapable": "^2.0.0" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/html-webpack-plugin" + }, + "peerDependencies": { + "@rspack/core": "0.x || 1.x", + "webpack": "^5.20.0" + }, + "peerDependenciesMeta": { + "@rspack/core": { + "optional": true + }, + "webpack": { + "optional": true + } + } + }, "node_modules/htmlparser2": { "version": "8.0.2", "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.2.tgz", @@ -11549,6 +11677,16 @@ "get-func-name": "^2.0.1" } }, + "node_modules/lower-case": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz", + "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==", + "dev": true, + "license": "MIT", + "dependencies": { + "tslib": "^2.0.3" + } + }, "node_modules/lru-cache": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", @@ -11929,6 +12067,17 @@ "@sinonjs/commons": "^3.0.0" } }, + "node_modules/no-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz", + "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "lower-case": "^2.0.2", + "tslib": "^2.0.3" + } + }, "node_modules/node-fetch": { "version": "2.7.0", "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", @@ -12143,6 +12292,8 @@ }, "node_modules/npm/node_modules/@isaacs/cliui": { "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@isaacs/cliui/-/cliui-8.0.2.tgz", + "integrity": "sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12170,6 +12321,8 @@ }, "node_modules/npm/node_modules/@isaacs/cliui/node_modules/emoji-regex": { "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", "inBundle": true, "license": "MIT" }, @@ -12191,6 +12344,8 @@ }, "node_modules/npm/node_modules/@isaacs/cliui/node_modules/strip-ansi": { "version": "7.1.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", + "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12205,11 +12360,15 @@ }, "node_modules/npm/node_modules/@isaacs/string-locale-compare": { "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@isaacs/string-locale-compare/-/string-locale-compare-1.1.0.tgz", + "integrity": "sha512-SQ7Kzhh9+D+ZW9MA0zkYv3VXhIDNx+LzM6EJ+/65I3QY+enU6Itte7E5XX7EWrqLW2FN4n06GWzBnPoC3th2aQ==", "inBundle": true, "license": "ISC" }, "node_modules/npm/node_modules/@npmcli/agent": { "version": "2.2.2", + "resolved": "https://registry.npmjs.org/@npmcli/agent/-/agent-2.2.2.tgz", + "integrity": "sha512-OrcNPXdpSl9UX7qPVRWbmWMCSXrcDa2M9DvrbOTj7ao1S4PlqVFYv9/yLKMkrJKZ/V5A/kDBC690or307i26Og==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12291,6 +12450,8 @@ }, "node_modules/npm/node_modules/@npmcli/fs": { "version": "3.1.1", + "resolved": "https://registry.npmjs.org/@npmcli/fs/-/fs-3.1.1.tgz", + "integrity": "sha512-q9CRWjpHCMIh5sVyefoD1cA7PkvILqCZsnSOEUUivORLjxCO/Irmue2DprETiNgEqktDBZaM1Bi+jrarx1XdCg==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12320,6 +12481,8 @@ }, "node_modules/npm/node_modules/@npmcli/installed-package-contents": { "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@npmcli/installed-package-contents/-/installed-package-contents-2.1.0.tgz", + "integrity": "sha512-c8UuGLeZpm69BryRykLuKRyKFZYJsZSCT4aVY5ds4omyZqJ172ApzgfKJ5eV/r3HgLdUYgFVe54KSFVjKoe27w==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12335,6 +12498,8 @@ }, "node_modules/npm/node_modules/@npmcli/map-workspaces": { "version": "3.0.6", + "resolved": "https://registry.npmjs.org/@npmcli/map-workspaces/-/map-workspaces-3.0.6.tgz", + "integrity": "sha512-tkYs0OYnzQm6iIRdfy+LcLBjcKuQCeE5YLb8KnrIlutJfheNaPvPpgoFEyEFgbjzl5PLZ3IA/BWAwRU0eHuQDA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12349,6 +12514,8 @@ }, "node_modules/npm/node_modules/@npmcli/metavuln-calculator": { "version": "7.1.1", + "resolved": "https://registry.npmjs.org/@npmcli/metavuln-calculator/-/metavuln-calculator-7.1.1.tgz", + "integrity": "sha512-Nkxf96V0lAx3HCpVda7Vw4P23RILgdi/5K1fmj2tZkWIYLpXAN8k2UVVOsW16TsS5F8Ws2I7Cm+PU1/rsVF47g==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12364,6 +12531,8 @@ }, "node_modules/npm/node_modules/@npmcli/name-from-folder": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@npmcli/name-from-folder/-/name-from-folder-2.0.0.tgz", + "integrity": "sha512-pwK+BfEBZJbKdNYpHHRTNBwBoqrN/iIMO0AiGvYsp3Hoaq0WbgGSWQR6SCldZovoDpY3yje5lkFUe6gsDgJ2vg==", "inBundle": true, "license": "ISC", "engines": { @@ -12372,6 +12541,8 @@ }, "node_modules/npm/node_modules/@npmcli/node-gyp": { "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@npmcli/node-gyp/-/node-gyp-3.0.0.tgz", + "integrity": "sha512-gp8pRXC2oOxu0DUE1/M3bYtb1b3/DbJ5aM113+XJBgfXdussRAsX0YOrOhdd8WvnAR6auDBvJomGAkLKA5ydxA==", "inBundle": true, "license": "ISC", "engines": { @@ -12397,6 +12568,8 @@ }, "node_modules/npm/node_modules/@npmcli/promise-spawn": { "version": "7.0.2", + "resolved": "https://registry.npmjs.org/@npmcli/promise-spawn/-/promise-spawn-7.0.2.tgz", + "integrity": "sha512-xhfYPXoV5Dy4UkY0D+v2KkwvnDfiA/8Mt3sWCGI/hM03NsYIH8ZaG6QzS9x7pje5vHZBZJ2v6VRFVTWACnqcmQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12408,6 +12581,8 @@ }, "node_modules/npm/node_modules/@npmcli/query": { "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@npmcli/query/-/query-3.1.0.tgz", + "integrity": "sha512-C/iR0tk7KSKGldibYIB9x8GtO/0Bd0I2mhOaDb8ucQL/bQVTmGoeREaFj64Z5+iCBRf3dQfed0CjJL7I8iTkiQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12427,6 +12602,8 @@ }, "node_modules/npm/node_modules/@npmcli/run-script": { "version": "8.1.0", + "resolved": "https://registry.npmjs.org/@npmcli/run-script/-/run-script-8.1.0.tgz", + "integrity": "sha512-y7efHHwghQfk28G2z3tlZ67pLG0XdfYbcVG26r7YIXALRsrVQcTq4/tdenSmdOrEsNahIYA/eh8aEVROWGFUDg==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12443,6 +12620,8 @@ }, "node_modules/npm/node_modules/@pkgjs/parseargs": { "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@pkgjs/parseargs/-/parseargs-0.11.0.tgz", + "integrity": "sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==", "inBundle": true, "license": "MIT", "optional": true, @@ -12452,6 +12631,8 @@ }, "node_modules/npm/node_modules/@sigstore/bundle": { "version": "2.3.2", + "resolved": "https://registry.npmjs.org/@sigstore/bundle/-/bundle-2.3.2.tgz", + "integrity": "sha512-wueKWDk70QixNLB363yHc2D2ItTgYiMTdPwK8D9dKQMR3ZQ0c35IxP5xnwQ8cNLoCgCRcHf14kE+CLIvNX1zmA==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -12463,6 +12644,8 @@ }, "node_modules/npm/node_modules/@sigstore/core": { "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@sigstore/core/-/core-1.1.0.tgz", + "integrity": "sha512-JzBqdVIyqm2FRQCulY6nbQzMpJJpSiJ8XXWMhtOX9eKgaXXpfNOF53lzQEjIydlStnd/eFtuC1dW4VYdD93oRg==", "inBundle": true, "license": "Apache-2.0", "engines": { @@ -12471,6 +12654,8 @@ }, "node_modules/npm/node_modules/@sigstore/protobuf-specs": { "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@sigstore/protobuf-specs/-/protobuf-specs-0.3.2.tgz", + "integrity": "sha512-c6B0ehIWxMI8wiS/bj6rHMPqeFvngFV7cDU/MY+B16P9Z3Mp9k8L93eYZ7BYzSickzuqAQqAq0V956b3Ju6mLw==", "inBundle": true, "license": "Apache-2.0", "engines": { @@ -12479,6 +12664,8 @@ }, "node_modules/npm/node_modules/@sigstore/sign": { "version": "2.3.2", + "resolved": "https://registry.npmjs.org/@sigstore/sign/-/sign-2.3.2.tgz", + "integrity": "sha512-5Vz5dPVuunIIvC5vBb0APwo7qKA4G9yM48kPWJT+OEERs40md5GoUR1yedwpekWZ4m0Hhw44m6zU+ObsON+iDA==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -12495,6 +12682,8 @@ }, "node_modules/npm/node_modules/@sigstore/tuf": { "version": "2.3.4", + "resolved": "https://registry.npmjs.org/@sigstore/tuf/-/tuf-2.3.4.tgz", + "integrity": "sha512-44vtsveTPUpqhm9NCrbU8CWLe3Vck2HO1PNLw7RIajbB7xhtn5RBPm1VNSCMwqGYHhDsBJG8gDF0q4lgydsJvw==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -12507,6 +12696,8 @@ }, "node_modules/npm/node_modules/@sigstore/verify": { "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@sigstore/verify/-/verify-1.2.1.tgz", + "integrity": "sha512-8iKx79/F73DKbGfRf7+t4dqrc0bRr0thdPrxAtCKWRm/F0tG71i6O1rvlnScncJLLBZHn3h8M3c1BSUAb9yu8g==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -12520,6 +12711,8 @@ }, "node_modules/npm/node_modules/@tufjs/canonical-json": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@tufjs/canonical-json/-/canonical-json-2.0.0.tgz", + "integrity": "sha512-yVtV8zsdo8qFHe+/3kw81dSLyF7D576A5cCFCi4X7B39tWT7SekaEFUnvnWJHz+9qO7qJTah1JbrDjWKqFtdWA==", "inBundle": true, "license": "MIT", "engines": { @@ -12528,6 +12721,8 @@ }, "node_modules/npm/node_modules/@tufjs/models": { "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@tufjs/models/-/models-2.0.1.tgz", + "integrity": "sha512-92F7/SFyufn4DXsha9+QfKnN03JGqtMFMXgSHbZOo8JG59WkTni7UzAouNQDf7AuP9OAMxVOPQcqG3sB7w+kkg==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12540,6 +12735,8 @@ }, "node_modules/npm/node_modules/abbrev": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-2.0.0.tgz", + "integrity": "sha512-6/mh1E2u2YgEsCHdY0Yx5oW+61gZU+1vXaoiHHrpKeuRNNgFvS+/jrwHiQhB5apAf5oB7UB7E19ol2R2LKH8hQ==", "inBundle": true, "license": "ISC", "engines": { @@ -12548,6 +12745,8 @@ }, "node_modules/npm/node_modules/agent-base": { "version": "7.1.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.1.tgz", + "integrity": "sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12559,6 +12758,8 @@ }, "node_modules/npm/node_modules/aggregate-error": { "version": "3.1.0", + "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", + "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12571,6 +12772,8 @@ }, "node_modules/npm/node_modules/ansi-regex": { "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", "inBundle": true, "license": "MIT", "engines": { @@ -12579,6 +12782,8 @@ }, "node_modules/npm/node_modules/ansi-styles": { "version": "6.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.2.1.tgz", + "integrity": "sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==", "inBundle": true, "license": "MIT", "engines": { @@ -12590,21 +12795,29 @@ }, "node_modules/npm/node_modules/aproba": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/aproba/-/aproba-2.0.0.tgz", + "integrity": "sha512-lYe4Gx7QT+MKGbDsA+Z+he/Wtef0BiwDOlK/XkBrdfsh9J/jPPXbX0tE9x9cl27Tmu5gg3QUbUrQYa/y+KOHPQ==", "inBundle": true, "license": "ISC" }, "node_modules/npm/node_modules/archy": { "version": "1.0.0", + "resolved": "https://registry.npmjs.org/archy/-/archy-1.0.0.tgz", + "integrity": "sha512-Xg+9RwCg/0p32teKdGMPTPnVXKD0w3DfHnFTficozsAgsvq2XenPJq/MYpzzQ/v8zrOyJn6Ds39VA4JIDwFfqw==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/balanced-match": { "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/bin-links": { "version": "4.0.4", + "resolved": "https://registry.npmjs.org/bin-links/-/bin-links-4.0.4.tgz", + "integrity": "sha512-cMtq4W5ZsEwcutJrVId+a/tjt8GSbS+h0oNkdl6+6rBuEv8Ot33Bevj5KPm40t309zuhVic8NjpuL42QCiJWWA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12619,6 +12832,8 @@ }, "node_modules/npm/node_modules/binary-extensions": { "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", "inBundle": true, "license": "MIT", "engines": { @@ -12630,6 +12845,8 @@ }, "node_modules/npm/node_modules/brace-expansion": { "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12660,6 +12877,8 @@ }, "node_modules/npm/node_modules/chalk": { "version": "5.3.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.3.0.tgz", + "integrity": "sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==", "inBundle": true, "license": "MIT", "engines": { @@ -12671,6 +12890,8 @@ }, "node_modules/npm/node_modules/chownr": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", "inBundle": true, "license": "ISC", "engines": { @@ -12679,6 +12900,8 @@ }, "node_modules/npm/node_modules/ci-info": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-4.0.0.tgz", + "integrity": "sha512-TdHqgGf9odd8SXNuxtUBVx8Nv+qZOejE6qyqiy5NtbYYQOeFa6zmHkxlPzmaLxWWHsU6nJmB7AETdVPi+2NBUg==", "funding": [ { "type": "github", @@ -12693,6 +12916,8 @@ }, "node_modules/npm/node_modules/cidr-regex": { "version": "4.1.1", + "resolved": "https://registry.npmjs.org/cidr-regex/-/cidr-regex-4.1.1.tgz", + "integrity": "sha512-ekKcVp+iRB9zlKFXyx7io7nINgb0oRjgRdXNEodp1OuxRui8FXr/CA40Tz1voWUp9DPPrMyQKy01vJhDo4N1lw==", "inBundle": true, "license": "BSD-2-Clause", "dependencies": { @@ -12704,6 +12929,8 @@ }, "node_modules/npm/node_modules/clean-stack": { "version": "2.2.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", + "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", "inBundle": true, "license": "MIT", "engines": { @@ -12712,6 +12939,8 @@ }, "node_modules/npm/node_modules/cli-columns": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/cli-columns/-/cli-columns-4.0.0.tgz", + "integrity": "sha512-XW2Vg+w+L9on9wtwKpyzluIPCWXjaBahI7mTcYjx+BVIYD9c3yqcv/yKC7CmdCZat4rq2yiE1UMSJC5ivKfMtQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12724,6 +12953,8 @@ }, "node_modules/npm/node_modules/cmd-shim": { "version": "6.0.3", + "resolved": "https://registry.npmjs.org/cmd-shim/-/cmd-shim-6.0.3.tgz", + "integrity": "sha512-FMabTRlc5t5zjdenF6mS0MBeFZm0XqHqeOkcskKFb/LYCcRQ5fVgLOHVc4Lq9CqABd9zhjwPjMBCJvMCziSVtA==", "inBundle": true, "license": "ISC", "engines": { @@ -12732,6 +12963,8 @@ }, "node_modules/npm/node_modules/color-convert": { "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12743,16 +12976,22 @@ }, "node_modules/npm/node_modules/color-name": { "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/common-ancestor-path": { "version": "1.0.1", + "resolved": "https://registry.npmjs.org/common-ancestor-path/-/common-ancestor-path-1.0.1.tgz", + "integrity": "sha512-L3sHRo1pXXEqX8VU28kfgUY+YGsk09hPqZiZmLacNib6XNTCM8ubYeT7ryXQw8asB1sKgcU5lkB7ONug08aB8w==", "inBundle": true, "license": "ISC" }, "node_modules/npm/node_modules/cross-spawn": { "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12780,6 +13019,8 @@ }, "node_modules/npm/node_modules/cssesc": { "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", "inBundle": true, "license": "MIT", "bin": { @@ -12791,6 +13032,8 @@ }, "node_modules/npm/node_modules/debug": { "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12812,6 +13055,8 @@ }, "node_modules/npm/node_modules/diff": { "version": "5.2.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-5.2.0.tgz", + "integrity": "sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==", "inBundle": true, "license": "BSD-3-Clause", "engines": { @@ -12820,16 +13065,22 @@ }, "node_modules/npm/node_modules/eastasianwidth": { "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/emoji-regex": { "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/encoding": { "version": "0.1.13", + "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.13.tgz", + "integrity": "sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==", "inBundle": true, "license": "MIT", "optional": true, @@ -12839,6 +13090,8 @@ }, "node_modules/npm/node_modules/env-paths": { "version": "2.2.1", + "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", + "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", "inBundle": true, "license": "MIT", "engines": { @@ -12847,16 +13100,22 @@ }, "node_modules/npm/node_modules/err-code": { "version": "2.0.3", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-2.0.3.tgz", + "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/exponential-backoff": { "version": "3.1.1", + "resolved": "https://registry.npmjs.org/exponential-backoff/-/exponential-backoff-3.1.1.tgz", + "integrity": "sha512-dX7e/LHVJ6W3DE1MHWi9S1EYzDESENfLrYohG2G++ovZrYOkm4Knwa0mc1cn84xJOR4KEU0WSchhLbd0UklbHw==", "inBundle": true, "license": "Apache-2.0" }, "node_modules/npm/node_modules/fastest-levenshtein": { "version": "1.0.16", + "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz", + "integrity": "sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg==", "inBundle": true, "license": "MIT", "engines": { @@ -12880,6 +13139,8 @@ }, "node_modules/npm/node_modules/fs-minipass": { "version": "3.0.3", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-3.0.3.tgz", + "integrity": "sha512-XUBA9XClHbnJWSfBzjkm6RvPsyg3sryZt06BEQoXcF7EK/xpGaQYJgQKDJSUH5SGZ76Y7pFx1QBnXz09rU5Fbw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12891,6 +13152,8 @@ }, "node_modules/npm/node_modules/function-bind": { "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", "inBundle": true, "license": "MIT", "funding": { @@ -12920,11 +13183,15 @@ }, "node_modules/npm/node_modules/graceful-fs": { "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", "inBundle": true, "license": "ISC" }, "node_modules/npm/node_modules/hasown": { "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12936,6 +13203,8 @@ }, "node_modules/npm/node_modules/hosted-git-info": { "version": "7.0.2", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-7.0.2.tgz", + "integrity": "sha512-puUZAUKT5m8Zzvs72XWy3HtvVbTWljRE66cP60bxJzAqf2DgICo7lYTY2IHUmLnNpjYvw5bvmoHvPc0QO2a62w==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12947,11 +13216,15 @@ }, "node_modules/npm/node_modules/http-cache-semantics": { "version": "4.1.1", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz", + "integrity": "sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ==", "inBundle": true, "license": "BSD-2-Clause" }, "node_modules/npm/node_modules/http-proxy-agent": { "version": "7.0.2", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz", + "integrity": "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig==", "inBundle": true, "license": "MIT", "dependencies": { @@ -12976,6 +13249,8 @@ }, "node_modules/npm/node_modules/iconv-lite": { "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", "inBundle": true, "license": "MIT", "optional": true, @@ -12988,6 +13263,8 @@ }, "node_modules/npm/node_modules/ignore-walk": { "version": "6.0.5", + "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-6.0.5.tgz", + "integrity": "sha512-VuuG0wCnjhnylG1ABXT3dAuIpTNDs/G8jlpmwXY03fXoXy/8ZK8/T+hMzt8L4WnrLCJgdybqgPagnF/f97cg3A==", "inBundle": true, "license": "ISC", "dependencies": { @@ -12999,6 +13276,8 @@ }, "node_modules/npm/node_modules/imurmurhash": { "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", "inBundle": true, "license": "MIT", "engines": { @@ -13007,6 +13286,8 @@ }, "node_modules/npm/node_modules/indent-string": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", "inBundle": true, "license": "MIT", "engines": { @@ -13015,6 +13296,8 @@ }, "node_modules/npm/node_modules/ini": { "version": "4.1.3", + "resolved": "https://registry.npmjs.org/ini/-/ini-4.1.3.tgz", + "integrity": "sha512-X7rqawQBvfdjS10YU1y1YVreA3SsLrW9dX2CewP2EbBJM4ypVNLDkO5y04gejPwKIY9lR+7r9gn3rFPt/kmWFg==", "inBundle": true, "license": "ISC", "engines": { @@ -13023,6 +13306,8 @@ }, "node_modules/npm/node_modules/init-package-json": { "version": "6.0.3", + "resolved": "https://registry.npmjs.org/init-package-json/-/init-package-json-6.0.3.tgz", + "integrity": "sha512-Zfeb5ol+H+eqJWHTaGca9BovufyGeIfr4zaaBorPmJBMrJ+KBnN+kQx2ZtXdsotUTgldHmHQV44xvUWOUA7E2w==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13040,6 +13325,8 @@ }, "node_modules/npm/node_modules/ip-address": { "version": "9.0.5", + "resolved": "https://registry.npmjs.org/ip-address/-/ip-address-9.0.5.tgz", + "integrity": "sha512-zHtQzGojZXTwZTHQqra+ETKd4Sn3vgi7uBmlPoXVWZqYvuKmtI0l/VZTjqGmJY9x88GGOaZ9+G9ES8hC4T4X8g==", "inBundle": true, "license": "MIT", "dependencies": { @@ -13052,6 +13339,8 @@ }, "node_modules/npm/node_modules/ip-regex": { "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-5.0.0.tgz", + "integrity": "sha512-fOCG6lhoKKakwv+C6KdsOnGvgXnmgfmp0myi3bcNwj3qfwPAxRKWEuFhvEFF7ceYIz6+1jRZ+yguLFAmUNPEfw==", "inBundle": true, "license": "MIT", "engines": { @@ -13063,6 +13352,8 @@ }, "node_modules/npm/node_modules/is-cidr": { "version": "5.1.0", + "resolved": "https://registry.npmjs.org/is-cidr/-/is-cidr-5.1.0.tgz", + "integrity": "sha512-OkVS+Ht2ssF27d48gZdB+ho1yND1VbkJRKKS6Pc1/Cw7uqkd9IOJg8/bTwBDQL6tfBhSdguPRnlGiE8pU/X5NQ==", "inBundle": true, "license": "BSD-2-Clause", "dependencies": { @@ -13085,6 +13376,8 @@ }, "node_modules/npm/node_modules/is-fullwidth-code-point": { "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", "inBundle": true, "license": "MIT", "engines": { @@ -13093,11 +13386,15 @@ }, "node_modules/npm/node_modules/is-lambda": { "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-lambda/-/is-lambda-1.0.1.tgz", + "integrity": "sha512-z7CMFGNrENq5iFB9Bqo64Xk6Y9sg+epq1myIcdHaGnbMTYOxvzsEtdYqQUylB7LxfkvgrrjP32T6Ywciio9UIQ==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/isexe": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", "inBundle": true, "license": "ISC" }, @@ -13120,11 +13417,15 @@ }, "node_modules/npm/node_modules/jsbn": { "version": "1.1.0", + "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-1.1.0.tgz", + "integrity": "sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/json-parse-even-better-errors": { "version": "3.0.2", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-3.0.2.tgz", + "integrity": "sha512-fi0NG4bPjCHunUJffmLd0gxssIgkNmArMvis4iNah6Owg1MCJjWhEcDLmsK6iGkJq3tHwbDkTlce70/tmXN4cQ==", "inBundle": true, "license": "MIT", "engines": { @@ -13133,6 +13434,8 @@ }, "node_modules/npm/node_modules/json-stringify-nice": { "version": "1.1.4", + "resolved": "https://registry.npmjs.org/json-stringify-nice/-/json-stringify-nice-1.1.4.tgz", + "integrity": "sha512-5Z5RFW63yxReJ7vANgW6eZFGWaQvnPE3WNmZoOJrSkGju2etKA2L5rrOa1sm877TVTFt57A80BH1bArcmlLfPw==", "inBundle": true, "license": "ISC", "funding": { @@ -13141,6 +13444,8 @@ }, "node_modules/npm/node_modules/jsonparse": { "version": "1.3.1", + "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", + "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==", "engines": [ "node >= 0.2.0" ], @@ -13149,16 +13454,22 @@ }, "node_modules/npm/node_modules/just-diff": { "version": "6.0.2", + "resolved": "https://registry.npmjs.org/just-diff/-/just-diff-6.0.2.tgz", + "integrity": "sha512-S59eriX5u3/QhMNq3v/gm8Kd0w8OS6Tz2FS1NG4blv+z0MuQcBRJyFWjdovM0Rad4/P4aUPFtnkNjMjyMlMSYA==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/just-diff-apply": { "version": "5.5.0", + "resolved": "https://registry.npmjs.org/just-diff-apply/-/just-diff-apply-5.5.0.tgz", + "integrity": "sha512-OYTthRfSh55WOItVqwpefPtNt2VdKsq5AnAK6apdtR6yCH8pr0CmSr710J0Mf+WdQy7K/OzMy7K2MgAfdQURDw==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/libnpmaccess": { "version": "8.0.6", + "resolved": "https://registry.npmjs.org/libnpmaccess/-/libnpmaccess-8.0.6.tgz", + "integrity": "sha512-uM8DHDEfYG6G5gVivVl+yQd4pH3uRclHC59lzIbSvy7b5FEwR+mU49Zq1jEyRtRFv7+M99mUW9S0wL/4laT4lw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13220,6 +13531,8 @@ }, "node_modules/npm/node_modules/libnpmhook": { "version": "10.0.5", + "resolved": "https://registry.npmjs.org/libnpmhook/-/libnpmhook-10.0.5.tgz", + "integrity": "sha512-XulT+N/s3o9oFlIq6pGRv3OG2qR1NVRbVQOKLchycDwyf16RZA3oXbeEgs2H3oE7hRZPUMBZqsalQXMMPal3cQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13232,6 +13545,8 @@ }, "node_modules/npm/node_modules/libnpmorg": { "version": "6.0.6", + "resolved": "https://registry.npmjs.org/libnpmorg/-/libnpmorg-6.0.6.tgz", + "integrity": "sha512-4MVxsAS4H2z7su/sU0GsrirfBm4ssfqPRSDvoZ8qmRw58kEWJ0qE0cQ2VilRlFgCWKzKPhfoPeyNPyxBTnOusA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13258,6 +13573,8 @@ }, "node_modules/npm/node_modules/libnpmpublish": { "version": "9.0.9", + "resolved": "https://registry.npmjs.org/libnpmpublish/-/libnpmpublish-9.0.9.tgz", + "integrity": "sha512-26zzwoBNAvX9AWOPiqqF6FG4HrSCPsHFkQm7nT+xU1ggAujL/eae81RnCv4CJ2In9q9fh10B88sYSzKCUh/Ghg==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13276,6 +13593,8 @@ }, "node_modules/npm/node_modules/libnpmsearch": { "version": "7.0.6", + "resolved": "https://registry.npmjs.org/libnpmsearch/-/libnpmsearch-7.0.6.tgz", + "integrity": "sha512-PmiER4bgiIqN9OjBtgPn2/PxwU+OdJWtLBFM+vewOrn4VmaNAHSUKDt/wxOOkZSDLyMICVUBp61Ji1+XxhSrKw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13287,6 +13606,8 @@ }, "node_modules/npm/node_modules/libnpmteam": { "version": "6.0.5", + "resolved": "https://registry.npmjs.org/libnpmteam/-/libnpmteam-6.0.5.tgz", + "integrity": "sha512-iJW4Cq42GMqMwZEV+Mx8ZLj0Np5kGXQ9P/BAekHjIpYC1v3/vJqbmfJkzkwFvGxEhUotmx+xpLChZCDJ7c3rxA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13299,6 +13620,8 @@ }, "node_modules/npm/node_modules/libnpmversion": { "version": "6.0.3", + "resolved": "https://registry.npmjs.org/libnpmversion/-/libnpmversion-6.0.3.tgz", + "integrity": "sha512-Kjk1anQ9sPn7E/qF1jXumItvr2OA1914tYWkSTXH9G2rYoY+Ol1+KNrWfGeje2aBvFfKlt4VeKdCfM3yxMXNBw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13322,6 +13645,8 @@ }, "node_modules/npm/node_modules/make-fetch-happen": { "version": "13.0.1", + "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-13.0.1.tgz", + "integrity": "sha512-cKTUFc/rbKUd/9meOvgrpJ2WrNzymt6jfRDdwg5UCnVzv9dTpEj9JS5m3wtziXVCjluIXyL8pcaukYqezIzZQA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13358,6 +13683,8 @@ }, "node_modules/npm/node_modules/minipass": { "version": "7.1.2", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz", + "integrity": "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==", "inBundle": true, "license": "ISC", "engines": { @@ -13366,6 +13693,8 @@ }, "node_modules/npm/node_modules/minipass-collect": { "version": "2.0.1", + "resolved": "https://registry.npmjs.org/minipass-collect/-/minipass-collect-2.0.1.tgz", + "integrity": "sha512-D7V8PO9oaz7PWGLbCACuI1qEOsq7UKfLotx/C0Aet43fCUB/wfQ7DYeq2oR/svFJGYDHPr38SHATeaj/ZoKHKw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13377,6 +13706,8 @@ }, "node_modules/npm/node_modules/minipass-fetch": { "version": "3.0.5", + "resolved": "https://registry.npmjs.org/minipass-fetch/-/minipass-fetch-3.0.5.tgz", + "integrity": "sha512-2N8elDQAtSnFV0Dk7gt15KHsS0Fyz6CbYZ360h0WTYV1Ty46li3rAXVOQj1THMNLdmrD9Vt5pBPtWtVkpwGBqg==", "inBundle": true, "license": "MIT", "dependencies": { @@ -13393,6 +13724,8 @@ }, "node_modules/npm/node_modules/minipass-flush": { "version": "1.0.5", + "resolved": "https://registry.npmjs.org/minipass-flush/-/minipass-flush-1.0.5.tgz", + "integrity": "sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13435,6 +13768,8 @@ }, "node_modules/npm/node_modules/minipass-pipeline": { "version": "1.2.4", + "resolved": "https://registry.npmjs.org/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz", + "integrity": "sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13457,6 +13792,8 @@ }, "node_modules/npm/node_modules/minipass-sized": { "version": "1.0.3", + "resolved": "https://registry.npmjs.org/minipass-sized/-/minipass-sized-1.0.3.tgz", + "integrity": "sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13479,6 +13816,8 @@ }, "node_modules/npm/node_modules/minizlib": { "version": "2.1.2", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", "inBundle": true, "license": "MIT", "dependencies": { @@ -13502,6 +13841,8 @@ }, "node_modules/npm/node_modules/mkdirp": { "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", "inBundle": true, "license": "MIT", "bin": { @@ -13513,11 +13854,15 @@ }, "node_modules/npm/node_modules/ms": { "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/mute-stream": { "version": "1.0.0", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-1.0.0.tgz", + "integrity": "sha512-avsJQhyd+680gKXyG/sQc0nXaC6rBkPOfyHYcFb9+hdkqQkR9bdnkJ0AMZhke0oesPqIO+mFFJ+IdBc7mst4IA==", "inBundle": true, "license": "ISC", "engines": { @@ -13526,6 +13871,8 @@ }, "node_modules/npm/node_modules/negotiator": { "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", "inBundle": true, "license": "MIT", "engines": { @@ -13565,6 +13912,8 @@ }, "node_modules/npm/node_modules/nopt": { "version": "7.2.1", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-7.2.1.tgz", + "integrity": "sha512-taM24ViiimT/XntxbPyJQzCG+p4EKOpgD3mxFwW38mGjVUrfERQOeY4EDHjdnptttfHuHQXFx+lTP08Q+mLa/w==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13593,6 +13942,8 @@ }, "node_modules/npm/node_modules/npm-audit-report": { "version": "5.0.0", + "resolved": "https://registry.npmjs.org/npm-audit-report/-/npm-audit-report-5.0.0.tgz", + "integrity": "sha512-EkXrzat7zERmUhHaoren1YhTxFwsOu5jypE84k6632SXTHcQE1z8V51GC6GVZt8LxkC+tbBcKMUBZAgk8SUSbw==", "inBundle": true, "license": "ISC", "engines": { @@ -13601,6 +13952,8 @@ }, "node_modules/npm/node_modules/npm-bundled": { "version": "3.0.1", + "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-3.0.1.tgz", + "integrity": "sha512-+AvaheE/ww1JEwRHOrn4WHNzOxGtVp+adrg2AeZS/7KuxGUYFuBta98wYpfHBbJp6Tg6j1NKSEVHNcfZzJHQwQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13612,6 +13965,8 @@ }, "node_modules/npm/node_modules/npm-install-checks": { "version": "6.3.0", + "resolved": "https://registry.npmjs.org/npm-install-checks/-/npm-install-checks-6.3.0.tgz", + "integrity": "sha512-W29RiK/xtpCGqn6f3ixfRYGk+zRyr+Ew9F2E20BfXxT5/euLdA/Nm7fO7OeTGuAmTs30cpgInyJ0cYe708YTZw==", "inBundle": true, "license": "BSD-2-Clause", "dependencies": { @@ -13623,6 +13978,8 @@ }, "node_modules/npm/node_modules/npm-normalize-package-bin": { "version": "3.0.1", + "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-3.0.1.tgz", + "integrity": "sha512-dMxCf+zZ+3zeQZXKxmyuCKlIDPGuv8EF940xbkC4kQVDTtqoh6rJFO+JTKSA6/Rwi0getWmtuy4Itup0AMcaDQ==", "inBundle": true, "license": "ISC", "engines": { @@ -13645,6 +14002,8 @@ }, "node_modules/npm/node_modules/npm-packlist": { "version": "8.0.2", + "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-8.0.2.tgz", + "integrity": "sha512-shYrPFIS/JLP4oQmAwDyk5HcyysKW8/JLTEA32S0Z5TzvpaeeX2yMFfoK1fjEBnCBvVyIB/Jj/GBFdm0wsgzbA==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13670,6 +14029,8 @@ }, "node_modules/npm/node_modules/npm-profile": { "version": "10.0.0", + "resolved": "https://registry.npmjs.org/npm-profile/-/npm-profile-10.0.0.tgz", + "integrity": "sha512-DXnge3nHYnEnPxmVd/kPmgcXKXwVUqFihGnU+EJUiu5mIOs3awq6zEm0rRp3kSQNhFsoqdLu8L1TIfRyeBOCog==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13700,6 +14061,8 @@ }, "node_modules/npm/node_modules/npm-user-validate": { "version": "2.0.1", + "resolved": "https://registry.npmjs.org/npm-user-validate/-/npm-user-validate-2.0.1.tgz", + "integrity": "sha512-d17PKaF2h8LSGFl5j4b1gHOJt1fgH7YUcCm1kNSJvaLWWKXlBsuUvx0bBEkr0qhsVA9XP5LtRZ83hdlhm2QkgA==", "inBundle": true, "license": "BSD-2-Clause", "engines": { @@ -13708,6 +14071,8 @@ }, "node_modules/npm/node_modules/p-map": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -13722,6 +14087,8 @@ }, "node_modules/npm/node_modules/pacote": { "version": "18.0.6", + "resolved": "https://registry.npmjs.org/pacote/-/pacote-18.0.6.tgz", + "integrity": "sha512-+eK3G27SMwsB8kLIuj4h1FUhHtwiEUo21Tw8wNjmvdlpOEr613edv+8FUsTj/4F/VN5ywGE19X18N7CC2EJk6A==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13752,6 +14119,8 @@ }, "node_modules/npm/node_modules/parse-conflict-json": { "version": "3.0.1", + "resolved": "https://registry.npmjs.org/parse-conflict-json/-/parse-conflict-json-3.0.1.tgz", + "integrity": "sha512-01TvEktc68vwbJOtWZluyWeVGWjP+bZwXtPDMQVbBKzbJ/vZBif0L69KH1+cHv1SZ6e0FKLvjyHe8mqsIqYOmw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13765,6 +14134,8 @@ }, "node_modules/npm/node_modules/path-key": { "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", "inBundle": true, "license": "MIT", "engines": { @@ -13773,6 +14144,8 @@ }, "node_modules/npm/node_modules/path-scurry": { "version": "1.11.1", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.11.1.tgz", + "integrity": "sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA==", "inBundle": true, "license": "BlueOak-1.0.0", "dependencies": { @@ -13800,6 +14173,8 @@ }, "node_modules/npm/node_modules/proc-log": { "version": "4.2.0", + "resolved": "https://registry.npmjs.org/proc-log/-/proc-log-4.2.0.tgz", + "integrity": "sha512-g8+OnU/L2v+wyiVK+D5fA34J7EH8jZ8DDlvwhRCMxmMj7UCBvxiO1mGeN+36JXIKF4zevU4kRBd8lVgG9vLelA==", "inBundle": true, "license": "ISC", "engines": { @@ -13808,6 +14183,8 @@ }, "node_modules/npm/node_modules/proggy": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/proggy/-/proggy-2.0.0.tgz", + "integrity": "sha512-69agxLtnI8xBs9gUGqEnK26UfiexpHy+KUpBQWabiytQjnn5wFY8rklAi7GRfABIuPNnQ/ik48+LGLkYYJcy4A==", "inBundle": true, "license": "ISC", "engines": { @@ -13816,6 +14193,8 @@ }, "node_modules/npm/node_modules/promise-all-reject-late": { "version": "1.0.1", + "resolved": "https://registry.npmjs.org/promise-all-reject-late/-/promise-all-reject-late-1.0.1.tgz", + "integrity": "sha512-vuf0Lf0lOxyQREH7GDIOUMLS7kz+gs8i6B+Yi8dC68a2sychGrHTJYghMBD6k7eUcH0H5P73EckCA48xijWqXw==", "inBundle": true, "license": "ISC", "funding": { @@ -13832,11 +14211,15 @@ }, "node_modules/npm/node_modules/promise-inflight": { "version": "1.0.1", + "resolved": "https://registry.npmjs.org/promise-inflight/-/promise-inflight-1.0.1.tgz", + "integrity": "sha512-6zWPyEOFaQBJYcGMHBKTKJ3u6TBsnMFOIZSa6ce1e/ZrrsOlnHRHbabMjLiBYKp+n44X9eUI6VUPaukCXHuG4g==", "inBundle": true, "license": "ISC" }, "node_modules/npm/node_modules/promise-retry": { "version": "2.0.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-2.0.1.tgz", + "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", "inBundle": true, "license": "MIT", "dependencies": { @@ -13849,6 +14232,8 @@ }, "node_modules/npm/node_modules/promzard": { "version": "1.0.2", + "resolved": "https://registry.npmjs.org/promzard/-/promzard-1.0.2.tgz", + "integrity": "sha512-2FPputGL+mP3jJ3UZg/Dl9YOkovB7DX0oOr+ck5QbZ5MtORtds8k/BZdn+02peDLI8/YWbmzx34k5fA+fHvCVQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13860,6 +14245,8 @@ }, "node_modules/npm/node_modules/qrcode-terminal": { "version": "0.12.0", + "resolved": "https://registry.npmjs.org/qrcode-terminal/-/qrcode-terminal-0.12.0.tgz", + "integrity": "sha512-EXtzRZmC+YGmGlDFbXKxQiMZNwCLEO6BANKXG4iCtSIM0yqc/pappSx3RIKr4r0uh5JsBckOXeKrB3Iz7mdQpQ==", "inBundle": true, "bin": { "qrcode-terminal": "bin/qrcode-terminal.js" @@ -13867,6 +14254,8 @@ }, "node_modules/npm/node_modules/read": { "version": "3.0.1", + "resolved": "https://registry.npmjs.org/read/-/read-3.0.1.tgz", + "integrity": "sha512-SLBrDU/Srs/9EoWhU5GdbAoxG1GzpQHo/6qiGItaoLJ1thmYpcNIM1qISEUvyHBzfGlWIyd6p2DNi1oV1VmAuw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13878,6 +14267,8 @@ }, "node_modules/npm/node_modules/read-cmd-shim": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/read-cmd-shim/-/read-cmd-shim-4.0.0.tgz", + "integrity": "sha512-yILWifhaSEEytfXI76kB9xEEiG1AiozaCJZ83A87ytjRiN+jVibXjedjCRNjoZviinhG+4UkalO3mWTd8u5O0Q==", "inBundle": true, "license": "ISC", "engines": { @@ -13886,6 +14277,8 @@ }, "node_modules/npm/node_modules/read-package-json-fast": { "version": "3.0.2", + "resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-3.0.2.tgz", + "integrity": "sha512-0J+Msgym3vrLOUB3hzQCuZHII0xkNGCtz/HJH9xZshwv9DbDwkw1KaE3gx/e2J5rpEY5rtOy6cyhKOPrkP7FZw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -13898,6 +14291,8 @@ }, "node_modules/npm/node_modules/retry": { "version": "0.12.0", + "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz", + "integrity": "sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow==", "inBundle": true, "license": "MIT", "engines": { @@ -13906,6 +14301,8 @@ }, "node_modules/npm/node_modules/safer-buffer": { "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", "inBundle": true, "license": "MIT", "optional": true @@ -13923,6 +14320,8 @@ }, "node_modules/npm/node_modules/shebang-command": { "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", "inBundle": true, "license": "MIT", "dependencies": { @@ -13934,6 +14333,8 @@ }, "node_modules/npm/node_modules/shebang-regex": { "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "inBundle": true, "license": "MIT", "engines": { @@ -13942,6 +14343,8 @@ }, "node_modules/npm/node_modules/signal-exit": { "version": "4.1.0", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", + "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", "inBundle": true, "license": "ISC", "engines": { @@ -13953,6 +14356,8 @@ }, "node_modules/npm/node_modules/sigstore": { "version": "2.3.1", + "resolved": "https://registry.npmjs.org/sigstore/-/sigstore-2.3.1.tgz", + "integrity": "sha512-8G+/XDU8wNsJOQS5ysDVO0Etg9/2uA5gR9l4ZwijjlwxBcrU6RPfwi2+jJmbP+Ap1Hlp/nVAaEO4Fj22/SL2gQ==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -13969,6 +14374,8 @@ }, "node_modules/npm/node_modules/smart-buffer": { "version": "4.2.0", + "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", + "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", "inBundle": true, "license": "MIT", "engines": { @@ -14004,6 +14411,8 @@ }, "node_modules/npm/node_modules/spdx-correct": { "version": "3.2.0", + "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.2.0.tgz", + "integrity": "sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -14022,11 +14431,15 @@ }, "node_modules/npm/node_modules/spdx-exceptions": { "version": "2.5.0", + "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz", + "integrity": "sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w==", "inBundle": true, "license": "CC-BY-3.0" }, "node_modules/npm/node_modules/spdx-expression-parse": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-4.0.0.tgz", + "integrity": "sha512-Clya5JIij/7C6bRR22+tnGXbc4VKlibKSVj2iHvVeX5iMW7s1SIQlqu699JkODJJIhh/pUu8L0/VLh8xflD+LQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14041,11 +14454,15 @@ }, "node_modules/npm/node_modules/sprintf-js": { "version": "1.1.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.3.tgz", + "integrity": "sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==", "inBundle": true, "license": "BSD-3-Clause" }, "node_modules/npm/node_modules/ssri": { "version": "10.0.6", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-10.0.6.tgz", + "integrity": "sha512-MGrFH9Z4NP9Iyhqn16sDtBpRRNJ0Y2hNa6D65h736fVSaPCHr4DM4sWUNvVaSuC+0OBGhwsrydQwmgfg5LncqQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -14057,6 +14474,8 @@ }, "node_modules/npm/node_modules/string-width": { "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14071,6 +14490,8 @@ "node_modules/npm/node_modules/string-width-cjs": { "name": "string-width", "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14084,6 +14505,8 @@ }, "node_modules/npm/node_modules/strip-ansi": { "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14096,6 +14519,8 @@ "node_modules/npm/node_modules/strip-ansi-cjs": { "name": "strip-ansi", "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14107,6 +14532,8 @@ }, "node_modules/npm/node_modules/supports-color": { "version": "9.4.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-9.4.0.tgz", + "integrity": "sha512-VL+lNrEoIXww1coLPOmiEmK/0sGigko5COxI09KzHc2VJXJsQ37UaQ+8quuxjDeA7+KnLGTWRyOXSLLR2Wb4jw==", "inBundle": true, "license": "MIT", "engines": { @@ -14118,6 +14545,8 @@ }, "node_modules/npm/node_modules/tar": { "version": "6.2.1", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.2.1.tgz", + "integrity": "sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A==", "inBundle": true, "license": "ISC", "dependencies": { @@ -14164,16 +14593,22 @@ }, "node_modules/npm/node_modules/text-table": { "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/tiny-relative-date": { "version": "1.3.0", + "resolved": "https://registry.npmjs.org/tiny-relative-date/-/tiny-relative-date-1.3.0.tgz", + "integrity": "sha512-MOQHpzllWxDCHHaDno30hhLfbouoYlOI8YlMNtvKe1zXbjEVhbcEovQxvZrPvtiYW630GQDoMMarCnjfyfHA+A==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/treeverse": { "version": "3.0.0", + "resolved": "https://registry.npmjs.org/treeverse/-/treeverse-3.0.0.tgz", + "integrity": "sha512-gcANaAnd2QDZFmHFEOF4k7uc1J/6a6z3DJMd/QwEyxLoKGiptJRwid582r7QIsFlFMIZ3SnxfS52S4hm2DHkuQ==", "inBundle": true, "license": "ISC", "engines": { @@ -14182,6 +14617,8 @@ }, "node_modules/npm/node_modules/tuf-js": { "version": "2.2.1", + "resolved": "https://registry.npmjs.org/tuf-js/-/tuf-js-2.2.1.tgz", + "integrity": "sha512-GwIJau9XaA8nLVbUXsN3IlFi7WmQ48gBUrl3FTkkL/XLu/POhBzfmX9hd33FNMX1qAsfl6ozO1iMmW9NC8YniA==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14195,6 +14632,8 @@ }, "node_modules/npm/node_modules/unique-filename": { "version": "3.0.0", + "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-3.0.0.tgz", + "integrity": "sha512-afXhuC55wkAmZ0P18QsVE6kp8JaxrEokN2HGIoIVv2ijHQd419H0+6EigAFcIzXeMIkcIkNBpB3L/DXB3cTS/g==", "inBundle": true, "license": "ISC", "dependencies": { @@ -14206,6 +14645,8 @@ }, "node_modules/npm/node_modules/unique-slug": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/unique-slug/-/unique-slug-4.0.0.tgz", + "integrity": "sha512-WrcA6AyEfqDX5bWige/4NQfPZMtASNVxdmWR76WESYQVAACSgWcR6e9i0mofqqBxYFtL4oAxPIptY73/0YE1DQ==", "inBundle": true, "license": "ISC", "dependencies": { @@ -14217,11 +14658,15 @@ }, "node_modules/npm/node_modules/util-deprecate": { "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", "inBundle": true, "license": "MIT" }, "node_modules/npm/node_modules/validate-npm-package-license": { "version": "3.0.4", + "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", + "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", "inBundle": true, "license": "Apache-2.0", "dependencies": { @@ -14240,6 +14685,8 @@ }, "node_modules/npm/node_modules/validate-npm-package-name": { "version": "5.0.1", + "resolved": "https://registry.npmjs.org/validate-npm-package-name/-/validate-npm-package-name-5.0.1.tgz", + "integrity": "sha512-OljLrQ9SQdOUqTaQxqL5dEfZWrXExyyWsozYlAWFawPVNuD83igl7uJD2RTkNMbniIYgt8l81eCJGIdQF7avLQ==", "inBundle": true, "license": "ISC", "engines": { @@ -14248,11 +14695,15 @@ }, "node_modules/npm/node_modules/walk-up-path": { "version": "3.0.1", + "resolved": "https://registry.npmjs.org/walk-up-path/-/walk-up-path-3.0.1.tgz", + "integrity": "sha512-9YlCL/ynK3CTlrSRrDxZvUauLzAswPCrsaCgilqFevUYpeEW0/3ScEjaa3kbW/T0ghhkEr7mv+fpjqn1Y1YuTA==", "inBundle": true, "license": "ISC" }, "node_modules/npm/node_modules/which": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/which/-/which-4.0.0.tgz", + "integrity": "sha512-GlaYyEb07DPxYCKhKzplCWBJtvxZcZMrL+4UkrTSJHHPyZU4mYYTv3qaOe77H7EODLSSopAUFAc6W8U4yqvscg==", "inBundle": true, "license": "ISC", "dependencies": { @@ -14275,6 +14726,8 @@ }, "node_modules/npm/node_modules/wrap-ansi": { "version": "8.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-8.1.0.tgz", + "integrity": "sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14292,6 +14745,8 @@ "node_modules/npm/node_modules/wrap-ansi-cjs": { "name": "wrap-ansi", "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14333,6 +14788,8 @@ }, "node_modules/npm/node_modules/wrap-ansi/node_modules/emoji-regex": { "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", "inBundle": true, "license": "MIT" }, @@ -14354,6 +14811,8 @@ }, "node_modules/npm/node_modules/wrap-ansi/node_modules/strip-ansi": { "version": "7.1.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.1.0.tgz", + "integrity": "sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==", "inBundle": true, "license": "MIT", "dependencies": { @@ -14368,6 +14827,8 @@ }, "node_modules/npm/node_modules/write-file-atomic": { "version": "5.0.1", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-5.0.1.tgz", + "integrity": "sha512-+QU2zd6OTD8XWIJCbffaiQeH9U73qIqafo1x6V1snCWYGJf6cVE0cDR4D8xRzcEnfI21IFrUPzPGtcPf8AC+Rw==", "inBundle": true, "license": "ISC", "dependencies": { @@ -14380,6 +14841,8 @@ }, "node_modules/npm/node_modules/yallist": { "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", "inBundle": true, "license": "ISC" }, @@ -14600,6 +15063,17 @@ "node": ">=6" } }, + "node_modules/param-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.4.tgz", + "integrity": "sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==", + "dev": true, + "license": "MIT", + "dependencies": { + "dot-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, "node_modules/parent-module": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", @@ -14651,6 +15125,17 @@ "url": "https://github.com/inikulin/parse5?sponsor=1" } }, + "node_modules/pascal-case": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.2.tgz", + "integrity": "sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==", + "dev": true, + "license": "MIT", + "dependencies": { + "no-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, "node_modules/path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -14971,6 +15456,17 @@ "node": ">=6.0.0" } }, + "node_modules/pretty-error": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/pretty-error/-/pretty-error-4.0.0.tgz", + "integrity": "sha512-AoJ5YMAcXKYxKhuJGdcvse+Voc6v1RgnsR3nWcYU7q4t6z0Q6T86sv5Zq8VIRbOWWFpvdGE83LtdSMNd+6Y0xw==", + "dev": true, + "license": "MIT", + "dependencies": { + "lodash": "^4.17.20", + "renderkid": "^3.0.0" + } + }, "node_modules/pretty-format": { "version": "29.6.1", "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.6.1.tgz", @@ -15554,6 +16050,123 @@ "jsesc": "bin/jsesc" } }, + "node_modules/relateurl": { + "version": "0.2.7", + "resolved": "https://registry.npmjs.org/relateurl/-/relateurl-0.2.7.tgz", + "integrity": "sha512-G08Dxvm4iDN3MLM0EsP62EDV9IuhXPR6blNz6Utcp7zyV3tr4HVNINt6MpaRWbxoOHT3Q7YN2P+jaHX8vUbgog==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/renderkid": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/renderkid/-/renderkid-3.0.0.tgz", + "integrity": "sha512-q/7VIQA8lmM1hF+jn+sFSPWGlMkSAeNYcPLmDQx2zzuiDfaLrOmumR8iaUKlenFgh0XRPIUeSPlH3A+AW3Z5pg==", + "dev": true, + "license": "MIT", + "dependencies": { + "css-select": "^4.1.3", + "dom-converter": "^0.2.0", + "htmlparser2": "^6.1.0", + "lodash": "^4.17.21", + "strip-ansi": "^6.0.1" + } + }, + "node_modules/renderkid/node_modules/css-select": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-4.3.0.tgz", + "integrity": "sha512-wPpOYtnsVontu2mODhA19JrqWxNsfdatRKd64kmpRbQgh1KtItko5sTnEpPdpSaJszTOhEMlF/RPz28qj4HqhQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^6.0.1", + "domhandler": "^4.3.1", + "domutils": "^2.8.0", + "nth-check": "^2.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/renderkid/node_modules/dom-serializer": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.4.1.tgz", + "integrity": "sha512-VHwB3KfrcOOkelEG2ZOfxqLZdfkil8PtJi4P8N2MMXucZq2yLp75ClViUlOVwyoHEDjYU433Aq+5zWP61+RGag==", + "dev": true, + "license": "MIT", + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + }, + "funding": { + "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" + } + }, + "node_modules/renderkid/node_modules/domhandler": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.3.1.tgz", + "integrity": "sha512-GrwoxYN+uWlzO8uhUXRl0P+kHE4GtVPfYzVLcUxPL7KNdHKj66vvlhiweIHqYYXWlw+T8iLMp42Lm67ghw4WMQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "domelementtype": "^2.2.0" + }, + "engines": { + "node": ">= 4" + }, + "funding": { + "url": "https://github.com/fb55/domhandler?sponsor=1" + } + }, + "node_modules/renderkid/node_modules/domutils": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.8.0.tgz", + "integrity": "sha512-w96Cjofp72M5IIhpjgobBimYEfoPjx1Vx0BSX9P30WBdZW2WIKU0T1Bd0kz2eNZ9ikjKgHbEyKx8BB6H1L3h3A==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" + }, + "funding": { + "url": "https://github.com/fb55/domutils?sponsor=1" + } + }, + "node_modules/renderkid/node_modules/entities": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", + "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", + "dev": true, + "license": "BSD-2-Clause", + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/renderkid/node_modules/htmlparser2": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-6.1.0.tgz", + "integrity": "sha512-gyyPk6rgonLFEDGoeRgQNaEUvdJ4ktTmmUh/h2t7s+M8oPpIPxgNACWa+6ESR57kXstwqPiCut0V8NRpcwgU7A==", + "dev": true, + "funding": [ + "https://github.com/fb55/htmlparser2?sponsor=1", + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ], + "license": "MIT", + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.0.0", + "domutils": "^2.5.2", + "entities": "^2.0.0" + } + }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -16874,6 +17487,13 @@ "node": ">=6.14.2" } }, + "node_modules/utila": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/utila/-/utila-0.4.0.tgz", + "integrity": "sha512-Z0DbgELS9/L/75wZbro8xAnT50pBVFQZ+hUEueGDU5FN51YSCYM+jdxsfCiHjwNP/4LCDD0i/graKpeBnOXKRA==", + "dev": true, + "license": "MIT" + }, "node_modules/v8-compile-cache-lib": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", diff --git a/frontend/package.json b/frontend/package.json index 64417b35..5113f93f 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -40,6 +40,7 @@ "eslint-plugin-promise": "^6.1.1", "eslint-plugin-react": "^7.34.0", "eslint-plugin-react-hooks": "^4.6.0", + "html-webpack-plugin": "^5.6.0", "jest": "^29.6.1", "prettier": "^3.3.3", "ts-node": "^10.9.2", diff --git a/frontend/src/App.tsx b/frontend/src/App.tsx index 9644bf7c..7e6d40cd 100644 --- a/frontend/src/App.tsx +++ b/frontend/src/App.tsx @@ -15,6 +15,7 @@ import { GarageContextProvider } from './contexts/GarageContext'; import { FederationContextProvider } from './contexts/FederationContext'; const App = (): JSX.Element => { + const [client, _view] = window.RobosatsSettings.split('-'); return ( @@ -24,11 +25,7 @@ const App = (): JSX.Element => { - {window.NativeRobosats === undefined && window.RobosatsClient === undefined ? ( - - ) : ( - - )} + {client !== 'mobile' ? : }
diff --git a/frontend/src/basic/Main.tsx b/frontend/src/basic/Main.tsx index d78533d3..04b4db28 100644 --- a/frontend/src/basic/Main.tsx +++ b/frontend/src/basic/Main.tsx @@ -12,9 +12,10 @@ import { GarageContext, type UseGarageStoreType } from '../contexts/GarageContex import Routes from './Routes'; const getRouter = (): any => { - if (window.NativeRobosats === undefined && window.RobosatsClient === undefined) { + const [client, _view] = window.RobosatsSettings.split('-'); + if (client === 'web') { return BrowserRouter; - } else if (window.RobosatsClient === 'desktop-app') { + } else if (client === 'desktop') { return HashRouter; } else { return MemoryRouter; diff --git a/frontend/src/basic/RobotPage/RobotProfile.tsx b/frontend/src/basic/RobotPage/RobotProfile.tsx index 97b0db30..cbb2e385 100644 --- a/frontend/src/basic/RobotPage/RobotProfile.tsx +++ b/frontend/src/basic/RobotPage/RobotProfile.tsx @@ -43,7 +43,7 @@ const RobotProfile = ({ setView, width, }: RobotProfileProps): JSX.Element => { - const { windowSize } = useContext(AppContext); + const { windowSize, client } = useContext(AppContext); const { garage, slotUpdatedAt } = useContext(GarageContext); const { t } = useTranslation(); @@ -317,7 +317,7 @@ const RobotProfile = ({ - {window.NativeRobosats === undefined ? ( + {client !== 'mobile' ? (
- - + \ No newline at end of file diff --git a/frontend/templates/frontend/pro.html b/frontend/templates/frontend/pro.html index 4f97bb5a..da805f9c 100644 --- a/frontend/templates/frontend/pro.html +++ b/frontend/templates/frontend/pro.html @@ -1,7 +1,6 @@ - @@ -12,13 +11,13 @@ RoboSats PRO - Simple and Private Bitcoin Exchange - {% load static %} - - - - - - + + + + + + +