web_lib

Common web application libraries
git clone https://radroots.dev/git/web_lib.git
Log | Files | Refs | LICENSE

commit f2604e8fa8caf027ed2d6b6733cc142ff29c0403
parent 252f2eb7169389cc3db9860b7819f4f30a388ef9
Author: triesap <triesap@radroots.dev>
Date:   Mon, 22 Dec 2025 20:43:58 +0000

apps-lib-market: merge import

Diffstat:
Aapps-lib-market/.env.example | 1+
Aapps-lib-market/.gitignore | 43+++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/LICENSE | 675+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/justfile | 7+++++++
Aapps-lib-market/package.json | 61+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/components/features/profile-listing.svelte | 123+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/components/layouts/layout-column-entry.svelte | 14++++++++++++++
Aapps-lib-market/src/lib/components/layouts/layout-column-heading-display-simple.svelte | 18++++++++++++++++++
Aapps-lib-market/src/lib/components/layouts/layout-column-heading-view-buttons.svelte | 38++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/components/layouts/layout-column-heading.svelte | 34++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/components/layouts/layout-column.svelte | 11+++++++++++
Aapps-lib-market/src/lib/index.ts | 23+++++++++++++++++++++++
Aapps-lib-market/src/lib/types/profile/load.ts | 20++++++++++++++++++++
Aapps-lib-market/src/lib/types/profile/view.ts | 24++++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/_env.ts | 7+++++++
Aapps-lib-market/src/lib/utils/app/storage.ts | 22++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/app/theme.ts | 41+++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/lib.ts | 2++
Aapps-lib-market/src/lib/utils/nostr/events/listing/manager.svelte.ts | 51+++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/nostr/events/profile/manager.svelte.ts | 51+++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/nostr/stores/indexed_store.svelte.ts | 56++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/nostr/trade/listing/manager.svelte.ts | 815+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/utils/nostr/trade/listing/types.ts | 182+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/views/profile/profile-indexed.svelte | 430+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/views/profile/profile-network-nip05.svelte | 22++++++++++++++++++++++
Aapps-lib-market/src/lib/views/profile/profile-network-npub.svelte | 30++++++++++++++++++++++++++++++
Aapps-lib-market/src/lib/views/profile/profile-network-public-key.svelte | 18++++++++++++++++++
Aapps-lib-market/src/lib/views/profile/profile-network.svelte | 22++++++++++++++++++++++
Aapps-lib-market/src/lib/views/profile/profile.svelte | 19+++++++++++++++++++
Aapps-lib-market/svelte.config.js | 15+++++++++++++++
Aapps-lib-market/tsconfig.json | 13+++++++++++++
Aapps-lib-market/vite.config.ts | 11+++++++++++
32 files changed, 2899 insertions(+), 0 deletions(-)

diff --git a/apps-lib-market/.env.example b/apps-lib-market/.env.example @@ -0,0 +1 @@ +VITE_PUBLIC_RADROOTS_MARKET_RELAY_INDEXES_URL= diff --git a/apps-lib-market/.gitignore b/apps-lib-market/.gitignore @@ -0,0 +1,43 @@ +node_modules + +# Output +.output +.vercel +.netlify +.wrangler +/.svelte-kit +/build +dist +.turbo + +# OS +.DS_Store +Thumbs.db + +# Env +.env +.env.* +!.env.example +!.env.test + +# Vite +vite.config.js.timestamp-* +vite.config.ts.timestamp-* +vite.config.dev* + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# secrets +*.pem + +# local +.local +.vscode +notes*.txt +notes*.md +notes*.json +git-diff*.txt +justfile diff --git a/apps-lib-market/LICENSE b/apps-lib-market/LICENSE @@ -0,0 +1,674 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + <one line to give the program's name and a brief idea of what it does.> + Copyright (C) <year> <name of author> + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see <https://www.gnu.org/licenses/>. + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + <program> Copyright (C) <year> <name of author> + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +<https://www.gnu.org/licenses/>. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +<https://www.gnu.org/licenses/why-not-lgpl.html>. +\ No newline at end of file diff --git a/apps-lib-market/justfile b/apps-lib-market/justfile @@ -0,0 +1,6 @@ +gen: + gen-app-env.js && gen-package-exports.js --is_module --is_relative --dir src/lib --out src/lib +build: + just gen && yarn build +genp: + ts-generate-code-prompt.py --dir src +\ No newline at end of file diff --git a/apps-lib-market/package.json b/apps-lib-market/package.json @@ -0,0 +1,60 @@ +{ + "name": "@radroots/apps-lib-market", + "version": "0.0.0", + "private": true, + "license": "GPLv3", + "scripts": { + "dev": "svelte-package -w", + "prebuild": "npm run clean && npm run check", + "build": "svelte-package", + "preview": "vite preview", + "clean": "rimraf dist", + "prepare": "svelte-kit sync || echo ''", + "prepack": "svelte-kit sync && svelte-package && publint", + "check": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json", + "check:watch": "svelte-kit sync && svelte-check --tsconfig ./tsconfig.json --watch" + }, + "files": [ + "dist", + "!dist/**/*.test.*", + "!dist/**/*.spec.*" + ], + "sideEffects": [ + "**/*.css" + ], + "svelte": "./dist/index.js", + "types": "./dist/index.d.ts", + "type": "module", + "exports": { + ".": { + "types": "./dist/index.d.ts", + "svelte": "./dist/index.js" + } + }, + "peerDependencies": { + "@sveltejs/kit": "^2.22.0", + "svelte": "^5.0.0" + }, + "devDependencies": { + "@radroots/dev": "*", + "@sveltejs/adapter-auto": "^4.0.0", + "@sveltejs/kit": "^2.22.0", + "@sveltejs/package": "^2.0.0", + "@sveltejs/vite-plugin-svelte": "^6.0.0", + "@types/node": "^22.5.0", + "publint": "^0.3.2", + "svelte": "^5.0.0", + "svelte-check": "^4.0.0", + "typescript": "5.8.3", + "vite": "7.0.6" + }, + "dependencies": { + "@radroots/apps-lib": "*", + "@radroots/utils-nostr": "*", + "@radroots/utils": "*", + "@radroots/core-bindings": "*", + "@radroots/events-bindings": "*", + "@radroots/events-indexed-bindings": "*", + "@radroots/trade-bindings": "*" + } +} +\ No newline at end of file diff --git a/apps-lib-market/src/lib/components/features/profile-listing.svelte b/apps-lib-market/src/lib/components/features/profile-listing.svelte @@ -0,0 +1,123 @@ +<script lang="ts"> + import type { + IndexedEventsStorePayload, + OrderBundle, + TradeFlowService, + TradeListingBundle, + } from "$root"; + + import type { RadrootsListing } from "@radroots/events-bindings"; + import { + TradeListingStage, + type TradeListingOrderRequestPayload, + } from "@radroots/trade-bindings"; + + let { + basis, + }: { + basis: { + trade: TradeFlowService | null; + listing_event: IndexedEventsStorePayload<RadrootsListing>; + }; + } = $props(); + + function build_order_payload(): TradeListingOrderRequestPayload { + const listing = basis.listing_event?.data; + if (!listing) throw new Error(`!listing`); + + const quantity = listing.quantities[0]; + if (!quantity) throw new Error(`!q`); + + const price = listing.prices[0]; + if (!price) throw new Error(`!price`); + + const payload: TradeListingOrderRequestPayload = { + price, + quantity, + }; + return payload; + } + + // ---- Reactive bundle + latest order selection ---- + const bundle = $derived<TradeListingBundle | undefined>( + basis.trade?.get_trade_listing_bundle(basis.listing_event.id) || + undefined, + ); + + function pick_latest_order( + b?: TradeListingBundle, + ): OrderBundle | undefined { + if (!b) return undefined; + let best: OrderBundle | undefined; + for (const [, ob] of b.pending_orders) { + if (!best || (ob.last_update_at ?? 0) > (best.last_update_at ?? 0)) + best = ob; + } + for (const [, ob] of b.orders) { + if (!best || (ob.last_update_at ?? 0) > (best.last_update_at ?? 0)) + best = ob; + } + return best; + } + + const latest_order = $derived<OrderBundle | undefined>( + pick_latest_order(bundle), + ); + const is_loading = $derived<boolean>(!!latest_order?.loading); + + // Access results/feedback by enum key, not string + const last_order_result = $derived( + latest_order?.results?.[TradeListingStage.Order]?.at(-1), + ); + const last_feedback = $derived( + latest_order?.feedback?.[TradeListingStage.Order]?.at(-1), + ); + + // ---- Actions ---- + async function handle_order_click() { + if (!basis.trade) return; + try { + const payload = build_order_payload(); + const res = await basis.trade.order_request( + basis.listing_event.id, + payload, + ); + + if (!res.ok) { + console.warn("order_request failed", res.error, res.request); + return; + } + + const { request, result, order_id } = res; + console.log("order created:", { + request_id: request.id, + order_id, + result_id: result.id, + }); + } catch (err) { + console.error("order_request threw", err); + } + } +</script> + +<div class="flex flex-col gap-2"> + <button class="px-3 py-1 rounded bg-ly1" onclick={handle_order_click}> + {is_loading ? "ordering..." : "order"} + </button> + + {#if last_order_result} + <pre class="text-xs break-all">{JSON.stringify( + last_order_result, + null, + 2, + )}</pre> + {/if} + + {#if last_feedback} + <pre class="text-xs break-all opacity-80">{JSON.stringify( + last_feedback, + null, + 2, + )}</pre> + {/if} +</div> diff --git a/apps-lib-market/src/lib/components/layouts/layout-column-entry.svelte b/apps-lib-market/src/lib/components/layouts/layout-column-entry.svelte @@ -0,0 +1,14 @@ +<script lang="ts"> + import type { Snippet } from "svelte"; + + let { + children, + basis, + }: { children: Snippet; basis?: { classes?: string } } = $props(); +</script> + +<div + class={`flex flex-col w-full desktop:w-[700px] justify-start items-start ${basis?.classes}`} +> + {@render children()} +</div> diff --git a/apps-lib-market/src/lib/components/layouts/layout-column-heading-display-simple.svelte b/apps-lib-market/src/lib/components/layouts/layout-column-heading-display-simple.svelte @@ -0,0 +1,18 @@ +<script lang="ts"> + import type { Snippet } from "svelte"; + + let { row1, row2, row3 }: { row1: Snippet; row2: Snippet; row3: Snippet } = + $props(); +</script> + +<div class={`flex flex-row w-full gap-2 justify-start items-center`}> + {@render row1()} +</div> +<div class={`flex flex-col w-full gap-1 justify-start items-center`}> + <div class={`flex flex-row w-full justify-start items-center`}> + {@render row2()} + </div> + <div class={`flex flex-row w-full gap-2 justify-start items-center`}> + {@render row3()} + </div> +</div> diff --git a/apps-lib-market/src/lib/components/layouts/layout-column-heading-view-buttons.svelte b/apps-lib-market/src/lib/components/layouts/layout-column-heading-view-buttons.svelte @@ -0,0 +1,38 @@ +<script lang="ts"> + import { Flex, Glyph } from "@radroots/apps-lib"; + import type { CallbackPromise } from "@radroots/utils"; + + let view_index = $state(0); +</script> + +<div + class={`grid grid-cols-12 flex flex-row h-[44px] w-full justify-around items-center border-b-line border-b-cloak_grey/20`} +> + {#snippet button(index: number, icon: string, callback: CallbackPromise)} + <button + class={`relative col-span-4 flex flex-col h-full justify-center items-center`} + onclick={async () => { + view_index = index; + await callback(); + }} + > + <Glyph + basis={{ + classes: `text-black_panther`, + size: `sm`, + key: icon, + }} + /> + {#if view_index === index} + <div + class={`absolute bottom-0 left-0 flex flex-row h-[2px] w-full justify-start items-center bg-cloak_grey/40 el-re`} + > + <Flex /> + </div> + {/if} + </button> + {/snippet} + {@render button(0, `grid-nine`, async () => {})} + {@render button(1, `monitor-play`, async () => {})} + {@render button(2, `user-rectangle`, async () => {})} +</div> diff --git a/apps-lib-market/src/lib/components/layouts/layout-column-heading.svelte b/apps-lib-market/src/lib/components/layouts/layout-column-heading.svelte @@ -0,0 +1,34 @@ +<script lang="ts"> + import { Flex } from "@radroots/apps-lib"; + import type { Snippet } from "svelte"; + + let { + heading, + subheading, + footer, + }: { + heading: Snippet; + subheading: Snippet; + footer?: Snippet; + } = $props(); +</script> + +<div class={`flex flex-col w-full pb-2 justify-start items-start`}> + <div + class={`flex flex-row h-[120px] w-full px-4 gap-4 justify-center items-center`} + > + <div + class={`flex flex-row flex-shrink-0 h-[96px] w-[96px] justify-center items-center bg-lime-500/20 rounded-sm`} + > + <Flex /> + </div> + <div + class={`flex flex-col h-full w-full gap-2 justify-center items-start`} + > + {@render heading()} + </div> + </div> + <div class={`flex flex-col w-full px-4 py-1 justify-center items-start`}> + {@render subheading()} + </div> +</div> diff --git a/apps-lib-market/src/lib/components/layouts/layout-column.svelte b/apps-lib-market/src/lib/components/layouts/layout-column.svelte @@ -0,0 +1,11 @@ +<script lang="ts"> + import type { Snippet } from "svelte"; + + let { children }: { children: Snippet } = $props(); +</script> + +<div + class={`flex flex-col w-full desktop:p-8 gap-4 desktop:gap-8 justify-center items-center`} +> + {@render children()} +</div> diff --git a/apps-lib-market/src/lib/index.ts b/apps-lib-market/src/lib/index.ts @@ -0,0 +1,23 @@ +export { default as ProfileListing } from "./components/features/profile-listing.svelte" +export { default as LayoutColumnEntry } from "./components/layouts/layout-column-entry.svelte" +export { default as LayoutColumnHeadingDisplaySimple } from "./components/layouts/layout-column-heading-display-simple.svelte" +export { default as LayoutColumnHeadingViewButtons } from "./components/layouts/layout-column-heading-view-buttons.svelte" +export { default as LayoutColumnHeading } from "./components/layouts/layout-column-heading.svelte" +export { default as LayoutColumn } from "./components/layouts/layout-column.svelte" +export * from "./types/profile/load.js" +export * from "./types/profile/view.js" +export * from "./utils/app/storage.js" +export * from "./utils/app/theme.js" +export * from "./utils/lib.js" +export * from "./utils/nostr/events/listing/manager.svelte.js" +export * from "./utils/nostr/events/profile/manager.svelte.js" +export * from "./utils/nostr/stores/indexed_store.svelte.js" +export * from "./utils/nostr/trade/listing/manager.svelte.js" +export * from "./utils/nostr/trade/listing/types.js" +export { default as ProfileIndexed } from "./views/profile/profile-indexed.svelte" +export { default as ProfileNetworkNip05 } from "./views/profile/profile-network-nip05.svelte" +export { default as ProfileNetworkNpub } from "./views/profile/profile-network-npub.svelte" +export { default as ProfileNetworkPublicKey } from "./views/profile/profile-network-public-key.svelte" +export { default as ProfileNetwork } from "./views/profile/profile-network.svelte" +export { default as Profile } from "./views/profile/profile.svelte" + diff --git a/apps-lib-market/src/lib/types/profile/load.ts b/apps-lib-market/src/lib/types/profile/load.ts @@ -0,0 +1,20 @@ +import type { RadrootsCommentEventMetadata, RadrootsListingEventMetadata, RadrootsProfileEventMetadata } from "@radroots/events-bindings"; + +export type PageLoadProfileData = { + public_key: string; + npub?: string; + events: PageLoadProfileDataEvents; +}; + +export type PageLoadProfileDataEvents = + ( + { + profile: RadrootsProfileEventMetadata; + } | { + profile: RadrootsProfileEventMetadata; + listings: RadrootsListingEventMetadata[]; + listing_comments: Record<string, RadrootsCommentEventMetadata[]>; + followers: number; + following: number; + } + ); diff --git a/apps-lib-market/src/lib/types/profile/view.ts b/apps-lib-market/src/lib/types/profile/view.ts @@ -0,0 +1,23 @@ +import type { PageLoadProfileData } from "$root"; + +export type IProfileView = IProfileViewIndexed | IProfileViewNetwork; + +export type IProfileViewIndexed = { + indexed: PageLoadProfileData; +}; + +export type IProfileViewNetwork = { + unknown?: IProfileViewNetworkPublicKey | IProfileViewNetworkNpub | IProfileViewNetworkNip05; +}; + +export type IProfileViewNetworkPublicKey = { + public_key: string; +}; + +export type IProfileViewNetworkNpub = { + npub: string; +}; + +export type IProfileViewNetworkNip05 = { + nip05: string; +}; +\ No newline at end of file diff --git a/apps-lib-market/src/lib/utils/_env.ts b/apps-lib-market/src/lib/utils/_env.ts @@ -0,0 +1,6 @@ +const RADROOTS_MARKET_RELAY_INDEXES_URL = import.meta.env.VITE_PUBLIC_RADROOTS_MARKET_RELAY_INDEXES_URL; +if (!RADROOTS_MARKET_RELAY_INDEXES_URL || typeof RADROOTS_MARKET_RELAY_INDEXES_URL !== 'string') throw new Error('Missing env var: VITE_PUBLIC_RADROOTS_MARKET_RELAY_INDEXES_URL'); + +export const _env = { + RADROOTS_MARKET_RELAY_INDEXES_URL, +} as const; +\ No newline at end of file diff --git a/apps-lib-market/src/lib/utils/app/storage.ts b/apps-lib-market/src/lib/utils/app/storage.ts @@ -0,0 +1,21 @@ +import { IdbLib, type ThemeMode } from "@radroots/apps-lib"; + +export type GlobalConfig = { + theme_mode: ThemeMode; + theme_key: string; + locale: string; + global_relays: string[]; + npub: string; +}; + +export type GlobalConfigKeys = keyof GlobalConfig; + + +export type PageSession = { + draftId: string; + lastScroll: number; +}; + +export type PageSessionKeys = keyof PageSession; + +export const idb = new IdbLib<GlobalConfigKeys, GlobalConfig, PageSessionKeys, PageSession>(); +\ No newline at end of file diff --git a/apps-lib-market/src/lib/utils/app/theme.ts b/apps-lib-market/src/lib/utils/app/theme.ts @@ -0,0 +1,41 @@ +import { browser } from "$app/environment"; +import { idb } from "$root"; +import { get_store, get_system_theme, theme_key, theme_mode, theme_reset, theme_set, theme_toggle } from "@radroots/apps-lib"; + +export const toggle_theme = async (): Promise<void> => { + await theme_toggle(async (mode) => { + await idb.save_global("theme_mode", mode); + }); +}; + +export const init_theme = async (): Promise<void> => { + let mode = await idb.read_global("theme_mode"); + let key = await idb.read_global("theme_key"); + + const mode_sys = get_system_theme(); + + if (!mode || mode !== mode_sys) { + mode = mode_sys; + await idb.save_global("theme_mode", mode); + } + + if (!key) { + key = `os`; + await idb.save_global("theme_key", key); + } + + const $mode = get_store(theme_mode); + const $key = get_store(theme_key); + + if (mode === $mode && key === $key) return; + + theme_mode.set(mode); + theme_key.set(key); + theme_reset.set(true); +}; + +theme_reset.subscribe((reset) => { + if (!reset || !browser) return; + theme_set(get_store(theme_key), get_store(theme_mode)); + theme_reset.set(false); +}) diff --git a/apps-lib-market/src/lib/utils/lib.ts b/apps-lib-market/src/lib/utils/lib.ts @@ -0,0 +1 @@ +export const head_title_suffix = `โ€ข Radroots Market` +\ No newline at end of file diff --git a/apps-lib-market/src/lib/utils/nostr/events/listing/manager.svelte.ts b/apps-lib-market/src/lib/utils/nostr/events/listing/manager.svelte.ts @@ -0,0 +1,51 @@ +import { create_indexed_events_store, type IndexedEventsStorePayload } from '$root'; +import type { RadrootsListing } from "@radroots/events-bindings"; +import { KIND_RADROOTS_LISTING, type RadrootsListingNostrEvent } from '@radroots/utils-nostr'; + +export function create_radroots_listing_manager(initial_indexed: RadrootsListingNostrEvent[] = []) { + const store = create_indexed_events_store<RadrootsListing>({ + key_of: (p) => p.data?.d_tag, + }); + + const to_indexed_payload = (r: RadrootsListingNostrEvent): IndexedEventsStorePayload<RadrootsListing> => ({ + id: r.id, + kind: KIND_RADROOTS_LISTING, + author: r.author, + published_at: r.published_at ?? 0, + data: r.listing, + source: 'indexed', + }); + + store.init( + (initial_indexed ?? []) + .filter((r) => r?.listing?.d_tag) + .map(to_indexed_payload), + ); + + const on_parsed_event = (parsed: RadrootsListingNostrEvent) => { + store.add({ + id: parsed.id, + kind: parsed.kind ?? KIND_RADROOTS_LISTING, + author: parsed.author, + published_at: parsed.published_at, + data: parsed.listing, + source: 'nostr', + }); + } + + const init_from_indexed = (rows: RadrootsListingNostrEvent[]) => { + store.init( + (rows ?? []) + .filter((r) => r?.listing?.d_tag) + .map(to_indexed_payload), + ); + } + + return { + get list() { return store.list; }, + get map() { return store.map; }, + init_from_indexed, + on_parsed_event, + to_indexed_payload, + }; +} diff --git a/apps-lib-market/src/lib/utils/nostr/events/profile/manager.svelte.ts b/apps-lib-market/src/lib/utils/nostr/events/profile/manager.svelte.ts @@ -0,0 +1,51 @@ +import { create_indexed_events_store, type IndexedEventsStorePayload } from '$root'; +import type { RadrootsProfile } from "@radroots/events-bindings"; +import { KIND_RADROOTS_PROFILE, type RadrootsProfileNostrEvent } from '@radroots/utils-nostr'; + +export function create_radroots_profile_manager(initial_indexed?: RadrootsProfileNostrEvent) { + const store = create_indexed_events_store<RadrootsProfile>({ + key_of: (p) => p.author, + }); + + const to_indexed_payload = (r: RadrootsProfileNostrEvent): IndexedEventsStorePayload<RadrootsProfile> => ({ + id: r.id, + kind: KIND_RADROOTS_PROFILE, + author: r.author, + published_at: r.published_at ?? 0, + data: r.profile, + source: 'indexed', + }); + + if (initial_indexed?.author && initial_indexed.profile) { + store.init([to_indexed_payload(initial_indexed)]); + } else { + store.init([]); + } + + const on_parsed_event = (parsed: RadrootsProfileNostrEvent) => { + store.add({ + id: parsed.id, + kind: parsed.kind ?? KIND_RADROOTS_PROFILE, + author: parsed.author, + published_at: parsed.published_at, + data: parsed.profile, + source: 'nostr', + }); + } + + const init_from_indexed = (row?: RadrootsProfileNostrEvent) => { + if (row?.author && row.profile) { + store.init([to_indexed_payload(row)]); + } else { + store.init([]); + } + } + + return { + get list() { return store.list; }, + get map() { return store.map; }, + init_from_indexed, + on_parsed_event, + to_indexed_payload, + }; +} diff --git a/apps-lib-market/src/lib/utils/nostr/stores/indexed_store.svelte.ts b/apps-lib-market/src/lib/utils/nostr/stores/indexed_store.svelte.ts @@ -0,0 +1,56 @@ +import type { NdkEventBasis } from '@radroots/utils-nostr'; +import { SvelteMap } from 'svelte/reactivity'; + +export type IndexedEventsStoreSource = 'indexed' | 'nostr'; + +export type IndexedEventsStorePayload<T> = NdkEventBasis<number> & { + data: T; + source: IndexedEventsStoreSource; +} + +export type CreateIndexedEventsStoreOptions<T> = { + key_of: (p: IndexedEventsStorePayload<T>) => string | undefined; + is_newer?: (a: IndexedEventsStorePayload<T>, b: IndexedEventsStorePayload<T>) => boolean; +} + +export const default_is_newer = <T>(a: IndexedEventsStorePayload<T>, b: IndexedEventsStorePayload<T>) => { + const at = a.published_at ?? 0; + const bt = b.published_at ?? 0; + if (at !== bt) return at > bt; + if (a.source !== b.source) return a.source === 'nostr'; + return a.id > b.id; +}; + +export function create_indexed_events_store<T>(opts: CreateIndexedEventsStoreOptions<T>) { + let map = $state(new SvelteMap<string, IndexedEventsStorePayload<T>>()); + + const add = (p: IndexedEventsStorePayload<T>) => { + const key = opts.key_of(p); + if (!key) return; + const existing = map.get(key); + const newer = existing ? (opts.is_newer ?? default_is_newer)(p, existing) : true; + if (newer) map.set(key, p); + }; + + const init = (items: IndexedEventsStorePayload<T>[]) => { + const m = new SvelteMap<string, IndexedEventsStorePayload<T>>(); + for (const it of items) { + const key = opts.key_of(it); + if (!key) continue; + const ex = m.get(key); + if (!ex || (opts.is_newer ?? default_is_newer)(it, ex)) m.set(key, it); + } + map = m; + }; + + const list_raw = $derived(Array.from(map.values())); + + const list = $derived([...list_raw].sort((a, b) => (b.published_at ?? 0) - (a.published_at ?? 0))); + + return { + get map() { return map; }, + get list() { return list; }, + add, + init, + }; +} diff --git a/apps-lib-market/src/lib/utils/nostr/trade/listing/manager.svelte.ts b/apps-lib-market/src/lib/utils/nostr/trade/listing/manager.svelte.ts @@ -0,0 +1,815 @@ +import NDK, { NDKEvent, NDKSubscription, NDKUser } from "@nostr-dev-kit/ndk"; +import { KIND_JOB_FEEDBACK } from "@radroots/events-bindings"; +import { MARKER_LISTING, TradeListingStage, type TradeListingAcceptRequest, type TradeListingConveyanceRequest, type TradeListingFulfillmentRequest, type TradeListingInvoiceRequest, type TradeListingOrderRequestPayload, type TradeListingPaymentProofRequest, type TradeListingReceiptRequest } from "@radroots/trade-bindings"; +import { time_now_ms } from "@radroots/utils"; +import { + KIND_RADROOTS_LISTING, + REQUEST_KINDS, + RESULT_KINDS, + TAG_E, + get_event_tag, + get_job_input_data_for_marker, + get_trade_listing_stage_from_event_kind, + ndk_event_trade_listing_accept_request, + ndk_event_trade_listing_conveyance_request, + ndk_event_trade_listing_fulfillment_request, + ndk_event_trade_listing_invoice_request, + ndk_event_trade_listing_order_request, + ndk_event_trade_listing_payment_request, + ndk_event_trade_listing_receipt_request, +} from "@radroots/utils-nostr"; +import { SvelteMap, SvelteSet } from "svelte/reactivity"; +import type { + AcceptOptions, + ConveyanceOptions, + CreateTradeFlowServiceOptions, + FulfillmentOptions, + InvoiceOptions, + OrderBundle, + OrderRequestErr, + OrderRequestOk, + OrderRequestResult, + PaymentOptions, + ReceiptOptions, + StageActionErr, + StageActionOk, + StageActionResult, + StagePostInput, + StagePostOutput, + TradeFlowService, + TradeListingBundle, +} from "./types"; + +const MAX_ITEMS_PER_BUCKET = 50; + +type Waiter = { + since_ms: number; + resolve: (e: NDKEvent) => void; + reject: (err: Error) => void; + timer?: ReturnType<typeof setTimeout>; +}; + +function push_capped<T>(arr: T[], item: T, cap: number): void { + arr.push(item); + if (arr.length > cap) arr.splice(0, arr.length - cap); +} + +export class TradeFlowServiceImpl implements TradeFlowService { + private ndk: NDK; + private ndk_user_store: () => NDKUser; + + private sub: NDKSubscription | null = null; + + private events_to_thread = new Map<string, { listing_id: string; order_id?: string }>(); + private orphans_by_ref = new Map<string, NDKEvent[]>(); + private loading_ids = new SvelteSet<string>(); + private waiters_by_request = new Map<string, Set<Waiter>>(); + + private latest_update_event: NDKEvent | undefined = undefined; + private load_complete = false; + + private authors: string[] | undefined; + private kinds: number[]; + private default_timeout_ms: number; + + public listings = new SvelteMap<string, TradeListingBundle>(); + + private restarting = false; + + constructor(opts: CreateTradeFlowServiceOptions) { + this.ndk = opts.ndk; + this.ndk_user_store = opts.ndk_user_store; + + this.authors = opts.authors; + this.kinds = opts.kinds + ? opts.kinds + : [ + KIND_RADROOTS_LISTING, + ...Object.values(REQUEST_KINDS), + ...Object.values(RESULT_KINDS), + KIND_JOB_FEEDBACK, + ]; + this.default_timeout_ms = + typeof opts.default_timeout_ms === "number" ? opts.default_timeout_ms : 7000; + + this.restart_subscription(); + } + + get_latest_update(): NDKEvent | undefined { + return this.latest_update_event; + } + + set_filter_authors(authors?: string[] | undefined): void { + this.authors = authors; + this.restart_subscription(); + } + + set_filter_kinds(kinds: number[]): void { + this.kinds = kinds; + this.restart_subscription(); + } + + get_trade_listing_bundle(listing_id: string): TradeListingBundle | undefined { + return this.listings.get(listing_id); + } + + get_order_bundle(listing_id: string, order_id: string): OrderBundle | undefined { + const listing_bundle = this.listings.get(listing_id); + return listing_bundle ? listing_bundle.orders.get(order_id) : undefined; + } + + is_loading(event_id: string): boolean { + return this.loading_ids.has(event_id); + } + + on_event(ev: NDKEvent): void { + queueMicrotask(() => { + if (!this.restarting) this.ingest_event(ev); + }); + } + + async order_request( + listing_id: string, + payload: TradeListingOrderRequestPayload, + timeout_ms?: number + ): Promise<OrderRequestResult> { + try { + const request = await this.publish_request(() => { + const data = { event: { id: listing_id }, payload }; + return ndk_event_trade_listing_order_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }); + }); + + this.events_to_thread.set(request.id, { listing_id }); + const listing_bundle = this.ensure_listing(listing_id); + const order_bundle = this.ensure_order(listing_bundle, "pending", request.id); + this.attach_event_to_order(order_bundle, request); + this.index_event(request, listing_id, undefined); + + try { + const result = await this.await_response_for(request.id, timeout_ms); + const order_id = result.id; + const bundle = this.get_order_bundle(listing_id, order_id); + const ok: OrderRequestOk = { ok: true, request, result, order_id, bundle }; + return ok; + } catch { + this.update_loading_by_request(request.id, false); + const err: OrderRequestErr = { ok: false, error: "error.timeout", request }; + return err; + } + } catch { + const err: OrderRequestErr = { ok: false, error: "error.failed_to_publish" }; + return err; + } + } + + async accept_request(opts: AcceptOptions): Promise<StageActionResult<TradeListingStage.Accept>> { + const { listing_id, order_id, timeout_ms } = opts; + const prereq_id = this.resolve_input_event_id(TradeListingStage.Accept, listing_id, order_id); + if (!prereq_id) { + const err: StageActionErr<TradeListingStage.Accept> = { + ok: false, + stage: TradeListingStage.Accept, + error: "error.missing_prerequisite", + }; + return err; + } + try { + const data: TradeListingAcceptRequest = { + order_result_event_id: order_id, + listing_event_id: listing_id, + }; + const request = await this.publish_request(() => + ndk_event_trade_listing_accept_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }) + ); + return this.await_stage_result(TradeListingStage.Accept, { + listing_id, + order_id, + request, + timeout_ms, + }); + } catch { + const err: StageActionErr<TradeListingStage.Accept> = { + ok: false, + stage: TradeListingStage.Accept, + error: "error.failed_to_publish", + }; + return err; + } + } + + async conveyance_request( + opts: ConveyanceOptions + ): Promise<StageActionResult<TradeListingStage.Conveyance>> { + const { listing_id, order_id, method, timeout_ms } = opts; + const prereq_id = this.resolve_input_event_id( + TradeListingStage.Conveyance, + listing_id, + order_id + ); + if (!prereq_id) { + const err: StageActionErr<TradeListingStage.Conveyance> = { + ok: false, + stage: TradeListingStage.Conveyance, + error: "error.missing_prerequisite", + }; + return err; + } + try { + const data: TradeListingConveyanceRequest = { + accept_result_event_id: prereq_id, + method, + }; + const request = await this.publish_request(() => + ndk_event_trade_listing_conveyance_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }) + ); + return this.await_stage_result(TradeListingStage.Conveyance, { + listing_id, + order_id, + request, + timeout_ms, + }); + } catch { + const err: StageActionErr<TradeListingStage.Conveyance> = { + ok: false, + stage: TradeListingStage.Conveyance, + error: "error.failed_to_publish", + }; + return err; + } + } + + async invoice_request( + opts: InvoiceOptions + ): Promise<StageActionResult<TradeListingStage.Invoice>> { + const { listing_id, order_id, timeout_ms } = opts; + const prereq_id = this.resolve_input_event_id(TradeListingStage.Invoice, listing_id, order_id); + if (!prereq_id) { + const err: StageActionErr<TradeListingStage.Invoice> = { + ok: false, + stage: TradeListingStage.Invoice, + error: "error.missing_prerequisite", + }; + return err; + } + try { + const data: TradeListingInvoiceRequest = { accept_result_event_id: prereq_id }; + const request = await this.publish_request(() => + ndk_event_trade_listing_invoice_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }) + ); + return this.await_stage_result(TradeListingStage.Invoice, { + listing_id, + order_id, + request, + timeout_ms, + }); + } catch { + const err: StageActionErr<TradeListingStage.Invoice> = { + ok: false, + stage: TradeListingStage.Invoice, + error: "error.failed_to_publish", + }; + return err; + } + } + + async payment_request( + opts: PaymentOptions + ): Promise<StageActionResult<TradeListingStage.Payment>> { + const { listing_id, order_id, proof, timeout_ms } = opts; + const prereq_id = this.resolve_input_event_id(TradeListingStage.Payment, listing_id, order_id); + if (!prereq_id) { + const err: StageActionErr<TradeListingStage.Payment> = { + ok: false, + stage: TradeListingStage.Payment, + error: "error.missing_prerequisite", + }; + return err; + } + try { + const data: TradeListingPaymentProofRequest = { + invoice_result_event_id: prereq_id, + proof, + }; + const request = await this.publish_request(() => + ndk_event_trade_listing_payment_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }) + ); + return this.await_stage_result(TradeListingStage.Payment, { + listing_id, + order_id, + request, + timeout_ms, + }); + } catch { + const err: StageActionErr<TradeListingStage.Payment> = { + ok: false, + stage: TradeListingStage.Payment, + error: "error.failed_to_publish", + }; + return err; + } + } + + async fulfillment_request( + opts: FulfillmentOptions + ): Promise<StageActionResult<TradeListingStage.Fulfillment>> { + const { listing_id, order_id, timeout_ms } = opts; + const prereq_id = this.resolve_input_event_id( + TradeListingStage.Fulfillment, + listing_id, + order_id + ); + if (!prereq_id) { + const err: StageActionErr<TradeListingStage.Fulfillment> = { + ok: false, + stage: TradeListingStage.Fulfillment, + error: "error.missing_prerequisite", + }; + return err; + } + try { + const data: TradeListingFulfillmentRequest = { payment_result_event_id: prereq_id }; + const request = await this.publish_request(() => + ndk_event_trade_listing_fulfillment_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }) + ); + return this.await_stage_result(TradeListingStage.Fulfillment, { + listing_id, + order_id, + request, + timeout_ms, + }); + } catch { + const err: StageActionErr<TradeListingStage.Fulfillment> = { + ok: false, + stage: TradeListingStage.Fulfillment, + error: "error.failed_to_publish", + }; + return err; + } + } + + async receipt_request( + opts: ReceiptOptions + ): Promise<StageActionResult<TradeListingStage.Receipt>> { + const { listing_id, order_id, note, timeout_ms } = opts; + const prereq_id = this.resolve_input_event_id(TradeListingStage.Receipt, listing_id, order_id); + if (!prereq_id) { + const err: StageActionErr<TradeListingStage.Receipt> = { + ok: false, + stage: TradeListingStage.Receipt, + error: "error.missing_prerequisite", + }; + return err; + } + try { + const data: TradeListingReceiptRequest = note + ? { fulfillment_result_event_id: prereq_id, note } + : { fulfillment_result_event_id: prereq_id }; + const request = await this.publish_request(() => + ndk_event_trade_listing_receipt_request({ + ndk: this.ndk, + ndk_user: this.ndk_user_store(), + data, + }) + ); + return this.await_stage_result(TradeListingStage.Receipt, { + listing_id, + order_id, + request, + timeout_ms, + }); + } catch { + const err: StageActionErr<TradeListingStage.Receipt> = { + ok: false, + stage: TradeListingStage.Receipt, + error: "error.failed_to_publish", + }; + return err; + } + } + + post(input: StagePostInput): Promise<StagePostOutput> { + switch (input.stage) { + case TradeListingStage.Accept: + return this.accept_request(input.opts); + case TradeListingStage.Conveyance: + return this.conveyance_request(input.opts); + case TradeListingStage.Invoice: + return this.invoice_request(input.opts); + case TradeListingStage.Payment: + return this.payment_request(input.opts); + case TradeListingStage.Fulfillment: + return this.fulfillment_request(input.opts); + case TradeListingStage.Receipt: + return this.receipt_request(input.opts); + case TradeListingStage.Cancel: + case TradeListingStage.Refund: + return Promise.resolve({ + ok: false, + stage: input.stage, + error: "error.not_implemented", + }); + } + } + + destroy(): void { + if (this.sub) { + this.sub.stop(); + this.sub = null; + } + this.listings.clear(); + this.events_to_thread.clear(); + this.orphans_by_ref.clear(); + this.loading_ids.clear(); + this.latest_update_event = undefined; + this.load_complete = false; + + for (const set of this.waiters_by_request.values()) { + for (const w of set) { + if (w.timer) clearTimeout(w.timer); + w.reject(new Error("service destroyed")); + } + } + this.waiters_by_request.clear(); + } + + private async await_stage_result<S extends TradeListingStage>( + stage: S, + params: { + listing_id: string; + order_id: string; + request: NDKEvent; + timeout_ms?: number; + } + ): Promise<StageActionResult<S>> { + const { listing_id, order_id, request, timeout_ms } = params; + try { + const result = await this.await_response_for(request.id, timeout_ms); + const bundle = this.get_order_bundle(listing_id, order_id); + const ok: StageActionOk<S> = { ok: true, stage, request, result, order_id, bundle }; + return ok; + } catch { + this.update_loading_by_request(request.id, false); + const err: StageActionErr<S> = { ok: false, stage, error: "error.timeout", request }; + return err; + } + } + + private async publish_request(make: () => Promise<NDKEvent | undefined>): Promise<NDKEvent> { + const ev = await make(); + if (!ev) throw new Error("failed"); + queueMicrotask(() => this.ingest_event(ev)); + return ev; + } + + private async await_response_for(request_id: string, timeout_ms?: number): Promise<NDKEvent> { + this.loading_ids.add(request_id); + const since_ms = time_now_ms(); + + return new Promise<NDKEvent>((resolve, reject) => { + const cleanup = (w: Waiter) => { + const set = this.waiters_by_request.get(request_id); + if (set) { + set.delete(w); + if (set.size === 0) this.waiters_by_request.delete(request_id); + } + this.loading_ids.delete(request_id); + if (w.timer) clearTimeout(w.timer); + }; + + const waiter: Waiter = { + since_ms, + resolve: (e) => { + cleanup(waiter); + resolve(e); + }, + reject: () => { + cleanup(waiter); + reject(new Error("timeout")); + }, + }; + + const existing = this.waiters_by_request.get(request_id); + if (existing) existing.add(waiter); + else this.waiters_by_request.set(request_id, new Set<Waiter>([waiter])); + + const ms = typeof timeout_ms === "number" ? timeout_ms : this.default_timeout_ms; + waiter.timer = setTimeout(() => { + this.update_loading_by_request(request_id, false); + waiter.reject(new Error("timeout")); + }, ms); + }); + } + + private restart_subscription(): void { + if (this.sub) { + this.sub.stop(); + this.sub = null; + } + + this.listings.clear(); + this.events_to_thread.clear(); + this.orphans_by_ref.clear(); + this.loading_ids.clear(); + this.latest_update_event = undefined; + this.load_complete = false; + + const filter: { kinds: number[]; authors?: string[] } = { + kinds: this.kinds, + ...(Array.isArray(this.authors) ? { authors: this.authors } : {}), + }; + + const sub = this.ndk.subscribe(filter, { closeOnEose: false }); + + sub.on("event", (ev: NDKEvent) => { + queueMicrotask(() => { + if (!this.restarting) this.ingest_event(ev); + }); + }); + + sub.on("eose", () => { + this.load_complete = true; + }); + + sub.start(); + this.sub = sub; + this.restarting = false; + } + + private ensure_listing(listing_id: string): TradeListingBundle { + let listing_bundle = this.listings.get(listing_id); + if (!listing_bundle) { + listing_bundle = { listing: undefined, orders: new SvelteMap(), pending_orders: new SvelteMap() }; + this.listings.set(listing_id, listing_bundle); + } + return listing_bundle; + } + + private ensure_order( + listing_bundle: TradeListingBundle, + bucket: "pending" | "orders", + key: string + ): OrderBundle { + const map = bucket === "orders" ? listing_bundle.orders : listing_bundle.pending_orders; + let order_bundle = map.get(key); + if (!order_bundle) { + order_bundle = { + order_id: bucket === "orders" ? key : undefined, + listing_id: listing_bundle.listing ? listing_bundle.listing.id : "", + requests: Object.create(null), + results: Object.create(null), + feedback: Object.create(null), + started_at: time_now_ms(), + last_update_at: time_now_ms(), + loading: false, + }; + map.set(key, order_bundle); + } + return order_bundle; + } + + private attach_event_to_order(order_bundle: OrderBundle, ev: NDKEvent): void { + const stage = get_trade_listing_stage_from_event_kind(ev.kind); + if (!stage) return; + + const is_request_kind = Object.values(REQUEST_KINDS).includes(ev.kind); + const is_result_kind = Object.values(RESULT_KINDS).includes(ev.kind); + + if (is_request_kind) { + const arr = order_bundle.requests[stage] || (order_bundle.requests[stage] = []); + push_capped(arr, ev, MAX_ITEMS_PER_BUCKET); + if (ev.kind === REQUEST_KINDS.order) order_bundle.loading = true; + } else if (is_result_kind) { + const arr = order_bundle.results[stage] || (order_bundle.results[stage] = []); + push_capped(arr, ev, MAX_ITEMS_PER_BUCKET); + order_bundle.loading = false; + } else if (ev.kind === KIND_JOB_FEEDBACK) { + const arr = order_bundle.feedback[stage] || (order_bundle.feedback[stage] = []); + push_capped(arr, ev, MAX_ITEMS_PER_BUCKET); + } + + order_bundle.last_update_at = time_now_ms(); + } + + private index_event(ev: NDKEvent, listing_id: string, order_id: string | undefined): void { + if (!ev.id) return; + this.events_to_thread.set(ev.id, { listing_id, order_id }); + } + + private adopt_orphans(parent_id: string, listing_id: string, order_id?: string): void { + const children = this.orphans_by_ref.get(parent_id); + if (!children || children.length === 0) return; + this.orphans_by_ref.delete(parent_id); + for (const child of children) queueMicrotask(() => this.ingest_event(child)); + } + + private resolve_listing_id_from_ref(ref_id?: string): string | undefined { + if (!ref_id) return undefined; + const thread = this.events_to_thread.get(ref_id); + return thread ? thread.listing_id : undefined; + } + + private ingest_event(ev: NDKEvent): void { + if (!ev.id) return; + + if (ev.kind === KIND_RADROOTS_LISTING) { + const listing_id = ev.id; + const listing_bundle = this.ensure_listing(listing_id); + listing_bundle.listing = ev; + + for (const [, ob] of listing_bundle.orders) ob.listing_id = listing_id; + for (const [, ob] of listing_bundle.pending_orders) ob.listing_id = listing_id; + + this.index_event(ev, listing_id, undefined); + this.adopt_orphans(listing_id, listing_id, undefined); + return; + } + + const ref_req_id_raw = get_event_tag(ev.tags, TAG_E); + const ref_req_id = ref_req_id_raw ? ref_req_id_raw : undefined; + + if (ev.kind === REQUEST_KINDS.order) { + const listing_id = + get_job_input_data_for_marker(ev.tags, MARKER_LISTING) || + this.resolve_listing_id_from_ref(ref_req_id) || + ev.id; + + const listing_bundle = this.ensure_listing(listing_id); + const order_bundle = this.ensure_order(listing_bundle, "pending", ev.id); + + this.index_event(ev, listing_id, undefined); + this.attach_event_to_order(order_bundle, ev); + this.adopt_orphans(ev.id, listing_id, undefined); + return; + } + + if (ev.kind === RESULT_KINDS.order) { + const request_id = ref_req_id; + const listing_id = this.resolve_listing_id_from_ref(request_id || ""); + if (!listing_id) { + const arr = this.orphans_by_ref.get(request_id || "") || []; + push_capped(arr, ev, MAX_ITEMS_PER_BUCKET); + this.orphans_by_ref.set(request_id || "", arr); + return; + } + + const listing_bundle = this.ensure_listing(listing_id); + const order_id = ev.id; + + let order_bundle = request_id ? listing_bundle.pending_orders.get(request_id) : undefined; + if (request_id && listing_bundle.pending_orders.has(request_id)) + listing_bundle.pending_orders.delete(request_id); + if (!order_bundle) order_bundle = this.ensure_order(listing_bundle, "orders", order_id); + + if (order_bundle && !listing_bundle.orders.has(order_id)) { + order_bundle.order_id = order_id; + listing_bundle.orders.set(order_id, order_bundle); + } + + this.attach_event_to_order(order_bundle, ev); + this.index_event(ev, listing_id, order_id); + this.adopt_orphans(order_id, listing_id, order_id); + return; + } + + const listing_id = this.resolve_listing_id_from_ref(ref_req_id || ""); + if (!listing_id) { + const arr = this.orphans_by_ref.get(ref_req_id || "") || []; + push_capped(arr, ev, MAX_ITEMS_PER_BUCKET); + this.orphans_by_ref.set(ref_req_id || "", arr); + return; + } + + const listing_bundle = this.ensure_listing(listing_id); + const ref_thread = ref_req_id ? this.events_to_thread.get(ref_req_id) : undefined; + const order_id = ref_thread ? ref_thread.order_id : undefined; + + if (!order_id) { + if (ref_req_id && listing_bundle.pending_orders.has(ref_req_id)) { + const order_bundle = listing_bundle.pending_orders.get(ref_req_id); + if (order_bundle) { + this.attach_event_to_order(order_bundle, ev); + this.index_event(ev, listing_id, undefined); + this.adopt_orphans(ev.id, listing_id, undefined); + return; + } + } + const arr = this.orphans_by_ref.get(ref_req_id || "") || []; + push_capped(arr, ev, MAX_ITEMS_PER_BUCKET); + this.orphans_by_ref.set(ref_req_id || "", arr); + return; + } + + let order_bundle = listing_bundle.orders.get(order_id); + if (!order_bundle) { + order_bundle = this.ensure_order(listing_bundle, "orders", order_id); + listing_bundle.orders.set(order_id, order_bundle); + } + + this.attach_event_to_order(order_bundle, ev); + this.index_event(ev, listing_id, order_id); + this.adopt_orphans(ev.id, listing_id, order_id); + + const waiters = this.waiters_by_request.get(ref_req_id || ""); + if (waiters && waiters.size) { + const created_ms = (ev.created_at || 0) * 1000; + for (const w of Array.from(waiters)) { + if (created_ms > w.since_ms) w.resolve(ev); + } + } + + const is_result_or_feedback = + Object.values(RESULT_KINDS).includes(ev.kind) || ev.kind === KIND_JOB_FEEDBACK; + + if (this.load_complete && is_result_or_feedback) { + this.latest_update_event = ev; + } + } + + private update_loading_by_request(request_id: string, loading: boolean): void { + const thread = this.events_to_thread.get(request_id); + if (!thread) return; + + const listing_bundle = this.listings.get(thread.listing_id); + if (!listing_bundle) return; + + if (listing_bundle.pending_orders.has(request_id)) { + const order_bundle = listing_bundle.pending_orders.get(request_id); + if (order_bundle && order_bundle.loading !== loading) { + listing_bundle.pending_orders.set(request_id, { ...order_bundle, loading }); + } + return; + } + + if (thread.order_id && listing_bundle.orders.has(thread.order_id)) { + const order_bundle = listing_bundle.orders.get(thread.order_id); + if (order_bundle && order_bundle.loading !== loading) { + listing_bundle.orders.set(thread.order_id, { ...order_bundle, loading }); + } + } + } + + private resolve_input_event_id( + stage: Exclude<TradeListingStage, TradeListingStage.Order>, + listing_id: string, + order_id: string + ): string | undefined { + const bundle = this.get_order_bundle(listing_id, order_id); + if (!bundle) return undefined; + + const last_id = (arr?: NDKEvent[]) => { + if (!arr || arr.length === 0) return undefined; + return arr[arr.length - 1].id; + }; + + if ( + stage === TradeListingStage.Accept || + stage === TradeListingStage.Cancel || + stage === TradeListingStage.Refund + ) { + return order_id; + } + if (stage === TradeListingStage.Conveyance || stage === TradeListingStage.Invoice) { + return last_id(bundle.results.Accept); + } + if (stage === TradeListingStage.Payment) { + return last_id(bundle.results.Invoice); + } + if (stage === TradeListingStage.Fulfillment) { + return last_id(bundle.results.Payment); + } + if (stage === TradeListingStage.Receipt) { + return last_id(bundle.results.Fulfillment); + } + return undefined; + } +} + +export function create_trade_flow_service( + opts: CreateTradeFlowServiceOptions +): TradeFlowService { + return new TradeFlowServiceImpl(opts); +} diff --git a/apps-lib-market/src/lib/utils/nostr/trade/listing/types.ts b/apps-lib-market/src/lib/utils/nostr/trade/listing/types.ts @@ -0,0 +1,182 @@ +import { NDKEvent, NDKUser } from "@nostr-dev-kit/ndk"; +import type { ndk, StoreWritable } from "@radroots/apps-lib"; +import type { TradeListingConveyanceRequest, TradeListingOrderRequestPayload, TradeListingPaymentProofRequest, TradeListingStage } from "@radroots/trade-bindings"; +import type { SvelteMap } from "svelte/reactivity"; + +export type TradeListingStageKey = keyof typeof TradeListingStage; + +export type TradeFlowServiceError = + | "error.failed_to_publish" + | "error.timeout" + | "error.missing_payload" + | "error.missing_order_id" + | "error.missing_prerequisite" + | "error.not_implemented" + | "error.service_destroyed"; + +export interface OrderBundle { + order_id?: string; + listing_id: string; + requests: Partial<Record<TradeListingStage, NDKEvent[]>>; + results: Partial<Record<TradeListingStage, NDKEvent[]>>; + feedback: Partial<Record<TradeListingStage, NDKEvent[]>>; + started_at?: number; + last_update_at?: number; + loading: boolean; +} + +export interface TradeListingBundle { + listing?: NDKEvent; + orders: SvelteMap<string, OrderBundle>; + pending_orders: SvelteMap<string, OrderBundle>; +} + +export type OrderRequestOk = { + ok: true; + request: NDKEvent; + result: NDKEvent; + order_id: string; + bundle?: OrderBundle; +}; + +export type OrderRequestErr = { + ok: false; + error: TradeFlowServiceError; + request?: NDKEvent; +}; + +export type OrderRequestResult = OrderRequestOk | OrderRequestErr; + +export type StageActionOk<S extends TradeListingStage> = { + ok: true; + stage: S; + request: NDKEvent; + result: NDKEvent; + order_id: string; + bundle?: OrderBundle; +}; + +export type StageActionErr<S extends TradeListingStage> = { + ok: false; + stage: S; + error: TradeFlowServiceError; + request?: NDKEvent; +}; + +export type StageActionResult<S extends TradeListingStage> = StageActionOk<S> | StageActionErr<S>; + +export type AcceptOptions = { + listing_id: string; + order_id: string; + timeout_ms?: number; +}; + +export type ConveyanceOptions = { + listing_id: string; + order_id: string; + method: TradeListingConveyanceRequest["method"]; + timeout_ms?: number; +}; + +export type InvoiceOptions = { + listing_id: string; + order_id: string; + timeout_ms?: number; +}; + +export type PaymentOptions = { + listing_id: string; + order_id: string; + proof: TradeListingPaymentProofRequest["proof"]; + timeout_ms?: number; +}; + +export type FulfillmentOptions = { + listing_id: string; + order_id: string; + timeout_ms?: number; +}; + +export type ReceiptOptions = { + listing_id: string; + order_id: string; + note?: string; + timeout_ms?: number; +}; + +export type CancelOptions = { + listing_id: string; + order_id: string; + timeout_ms?: number; +}; + +export type RefundOptions = { + listing_id: string; + order_id: string; + timeout_ms?: number; +}; + + +export type StagePostInput = + | { stage: TradeListingStage.Accept; opts: AcceptOptions } + | { stage: TradeListingStage.Conveyance; opts: ConveyanceOptions } + | { stage: TradeListingStage.Invoice; opts: InvoiceOptions } + | { stage: TradeListingStage.Payment; opts: PaymentOptions } + | { stage: TradeListingStage.Fulfillment; opts: FulfillmentOptions } + | { stage: TradeListingStage.Receipt; opts: ReceiptOptions } + | { stage: TradeListingStage.Cancel; opts: CancelOptions } + | { stage: TradeListingStage.Refund; opts: RefundOptions }; + +export type StagePostOutput = + | StageActionResult<TradeListingStage.Accept> + | StageActionResult<TradeListingStage.Conveyance> + | StageActionResult<TradeListingStage.Invoice> + | StageActionResult<TradeListingStage.Payment> + | StageActionResult<TradeListingStage.Fulfillment> + | StageActionResult<TradeListingStage.Receipt> + | StageActionErr<TradeListingStage.Cancel> + | StageActionErr<TradeListingStage.Refund>; + +export interface CreateTradeFlowServiceOptions { + ndk: StoreWritable<typeof ndk>; + ndk_user_store: () => NDKUser; + authors?: string[]; + kinds?: number[]; + default_timeout_ms?: number; +} + +export interface TradeFlowService { + listings: SvelteMap<string, TradeListingBundle>; + + get_latest_update(): NDKEvent | undefined; + + set_filter_authors(authors?: string[] | undefined): void; + set_filter_kinds(kinds: number[]): void; + + get_trade_listing_bundle(listing_id: string): TradeListingBundle | undefined; + get_order_bundle(listing_id: string, order_id: string): OrderBundle | undefined; + is_loading(event_id: string): boolean; + + on_event(ev: NDKEvent): void; + + order_request( + listing_id: string, + payload: TradeListingOrderRequestPayload, + timeout_ms?: number + ): Promise<OrderRequestResult>; + + accept_request(opts: AcceptOptions): Promise<StageActionResult<TradeListingStage.Accept>>; + conveyance_request( + opts: ConveyanceOptions + ): Promise<StageActionResult<TradeListingStage.Conveyance>>; + invoice_request(opts: InvoiceOptions): Promise<StageActionResult<TradeListingStage.Invoice>>; + payment_request(opts: PaymentOptions): Promise<StageActionResult<TradeListingStage.Payment>>; + fulfillment_request( + opts: FulfillmentOptions + ): Promise<StageActionResult<TradeListingStage.Fulfillment>>; + receipt_request(opts: ReceiptOptions): Promise<StageActionResult<TradeListingStage.Receipt>>; + + post(input: StagePostInput): Promise<StagePostOutput>; + + destroy(): void; +} diff --git a/apps-lib-market/src/lib/views/profile/profile-indexed.svelte b/apps-lib-market/src/lib/views/profile/profile-indexed.svelte @@ -0,0 +1,430 @@ +<script lang="ts"> + import LayoutColumnEntry from "$lib/components/layouts/layout-column-entry.svelte"; + import LayoutColumnHeadingDisplaySimple from "$lib/components/layouts/layout-column-heading-display-simple.svelte"; + import LayoutColumnHeadingViewButtons from "$lib/components/layouts/layout-column-heading-view-buttons.svelte"; + import LayoutColumnHeading from "$lib/components/layouts/layout-column-heading.svelte"; + import LayoutColumn from "$lib/components/layouts/layout-column.svelte"; + import { + create_radroots_listing_manager, + create_radroots_profile_manager, + create_trade_flow_service, + head_title_suffix, + type IndexedEventsStorePayload, + type IProfileViewIndexed, + type TradeFlowService, + } from "$root"; + import { + NDKKind, + type NDKEvent, + type NDKUserProfile, + } from "@nostr-dev-kit/ndk"; + import { Glyph, ndk } from "@radroots/apps-lib"; + import type { RadrootsCoreQuantityPrice } from "@radroots/core-bindings"; + import type { + RadrootsListing, + RadrootsListingEventMetadata, + RadrootsListingQuantity, + RadrootsProfile, + RadrootsProfileEventMetadata, + } from "@radroots/events-bindings"; + import { + on_ndk_event, + type RadrootsListingNostrEvent, + type RadrootsProfileNostrEvent, + } from "@radroots/utils-nostr"; + import { onDestroy, onMount } from "svelte"; + + let { basis }: { basis: IProfileViewIndexed } = $props(); + + let trade: TradeFlowService | null = $state(null); + + const listings_mgr = create_radroots_listing_manager(); + const profiles_mgr = create_radroots_profile_manager(); + + function to_indexed_listing_payload_from_metadata( + m: RadrootsListingEventMetadata, + ): IndexedEventsStorePayload<RadrootsListing> { + return { + id: m.id, + kind: 30402, + author: m.author, + published_at: m.published_at ?? 0, + data: m.listing, + source: "indexed", + }; + } + + function to_indexed_profile_payload_from_metadata( + m: RadrootsProfileEventMetadata, + ): IndexedEventsStorePayload<RadrootsProfile> { + return { + id: m.id, + kind: 0, + author: m.author, + published_at: m.published_at ?? 0, + data: m.profile, + source: "indexed", + }; + } + + function current_listings_meta(): RadrootsListingEventMetadata[] { + return "listings" in basis.indexed.events + ? basis.indexed.events.listings + : []; + } + + function current_profile_meta(): RadrootsProfileEventMetadata { + return basis.indexed.events.profile; + } + + let listings_buffer = $state<IndexedEventsStorePayload<RadrootsListing>[]>( + current_listings_meta() + .filter((r) => r.listing && r.listing.d_tag) + .map(to_indexed_listing_payload_from_metadata), + ); + + let profile_buffer = + $state<IndexedEventsStorePayload<RadrootsProfile> | null>( + current_profile_meta() + ? to_indexed_profile_payload_from_metadata( + current_profile_meta(), + ) + : null, + ); + + let have_live_listings = $state(false); + let have_live_profiles = $state(false); + + const listings_view = $derived( + have_live_listings ? listings_mgr.list : listings_buffer, + ); + const profile_view = $derived( + have_live_profiles ? profiles_mgr.list?.[0] : profile_buffer, + ); + + let last_pk = $state(basis.indexed.public_key); + $effect(() => { + if (basis.indexed.public_key !== last_pk) { + const new_listings_meta = current_listings_meta(); + const new_profile_meta = current_profile_meta(); + + listings_buffer = new_listings_meta + .filter((r) => r.listing && r.listing.d_tag) + .map(to_indexed_listing_payload_from_metadata); + + profile_buffer = new_profile_meta + ? to_indexed_profile_payload_from_metadata(new_profile_meta) + : null; + + have_live_listings = false; + have_live_profiles = false; + + trade?.set_filter_authors([basis.indexed.public_key]); + + last_pk = basis.indexed.public_key; + } + }); + + const sub = $ndk.subscribe( + { + kinds: [NDKKind.Metadata, NDKKind.Classified], + authors: [basis.indexed.public_key], + }, + undefined, + { + onEvent: (event: NDKEvent) => { + const parsed = on_ndk_event(event); + if (parsed && "listing" in parsed) { + listings_mgr.on_parsed_event( + parsed as RadrootsListingNostrEvent, + ); + if (!have_live_listings) have_live_listings = true; + } else if (parsed && "profile" in parsed) { + profiles_mgr.on_parsed_event( + parsed as RadrootsProfileNostrEvent, + ); + if (!have_live_profiles) have_live_profiles = true; + } + }, + }, + ); + + onMount(async () => { + trade = create_trade_flow_service({ + ndk: $ndk, + ndk_user_store: () => { + const u = $ndk.activeUser; + if (!u) throw new Error("No active NDK user/signer found."); + return u; + }, + }); + trade.set_filter_authors([basis.indexed.public_key]); + }); + + onDestroy(() => { + sub?.stop(); + trade?.destroy(); + }); + + let ndk_profile: NDKUserProfile | null = $state(null); + const data_user = $derived( + $ndk.getUser({ pubkey: basis.indexed.public_key }), + ); + $effect(() => { + data_user.fetchProfile().then((profile) => { + if (profile) ndk_profile = profile; + }); + }); + + const head_title = $derived( + `${ + basis.indexed.events.profile.profile.display_name || + basis.indexed.events.profile.profile.name + } (@${basis.indexed.events.profile.profile.name}) ${head_title_suffix}`, + ); + + function fmtQty(q: RadrootsListingQuantity): string { + const v = q && q.value ? q.value : undefined; + const amt = v && v.amount ? v.amount : ""; + const unit = v && v.unit ? v.unit : ""; + const lab = v && v.label ? v.label : q && q.label ? q.label : ""; + const pieces = [amt, unit, lab].filter((s) => s && `${s}`.length > 0); + return pieces.join(" "); + } + + function fmtPrice(p: RadrootsCoreQuantityPrice): string { + const a = p && p.amount ? p.amount : undefined; + const q = p && p.quantity ? p.quantity : undefined; + const price = a && a.amount ? a.amount : ""; + const cur = a && a.currency ? a.currency : ""; + const qamt = q && q.amount ? q.amount : ""; + const qun = q && q.unit ? q.unit : ""; + const left = [price, cur] + .filter((s) => s && `${s}`.length > 0) + .join(" "); + const right = [qamt, qun] + .filter((s) => s && `${s}`.length > 0) + .join(" "); + return right ? `${left} per ${right}` : left; + } + + function commentsFor(listingId: string) { + if (!("listings" in basis.indexed.events)) return []; + if (!("listing_comments" in basis.indexed.events)) return []; + const key = listingId.toLowerCase(); + const m = basis.indexed.events.listing_comments; + return m && m[key] ? m[key] : []; + } + + function toDate(ts?: number): string { + if (!ts) return ""; + try { + const d = new Date(ts * 1000); + return d.toLocaleDateString(); + } catch { + return ""; + } + } +</script> + +<svelte:head> + <title>{head_title}</title> + <meta name="description" content={``} /> + <meta property="og:title" content={head_title} /> + <meta property="og:description" content={``} /> +</svelte:head> + +<LayoutColumn> + <LayoutColumnEntry basis={{ classes: `bg-white` }}> + <LayoutColumnHeading> + {#snippet heading()} + <LayoutColumnHeadingDisplaySimple> + {#snippet row1()} + <p + class={`font-br font-[600] text-base text-black_panther`} + > + {basis.indexed.events.profile.profile.name} + </p> + <Glyph + basis={{ + classes: `text-lime-500`, + size: `sm`, + key: `plant`, + }} + /> + {/snippet} + {#snippet row2()} + <p + class={`font-br font-[400] text-sm text-black_panther`} + > + {basis.indexed.events.profile.profile + .display_name || + basis.indexed.events.profile.profile.name} + </p> + {/snippet} + {#snippet row3()} + <p + class={`font-rsfd font-[600] text-sm text-black_panther`} + > + {`${30}M followers`} + </p> + <p + class={`font-rsfd font-[600] text-sm text-black_panther`} + > + {`${209} following`} + </p> + {/snippet} + </LayoutColumnHeadingDisplaySimple> + {/snippet} + {#snippet subheading()} + <p class={`font-sans font-[400] text-sm text-black_panther`}> + {basis.indexed.events.profile.profile.about} + </p> + {/snippet} + </LayoutColumnHeading> + <LayoutColumnHeadingViewButtons /> + </LayoutColumnEntry> + + {#if "listings" in basis.indexed.events} + <LayoutColumnEntry basis={{ classes: `gap-4` }}> + {#each basis.indexed.events.listings as ev (ev.id)} + <div class={`relative flex w-full flex-col`}> + <div class={`flex w-full flex-col gap-2 p-4`}> + <div class={`flex w-full flex-row justify-between`}> + <p + class={`font-sans text-base font-[500] text-black_panther`} + > + {ev.listing.product.title} + </p> + <p + class={`font-sans text-xs font-[500] text-cloak_grey`} + > + {toDate(ev.published_at)} + </p> + </div> + <p + class={`font-sans text-sm font-[400] text-black_panther/80`} + > + {ev.listing.product.summary} + </p> + <div class={`flex w-full flex-wrap gap-2 pt-1`}> + <span + class={`rounded-sm bg-ly1 px-2 py-0.5 text-xs font-[600] text-black_panther/90`} + > + {ev.listing.product.category} + </span> + {#if ev.listing.product.process} + <span + class={`rounded-sm bg-ly1 px-2 py-0.5 text-xs font-[600] text-black_panther/90`} + > + {ev.listing.product.process} + </span> + {/if} + {#if ev.listing.product.year} + <span + class={`rounded-sm bg-ly1 px-2 py-0.5 text-xs font-[600] text-black_panther/90`} + > + {ev.listing.product.year} + </span> + {/if} + {#if ev.listing.product.lot} + <span + class={`rounded-sm bg-ly1 px-2 py-0.5 text-xs font-[600] text-black_panther/90`} + > + {ev.listing.product.lot} + </span> + {/if} + </div> + <div class={`flex w-full flex-row gap-4 pt-2`}> + <div class={`flex flex-col`}> + <p + class={`font-sans text-xs font-[700] text-black_panther/70`} + > + Quantities + </p> + <p + class={`font-sans text-sm text-black_panther/90`} + > + {ev.listing.quantities + .map((q) => fmtQty(q)) + .filter((s) => s.length > 0) + .join(", ")} + </p> + </div> + <div class={`flex flex-col`}> + <p + class={`font-sans text-xs font-[700] text-black_panther/70`} + > + Prices + </p> + <p + class={`font-sans text-sm text-black_panther/90`} + > + {ev.listing.prices + .map((p) => fmtPrice(p)) + .filter((s) => s.length > 0) + .join(" ยท ")} + </p> + </div> + </div> + <div class={`flex w-full flex-row gap-2 pt-2`}> + <p + class={`font-sans text-xs font-[700] text-black_panther/70`} + > + Location + </p> + <p + class={`font-sans text-sm text-black_panther/90`} + > + {#if ev.listing.location} + {ev.listing.location.primary} + {ev.listing.location.city + ? `, ${ev.listing.location.city}` + : ""} + {ev.listing.location.region + ? `, ${ev.listing.location.region}` + : ""} + {ev.listing.location.country + ? `, ${ev.listing.location.country.toUpperCase()}` + : ""} + {:else} + {"Unlisted"} + {/if} + </p> + </div> + </div> + + <div class={`flex w-full flex-col gap-2 p-4`}> + <p + class={`font-sans text-xs font-[700] uppercase tracking-wide text-black_panther/70`} + > + Comments + </p> + {#each commentsFor(ev.id) as c (c.id)} + <div + class={`flex w-full flex-col gap-1 rounded-sm bg-ly1/40 p-3`} + > + <p + class={`font-sans text-xs font-[600] text-black_panther/70`} + > + {toDate(c.published_at)} + </p> + <p + class={`font-sans text-sm font-[400] text-black_panther`} + > + {c.comment && c.comment.content + ? c.comment.content + : ""} + </p> + </div> + {:else} + <p class={`font-sans text-sm text-cloak_grey`}> + No comments yet + </p> + {/each} + </div> + </div> + {/each} + </LayoutColumnEntry> + {/if} +</LayoutColumn> diff --git a/apps-lib-market/src/lib/views/profile/profile-network-nip05.svelte b/apps-lib-market/src/lib/views/profile/profile-network-nip05.svelte @@ -0,0 +1,22 @@ +<script lang="ts"> + import { + ProfileNetworkPublicKey, + type IProfileViewNetworkNip05, + } from "$root"; + + let { + basis, + }: { + basis: IProfileViewNetworkNip05; + } = $props(); + + let public_key: string | undefined = $state(undefined); +</script> + +{#if public_key} + <ProfileNetworkPublicKey basis={{ public_key }} /> +{:else} + <p class={`font-sans font-[400] text-base text-ly0-gl`}> + {`could not find a public key for ${basis.nip05}`} + </p> +{/if} diff --git a/apps-lib-market/src/lib/views/profile/profile-network-npub.svelte b/apps-lib-market/src/lib/views/profile/profile-network-npub.svelte @@ -0,0 +1,30 @@ +<script lang="ts"> + import { + type IProfileViewNetworkNpub, + ProfileNetworkPublicKey, + } from "$root"; + import { lib_nostr_npub_decode } from "@radroots/utils-nostr"; + import { error } from "@sveltejs/kit"; + import { onMount } from "svelte"; + + let { + basis, + }: { + basis: IProfileViewNetworkNpub; + } = $props(); + + let public_key: string | undefined = $state(undefined); + + onMount(async () => { + public_key = lib_nostr_npub_decode(basis.npub); + if (!public_key) error(404, `invalid:public_key:${public_key}`); + }); +</script> + +{#if public_key} + <ProfileNetworkPublicKey basis={{ public_key }} /> +{:else} + <p class={`font-sans font-[400] text-base text-ly0-gl`}> + {`not a valid npub ${basis.npub}`} + </p> +{/if} diff --git a/apps-lib-market/src/lib/views/profile/profile-network-public-key.svelte b/apps-lib-market/src/lib/views/profile/profile-network-public-key.svelte @@ -0,0 +1,18 @@ +<script lang="ts"> + import { type IProfileViewNetworkPublicKey } from "$root"; + + let { + basis, + }: { + basis: IProfileViewNetworkPublicKey; + } = $props(); +</script> + +<div class={`flex flex-col w-full px-4 pt-8 gap-4 justify-start items-start`}> + <p class={`font-sans font-[400] text-base text-ly0-gl`}> + {`profile`} + </p> + <p class={`font-sans font-[400] text-base text-ly0-gl break-all`}> + {`public_key `}{basis.public_key} + </p> +</div> diff --git a/apps-lib-market/src/lib/views/profile/profile-network.svelte b/apps-lib-market/src/lib/views/profile/profile-network.svelte @@ -0,0 +1,22 @@ +<script lang="ts"> + import { + type IProfileViewNetwork, + ProfileNetworkNip05, + ProfileNetworkNpub, + ProfileNetworkPublicKey, + } from "$root"; + + let { basis }: { basis: IProfileViewNetwork } = $props(); +</script> + +{#if basis.unknown && `public_key` in basis.unknown} + <ProfileNetworkPublicKey basis={basis.unknown} /> +{:else if basis.unknown && `npub` in basis.unknown} + <ProfileNetworkNpub basis={basis.unknown} /> +{:else if basis.unknown && `nip05` in basis.unknown} + <ProfileNetworkNip05 basis={basis.unknown} /> +{:else} + <p class={`font-sans font-[400] text-base text-ly0-gl`}> + {`profile not found`} + </p> +{/if} diff --git a/apps-lib-market/src/lib/views/profile/profile.svelte b/apps-lib-market/src/lib/views/profile/profile.svelte @@ -0,0 +1,19 @@ +<script lang="ts"> + import { type IProfileView, ProfileIndexed, ProfileNetwork } from "$root"; + + let { + basis, + }: { + basis: IProfileView; + } = $props(); + + $effect(() => { + console.log(JSON.stringify(basis, null, 4), `basis`); + }); +</script> + +{#if `indexed` in basis} + <ProfileIndexed {basis} /> +{:else if `unknown` in basis && basis.unknown} + <ProfileNetwork {basis} /> +{/if} diff --git a/apps-lib-market/svelte.config.js b/apps-lib-market/svelte.config.js @@ -0,0 +1,15 @@ +import adapter from '@sveltejs/adapter-auto'; +import { vitePreprocess } from '@sveltejs/vite-plugin-svelte'; + +/** @type {import('@sveltejs/kit').Config} */ +const config = { + preprocess: vitePreprocess(), + kit: { + adapter: adapter(), + alias: { + $root: './src/lib/index.js', + } + }, +}; + +export default config; diff --git a/apps-lib-market/tsconfig.json b/apps-lib-market/tsconfig.json @@ -0,0 +1,13 @@ +{ + "extends": "./.svelte-kit/tsconfig.json", + "compilerOptions": { + "allowJs": true, + "checkJs": true, + "esModuleInterop": true, + "forceConsistentCasingInFileNames": true, + "resolveJsonModule": true, + "skipLibCheck": true, + "sourceMap": true, + "strict": true, + } +} diff --git a/apps-lib-market/vite.config.ts b/apps-lib-market/vite.config.ts @@ -0,0 +1,11 @@ +import { sveltekit } from '@sveltejs/kit/vite'; +import { defineConfig } from 'vite'; + +export default defineConfig({ + resolve: { + alias: { + '$root': '/src/lib/index.js', + } + }, + plugins: [sveltekit()] +});