Merge branch 'master' into update-docs-api
This commit is contained in:
		
						commit
						7de3d789b3
					
				
							
								
								
									
										8
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										8
									
								
								LICENSE
									
									
									
									
									
								
							| @ -12,10 +12,10 @@ or any other contributor to The Mempool Open Source Project. | |||||||
| 
 | 
 | ||||||
| The Mempool Open Source Project®, Mempool Accelerator™, Mempool Enterprise®,  | The Mempool Open Source Project®, Mempool Accelerator™, Mempool Enterprise®,  | ||||||
| Mempool Liquidity™, mempool.space®, Be your own explorer™, Explore the full  | Mempool Liquidity™, mempool.space®, Be your own explorer™, Explore the full  | ||||||
| Bitcoin ecosystem™, the mempool Logo, the mempool Square logo, the mempool Blocks  | Bitcoin ecosystem™, Mempool Goggles™, the mempool Logo, the mempool Square logo,  | ||||||
| logo, the mempool Blocks 3 | 2 logo, the mempool.space Vertical Logo, and the  | the mempool Blocks logo, the mempool Blocks 3 | 2 logo, the mempool.space Vertical  | ||||||
| mempool.space Horizontal logo are registered trademarks or trademarks of Mempool | Logo, and the mempool.space Horizontal logo are registered trademarks or trademarks  | ||||||
| Space K.K in Japan, the United States, and/or other countries. | of Mempool Space K.K in Japan, the United States, and/or other countries. | ||||||
| 
 | 
 | ||||||
| See our full Trademark Policy and Guidelines for more details, published on  | See our full Trademark Policy and Guidelines for more details, published on  | ||||||
| <https://mempool.space/trademark-policy>. | <https://mempool.space/trademark-policy>. | ||||||
|  | |||||||
							
								
								
									
										675
									
								
								LICENSE.GPL-3.md
									
									
									
									
									
								
							
							
						
						
									
										675
									
								
								LICENSE.GPL-3.md
									
									
									
									
									
								
							| @ -1,675 +0,0 @@ | |||||||
| ### 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>. |  | ||||||
| @ -39,15 +39,25 @@ class FeeApi { | |||||||
|     const secondMedianFee = pBlocks[1] ? this.optimizeMedianFee(pBlocks[1], pBlocks[2], firstMedianFee) : this.defaultFee; |     const secondMedianFee = pBlocks[1] ? this.optimizeMedianFee(pBlocks[1], pBlocks[2], firstMedianFee) : this.defaultFee; | ||||||
|     const thirdMedianFee = pBlocks[2] ? this.optimizeMedianFee(pBlocks[2], pBlocks[3], secondMedianFee) : this.defaultFee; |     const thirdMedianFee = pBlocks[2] ? this.optimizeMedianFee(pBlocks[2], pBlocks[3], secondMedianFee) : this.defaultFee; | ||||||
| 
 | 
 | ||||||
|  |     let fastestFee = Math.max(minimumFee, firstMedianFee); | ||||||
|  |     let halfHourFee = Math.max(minimumFee, secondMedianFee); | ||||||
|  |     let hourFee = Math.max(minimumFee, thirdMedianFee); | ||||||
|  |     const economyFee = Math.max(minimumFee, Math.min(2 * minimumFee, thirdMedianFee)); | ||||||
|  | 
 | ||||||
|  |     // ensure recommendations always increase w/ priority
 | ||||||
|  |     fastestFee = Math.max(fastestFee, halfHourFee, hourFee, economyFee); | ||||||
|  |     halfHourFee = Math.max(halfHourFee, hourFee, economyFee); | ||||||
|  |     hourFee = Math.max(hourFee, economyFee); | ||||||
|  | 
 | ||||||
|     // explicitly enforce a minimum of ceil(mempoolminfee) on all recommendations.
 |     // explicitly enforce a minimum of ceil(mempoolminfee) on all recommendations.
 | ||||||
|     // simply rounding up recommended rates is insufficient, as the purging rate
 |     // simply rounding up recommended rates is insufficient, as the purging rate
 | ||||||
|     // can exceed the median rate of projected blocks in some extreme scenarios
 |     // can exceed the median rate of projected blocks in some extreme scenarios
 | ||||||
|     // (see https://bitcoin.stackexchange.com/a/120024)
 |     // (see https://bitcoin.stackexchange.com/a/120024)
 | ||||||
|     return { |     return { | ||||||
|       'fastestFee': Math.max(minimumFee, firstMedianFee), |       'fastestFee': fastestFee, | ||||||
|       'halfHourFee': Math.max(minimumFee, secondMedianFee), |       'halfHourFee': halfHourFee, | ||||||
|       'hourFee': Math.max(minimumFee, thirdMedianFee), |       'hourFee': hourFee, | ||||||
|       'economyFee': Math.max(minimumFee, Math.min(2 * minimumFee, thirdMedianFee)), |       'economyFee': economyFee, | ||||||
|       'minimumFee': minimumFee, |       'minimumFee': minimumFee, | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -282,8 +282,7 @@ class WebsocketHandler { | |||||||
|           } |           } | ||||||
| 
 | 
 | ||||||
|           if (Object.keys(response).length) { |           if (Object.keys(response).length) { | ||||||
|             const serializedResponse = this.serializeResponse(response); |             client.send(this.serializeResponse(response)); | ||||||
|             client.send(serializedResponse); |  | ||||||
|           } |           } | ||||||
|         } catch (e) { |         } catch (e) { | ||||||
|           logger.debug(`Error parsing websocket message from ${client['remoteAddress']}: ` + (e instanceof Error ? e.message : e)); |           logger.debug(`Error parsing websocket message from ${client['remoteAddress']}: ` + (e instanceof Error ? e.message : e)); | ||||||
| @ -392,8 +391,7 @@ class WebsocketHandler { | |||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       if (Object.keys(response).length) { |       if (Object.keys(response).length) { | ||||||
|         const serializedResponse = this.serializeResponse(response); |         client.send(this.serializeResponse(response)); | ||||||
|         client.send(serializedResponse); |  | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| @ -639,8 +637,7 @@ class WebsocketHandler { | |||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       if (Object.keys(response).length) { |       if (Object.keys(response).length) { | ||||||
|         const serializedResponse = this.serializeResponse(response); |         client.send(this.serializeResponse(response)); | ||||||
|         client.send(serializedResponse); |  | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| @ -738,10 +735,13 @@ class WebsocketHandler { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     const confirmedTxids: { [txid: string]: boolean } = {}; | ||||||
|  | 
 | ||||||
|     // Update mempool to remove transactions included in the new block
 |     // Update mempool to remove transactions included in the new block
 | ||||||
|     for (const txId of txIds) { |     for (const txId of txIds) { | ||||||
|       delete _memPool[txId]; |       delete _memPool[txId]; | ||||||
|       rbfCache.mined(txId); |       rbfCache.mined(txId); | ||||||
|  |       confirmedTxids[txId] = true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (config.MEMPOOL.ADVANCED_GBT_MEMPOOL) { |     if (config.MEMPOOL.ADVANCED_GBT_MEMPOOL) { | ||||||
| @ -773,6 +773,8 @@ class WebsocketHandler { | |||||||
|       'fees': fees, |       'fees': fees, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|  |     const mBlocksWithTransactions = mempoolBlocks.getMempoolBlocksWithTransactions(); | ||||||
|  | 
 | ||||||
|     const responseCache = { ...this.socketData }; |     const responseCache = { ...this.socketData }; | ||||||
|     function getCachedResponse(key, data): string { |     function getCachedResponse(key, data): string { | ||||||
|       if (!responseCache[key]) { |       if (!responseCache[key]) { | ||||||
| @ -808,7 +810,7 @@ class WebsocketHandler { | |||||||
| 
 | 
 | ||||||
|       if (client['track-tx']) { |       if (client['track-tx']) { | ||||||
|         const trackTxid = client['track-tx']; |         const trackTxid = client['track-tx']; | ||||||
|         if (trackTxid && txIds.indexOf(trackTxid) > -1) { |         if (trackTxid && confirmedTxids[trackTxid]) { | ||||||
|           response['txConfirmed'] = JSON.stringify(trackTxid); |           response['txConfirmed'] = JSON.stringify(trackTxid); | ||||||
|         } else { |         } else { | ||||||
|           const mempoolTx = _memPool[trackTxid]; |           const mempoolTx = _memPool[trackTxid]; | ||||||
| @ -880,17 +882,24 @@ class WebsocketHandler { | |||||||
| 
 | 
 | ||||||
|       if (client['track-mempool-block'] >= 0 && memPool.isInSync()) { |       if (client['track-mempool-block'] >= 0 && memPool.isInSync()) { | ||||||
|         const index = client['track-mempool-block']; |         const index = client['track-mempool-block']; | ||||||
|         if (mBlockDeltas && mBlockDeltas[index]) { | 
 | ||||||
|           response['projected-block-transactions'] = getCachedResponse(`projected-block-transactions-${index}`, { |         if (mBlockDeltas && mBlockDeltas[index] && mBlocksWithTransactions[index]?.transactions?.length) { | ||||||
|             index: index, |           if (mBlockDeltas[index].added.length > (mBlocksWithTransactions[index]?.transactions.length / 2)) { | ||||||
|             delta: mBlockDeltas[index], |             response['projected-block-transactions'] = getCachedResponse(`projected-block-transactions-full-${index}`, { | ||||||
|           }); |               index: index, | ||||||
|  |               blockTransactions: mBlocksWithTransactions[index].transactions, | ||||||
|  |             }); | ||||||
|  |           } else { | ||||||
|  |             response['projected-block-transactions'] = getCachedResponse(`projected-block-transactions-delta-${index}`, { | ||||||
|  |               index: index, | ||||||
|  |               delta: mBlockDeltas[index], | ||||||
|  |             }); | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       if (Object.keys(response).length) { |       if (Object.keys(response).length) { | ||||||
|         const serializedResponse = this.serializeResponse(response); |         client.send(this.serializeResponse(response)); | ||||||
|         client.send(serializedResponse); |  | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| @ -951,10 +960,27 @@ class WebsocketHandler { | |||||||
| 
 | 
 | ||||||
|   private printLogs(): void { |   private printLogs(): void { | ||||||
|     if (this.wss) { |     if (this.wss) { | ||||||
|  |       let numTxSubs = 0; | ||||||
|  |       let numProjectedSubs = 0; | ||||||
|  |       let numRbfSubs = 0; | ||||||
|  | 
 | ||||||
|  |       this.wss.clients.forEach((client) => { | ||||||
|  |         if (client['track-tx']) { | ||||||
|  |           numTxSubs++; | ||||||
|  |         } | ||||||
|  |         if (client['track-mempool-block'] != null && client['track-mempool-block'] >= 0) { | ||||||
|  |           numProjectedSubs++; | ||||||
|  |         } | ||||||
|  |         if (client['track-rbf']) { | ||||||
|  |           numRbfSubs++; | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  | 
 | ||||||
|       const count = this.wss?.clients?.size || 0; |       const count = this.wss?.clients?.size || 0; | ||||||
|       const diff = count - this.numClients; |       const diff = count - this.numClients; | ||||||
|       this.numClients = count; |       this.numClients = count; | ||||||
|       logger.debug(`${count} websocket clients | ${this.numConnected} connected | ${this.numDisconnected} disconnected | (${diff >= 0 ? '+' : ''}${diff})`); |       logger.debug(`${count} websocket clients | ${this.numConnected} connected | ${this.numDisconnected} disconnected | (${diff >= 0 ? '+' : ''}${diff})`); | ||||||
|  |       logger.debug(`websocket subscriptions: track-tx: ${numTxSubs}, track-mempool-block: ${numProjectedSubs} track-rbf: ${numRbfSubs}`); | ||||||
|       this.numConnected = 0; |       this.numConnected = 0; | ||||||
|       this.numDisconnected = 0; |       this.numDisconnected = 0; | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -422,7 +422,7 @@ | |||||||
|       Trademark Notice<br> |       Trademark Notice<br> | ||||||
|     </div> |     </div> | ||||||
|     <p> |     <p> | ||||||
|       The Mempool Open Source Project®, Mempool Accelerator™, Mempool Enterprise®, Mempool Liquidity™, mempool.space®, Be your own explorer™, Explore the full Bitcoin ecosystem™, the mempool logo;, the mempool Square logo;, the mempool Blocks logo;, the mempool Blocks 3 | 2 logo;, the mempool.space Vertical Logo;, and the mempool.space Horizontal logo are either registered trademarks or trademarks of Mempool Space K.K in Japan, the United States, and/or other countries. |       The Mempool Open Source Project®, Mempool Accelerator™, Mempool Enterprise®, Mempool Liquidity™, mempool.space®, Be your own explorer™, Explore the full Bitcoin ecosystem™, Mempool Goggles™, the mempool logo, the mempool Square logo, the mempool Blocks logo, the mempool Blocks 3 | 2 logo, the mempool.space Vertical Logo, and the mempool.space Horizontal logo are either registered trademarks or trademarks of Mempool Space K.K in Japan, the United States, and/or other countries. | ||||||
|     </p> |     </p> | ||||||
|     <p> |     <p> | ||||||
|       While our software is available under an open source software license, the copyright license does not include an implied right or license to use our trademarks. See our <a href="https://mempool.space/trademark-policy">Trademark Policy and Guidelines</a> for more details, published on <https://mempool.space/trademark-policy>. |       While our software is available under an open source software license, the copyright license does not include an implied right or license to use our trademarks. See our <a href="https://mempool.space/trademark-policy">Trademark Policy and Guidelines</a> for more details, published on <https://mempool.space/trademark-policy>. | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ import { Subscription, catchError, of, tap } from 'rxjs'; | |||||||
| import { StorageService } from '../../services/storage.service'; | import { StorageService } from '../../services/storage.service'; | ||||||
| import { Transaction } from '../../interfaces/electrs.interface'; | import { Transaction } from '../../interfaces/electrs.interface'; | ||||||
| import { nextRoundNumber } from '../../shared/common.utils'; | import { nextRoundNumber } from '../../shared/common.utils'; | ||||||
|  | import { AudioService } from '../../services/audio.service'; | ||||||
| 
 | 
 | ||||||
| export type AccelerationEstimate = { | export type AccelerationEstimate = { | ||||||
|   txSummary: TxSummary; |   txSummary: TxSummary; | ||||||
| @ -63,6 +64,7 @@ export class AcceleratePreviewComponent implements OnInit, OnDestroy, OnChanges | |||||||
|   constructor( |   constructor( | ||||||
|     private apiService: ApiService, |     private apiService: ApiService, | ||||||
|     private storageService: StorageService, |     private storageService: StorageService, | ||||||
|  |     private audioService: AudioService, | ||||||
|     private cd: ChangeDetectorRef |     private cd: ChangeDetectorRef | ||||||
|   ) { } |   ) { } | ||||||
| 
 | 
 | ||||||
| @ -186,6 +188,7 @@ export class AcceleratePreviewComponent implements OnInit, OnDestroy, OnChanges | |||||||
|       this.userBid |       this.userBid | ||||||
|     ).subscribe({ |     ).subscribe({ | ||||||
|       next: () => { |       next: () => { | ||||||
|  |         this.audioService.playSound('ascend-chime-cartoon'); | ||||||
|         this.showSuccess = true; |         this.showSuccess = true; | ||||||
|         this.scrollToPreviewWithTimeout('successAlert', 'center'); |         this.scrollToPreviewWithTimeout('successAlert', 'center'); | ||||||
|         this.estimateSubscription.unsubscribe(); |         this.estimateSubscription.unsubscribe(); | ||||||
|  | |||||||
| @ -1,10 +1,10 @@ | |||||||
| <div class="container-xl" style="min-height: 335px" [class.widget]="widget" [class.full-height]="!widget"> | <div class="container-xl widget-container" [class.widget]="widget" [class.full-height]="!widget"> | ||||||
|   <h1 *ngIf="!widget" class="float-left" i18n="master-page.blocks">Accelerations</h1> |   <h1 *ngIf="!widget" class="float-left" i18n="master-page.blocks">Accelerations</h1> | ||||||
|   <div *ngIf="!widget && isLoading" class="spinner-border ml-3" role="status"></div> |   <div *ngIf="!widget && isLoading" class="spinner-border ml-3" role="status"></div> | ||||||
| 
 | 
 | ||||||
|   <div class="clearfix"></div> |   <div class="clearfix"></div> | ||||||
| 
 | 
 | ||||||
|   <div style="min-height: 295px" *ngIf="accelerationList$ | async as accelerations"> |   <div class="acceleration-list" *ngIf="accelerationList$ | async as accelerations"> | ||||||
|     <table *ngIf="!accelerations || accelerations.length; else noData" class="table table-borderless table-fixed"> |     <table *ngIf="!accelerations || accelerations.length; else noData" class="table table-borderless table-fixed"> | ||||||
|       <thead> |       <thead> | ||||||
|         <th class="txid text-left" i18n="dashboard.latest-transactions.txid">TXID</th> |         <th class="txid text-left" i18n="dashboard.latest-transactions.txid">TXID</th> | ||||||
|  | |||||||
| @ -14,11 +14,24 @@ | |||||||
| .container-xl.legacy { | .container-xl.legacy { | ||||||
|   max-width: 1140px; |   max-width: 1140px; | ||||||
| } | } | ||||||
|  | .container-xl.widget-container { | ||||||
|  |   min-height: 335px; | ||||||
|  |   @media (max-width: 767px) { | ||||||
|  |     min-height: auto; | ||||||
|  |   } | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| .container { | .container { | ||||||
|   max-width: 100%; |   max-width: 100%; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .acceleration-list { | ||||||
|  |   min-height: 295px; | ||||||
|  |   @media (max-width: 767px) { | ||||||
|  |     min-height: auto; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| tr, td, th { | tr, td, th { | ||||||
|   border: 0px; |   border: 0px; | ||||||
|   padding-top: 0.65rem !important; |   padding-top: 0.65rem !important; | ||||||
| @ -51,34 +64,63 @@ tr, td, th { | |||||||
| 
 | 
 | ||||||
| .txid { | .txid { | ||||||
|   width: 25%; |   width: 25%; | ||||||
|   @media (max-width: 1100px) { |  | ||||||
|     padding-right: 10px; |  | ||||||
|   } |  | ||||||
|   @media (max-width: 875px) { |  | ||||||
|     display: none; |  | ||||||
|   } |  | ||||||
|   overflow: hidden; |   overflow: hidden; | ||||||
|   text-overflow: ellipsis; |   text-overflow: ellipsis; | ||||||
|   white-space: nowrap; |   white-space: nowrap; | ||||||
|   max-width: 30%; |   max-width: 30%; | ||||||
|  |   @media (max-width: 1060px) and (min-width: 768px) { | ||||||
|  |     display: none; | ||||||
|  |   } | ||||||
|  |   @media (max-width: 500px) { | ||||||
|  |     display: none; | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .fee { | .fee-rate { | ||||||
|   width: 35%; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .block { |  | ||||||
|   width: 20%; |   width: 20%; | ||||||
|  |   @media (max-width: 1060px) and (min-width: 768px) { | ||||||
|  |     text-align: start !important; | ||||||
|  |   } | ||||||
|  |   @media (max-width: 500px) { | ||||||
|  |     text-align: start !important; | ||||||
|  |   } | ||||||
|  |   @media (max-width: 840px) and (min-width: 768px) { | ||||||
|  |     display: none; | ||||||
|  |   } | ||||||
|  |   @media (max-width: 410px) { | ||||||
|  |     display: none; | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .bid { | .bid { | ||||||
|   width: 30%; |   width: 30%; | ||||||
|  |   min-width: 150px; | ||||||
|  |   @media (max-width: 840px) and (min-width: 768px) { | ||||||
|  |     text-align: start !important; | ||||||
|  |   } | ||||||
|  |   @media (max-width: 410px) { | ||||||
|  |     text-align: start !important; | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .time { | .time { | ||||||
|   width: 25%; |   width: 25%; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .fee { | ||||||
|  |   width: 35%; | ||||||
|  |   @media (max-width: 1060px) and (min-width: 768px) { | ||||||
|  |     text-align: start !important; | ||||||
|  |   } | ||||||
|  |   @media (max-width: 500px) { | ||||||
|  |     text-align: start !important; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .block { | ||||||
|  |   width: 20%; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .status { | .status { | ||||||
|   width: 20% |   width: 20% | ||||||
| } | } | ||||||
| @ -122,4 +164,7 @@ tr, td, th { | |||||||
|   flex-direction: row; |   flex-direction: row; | ||||||
|   align-items: center; |   align-items: center; | ||||||
|   justify-content: center; |   justify-content: center; | ||||||
|  |   @media (max-width: 767px) { | ||||||
|  |     height: 100px; | ||||||
|  |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -139,6 +139,9 @@ | |||||||
| } | } | ||||||
| .list-card { | .list-card { | ||||||
|   height: 410px; |   height: 410px; | ||||||
|  |   @media (max-width: 767px) { | ||||||
|  |     height: auto; | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mempool-block-wrapper { | .mempool-block-wrapper { | ||||||
|  | |||||||
| @ -11,7 +11,7 @@ export default class BlockScene { | |||||||
|   getColor: ((tx: TxView) => Color) = defaultColorFunction; |   getColor: ((tx: TxView) => Color) = defaultColorFunction; | ||||||
|   orientation: string; |   orientation: string; | ||||||
|   flip: boolean; |   flip: boolean; | ||||||
|   animationDuration: number = 1000; |   animationDuration: number = 900; | ||||||
|   configAnimationOffset: number | null; |   configAnimationOffset: number | null; | ||||||
|   animationOffset: number; |   animationOffset: number; | ||||||
|   highlightingEnabled: boolean; |   highlightingEnabled: boolean; | ||||||
|  | |||||||
| @ -58,6 +58,10 @@ | |||||||
|           <td *ngSwitchCase="'accelerated'"><span class="badge badge-accelerated" i18n="transaction.audit.accelerated">Accelerated</span></td> |           <td *ngSwitchCase="'accelerated'"><span class="badge badge-accelerated" i18n="transaction.audit.accelerated">Accelerated</span></td> | ||||||
|         </ng-container> |         </ng-container> | ||||||
|       </tr> |       </tr> | ||||||
|  |       <tr *ngIf="!auditEnabled && tx && tx.status === 'accelerated'"> | ||||||
|  |         <td class="td-width"></td> | ||||||
|  |         <td><span class="badge badge-accelerated" i18n="transaction.audit.accelerated">Accelerated</span></td> | ||||||
|  |       </tr> | ||||||
|     </tbody> |     </tbody> | ||||||
|   </table> |   </table> | ||||||
| </div> | </div> | ||||||
|  | |||||||
| @ -2,7 +2,7 @@ import { Component, OnInit, OnDestroy, ViewChild, ElementRef } from '@angular/co | |||||||
| import { ActivatedRoute, ParamMap } from '@angular/router'; | import { ActivatedRoute, ParamMap } from '@angular/router'; | ||||||
| import { ElectrsApiService } from '../../services/electrs-api.service'; | import { ElectrsApiService } from '../../services/electrs-api.service'; | ||||||
| import { switchMap, tap, throttleTime, catchError, shareReplay, startWith, pairwise, filter } from 'rxjs/operators'; | import { switchMap, tap, throttleTime, catchError, shareReplay, startWith, pairwise, filter } from 'rxjs/operators'; | ||||||
| import { of, Subscription, asyncScheduler } from 'rxjs'; | import { of, Subscription, asyncScheduler, forkJoin } from 'rxjs'; | ||||||
| import { StateService } from '../../services/state.service'; | import { StateService } from '../../services/state.service'; | ||||||
| import { SeoService } from '../../services/seo.service'; | import { SeoService } from '../../services/seo.service'; | ||||||
| import { OpenGraphService } from '../../services/opengraph.service'; | import { OpenGraphService } from '../../services/opengraph.service'; | ||||||
| @ -121,21 +121,37 @@ export class BlockPreviewComponent implements OnInit, OnDestroy { | |||||||
|     this.overviewSubscription = block$.pipe( |     this.overviewSubscription = block$.pipe( | ||||||
|       startWith(null), |       startWith(null), | ||||||
|       pairwise(), |       pairwise(), | ||||||
|       switchMap(([prevBlock, block]) => this.apiService.getStrippedBlockTransactions$(block.id) |       switchMap(([prevBlock, block]) => { | ||||||
|         .pipe( |           return forkJoin([ | ||||||
|           catchError((err) => { |             this.apiService.getStrippedBlockTransactions$(block.id) | ||||||
|             this.overviewError = err; |               .pipe( | ||||||
|             this.openGraphService.fail('block-viz-' + this.rawId); |                 catchError((err) => { | ||||||
|             return of([]); |                   this.overviewError = err; | ||||||
|           }), |                   this.openGraphService.fail('block-viz-' + this.rawId); | ||||||
|           switchMap((transactions) => { |                   return of([]); | ||||||
|             return of({ transactions, direction: 'down' }); |                 }), | ||||||
|           }) |                 switchMap((transactions) => { | ||||||
|         ) |                   return of(transactions); | ||||||
|  |                 }) | ||||||
|  |               ), | ||||||
|  |             this.stateService.env.ACCELERATOR === true && block.height > 819500 ? this.apiService.getAccelerationHistory$({ blockHash: block.id }) : of([]) | ||||||
|  |           ]); | ||||||
|  |         } | ||||||
|       ), |       ), | ||||||
|     ) |     ) | ||||||
|     .subscribe(({transactions, direction}: {transactions: TransactionStripped[], direction: string}) => { |     .subscribe(([transactions, accelerations]) => { | ||||||
|       this.strippedTransactions = transactions; |       this.strippedTransactions = transactions; | ||||||
|  | 
 | ||||||
|  |       const acceleratedInBlock = {}; | ||||||
|  |       for (const acc of accelerations) { | ||||||
|  |         acceleratedInBlock[acc.txid] = acc; | ||||||
|  |       } | ||||||
|  |       for (const tx of transactions) { | ||||||
|  |         if (acceleratedInBlock[tx.txid]) { | ||||||
|  |           tx.acc = true; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|       this.isLoadingOverview = false; |       this.isLoadingOverview = false; | ||||||
|       if (this.blockGraph) { |       if (this.blockGraph) { | ||||||
|         this.blockGraph.destroy(); |         this.blockGraph.destroy(); | ||||||
|  | |||||||
| @ -328,17 +328,28 @@ export class BlockComponent implements OnInit, OnDestroy { | |||||||
|                 this.overviewError = err; |                 this.overviewError = err; | ||||||
|                 return of(null); |                 return of(null); | ||||||
|               }) |               }) | ||||||
|             ) |             ), | ||||||
|  |           this.stateService.env.ACCELERATOR === true && block.height > 819500 ? this.apiService.getAccelerationHistory$({ blockHash: block.id }) : of([]) | ||||||
|         ]); |         ]); | ||||||
|       }) |       }) | ||||||
|     ) |     ) | ||||||
|     .subscribe(([transactions, blockAudit]) => { |     .subscribe(([transactions, blockAudit, accelerations]) => { | ||||||
|       if (transactions) { |       if (transactions) { | ||||||
|         this.strippedTransactions = transactions; |         this.strippedTransactions = transactions; | ||||||
|       } else { |       } else { | ||||||
|         this.strippedTransactions = []; |         this.strippedTransactions = []; | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|  |       const acceleratedInBlock = {}; | ||||||
|  |       for (const acc of accelerations) { | ||||||
|  |         acceleratedInBlock[acc.txid] = acc; | ||||||
|  |       } | ||||||
|  |       for (const tx of transactions) { | ||||||
|  |         if (acceleratedInBlock[tx.txid]) { | ||||||
|  |           tx.acc = true; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|       this.blockAudit = null; |       this.blockAudit = null; | ||||||
|       if (transactions && blockAudit) { |       if (transactions && blockAudit) { | ||||||
|         const inTemplate = {}; |         const inTemplate = {}; | ||||||
|  | |||||||
| @ -3,8 +3,8 @@ import { Component, ComponentRef, ViewChild, HostListener, Input, Output, EventE | |||||||
| import { StateService } from '../../services/state.service'; | import { StateService } from '../../services/state.service'; | ||||||
| import { MempoolBlockDelta, TransactionStripped } from '../../interfaces/websocket.interface'; | import { MempoolBlockDelta, TransactionStripped } from '../../interfaces/websocket.interface'; | ||||||
| import { BlockOverviewGraphComponent } from '../../components/block-overview-graph/block-overview-graph.component'; | import { BlockOverviewGraphComponent } from '../../components/block-overview-graph/block-overview-graph.component'; | ||||||
| import { Subscription, BehaviorSubject, merge, of } from 'rxjs'; | import { Subscription, BehaviorSubject, merge, of, timer } from 'rxjs'; | ||||||
| import { switchMap, filter } from 'rxjs/operators'; | import { switchMap, filter, concatMap, map } from 'rxjs/operators'; | ||||||
| import { WebsocketService } from '../../services/websocket.service'; | import { WebsocketService } from '../../services/websocket.service'; | ||||||
| import { RelativeUrlPipe } from '../../shared/pipes/relative-url/relative-url.pipe'; | import { RelativeUrlPipe } from '../../shared/pipes/relative-url/relative-url.pipe'; | ||||||
| import { Router } from '@angular/router'; | import { Router } from '@angular/router'; | ||||||
| @ -33,7 +33,11 @@ export class MempoolBlockOverviewComponent implements OnInit, OnDestroy, OnChang | |||||||
|   poolDirection: string = 'left'; |   poolDirection: string = 'left'; | ||||||
| 
 | 
 | ||||||
|   blockSub: Subscription; |   blockSub: Subscription; | ||||||
|   deltaSub: Subscription; |   rateLimit = 1000; | ||||||
|  |   private lastEventTime = Date.now() - this.rateLimit; | ||||||
|  |   private subId = 0; | ||||||
|  | 
 | ||||||
|  |   firstLoad: boolean = true; | ||||||
| 
 | 
 | ||||||
|   constructor( |   constructor( | ||||||
|     public stateService: StateService, |     public stateService: StateService, | ||||||
| @ -53,20 +57,81 @@ export class MempoolBlockOverviewComponent implements OnInit, OnDestroy, OnChang | |||||||
| 
 | 
 | ||||||
|   ngAfterViewInit(): void { |   ngAfterViewInit(): void { | ||||||
|     this.blockSub = merge( |     this.blockSub = merge( | ||||||
|         of(true), |       this.stateService.mempoolBlockTransactions$, | ||||||
|         this.stateService.connectionState$.pipe(filter((state) => state === 2)) |       this.stateService.mempoolBlockDelta$, | ||||||
|       ) |     ).pipe( | ||||||
|       .pipe(switchMap(() => this.stateService.mempoolBlockTransactions$)) |       concatMap(update => { | ||||||
|       .subscribe((transactionsStripped) => { |         const now = Date.now(); | ||||||
|         this.replaceBlock(transactionsStripped); |         const timeSinceLastEvent = now - this.lastEventTime; | ||||||
|       }); |         this.lastEventTime = Math.max(now, this.lastEventTime + this.rateLimit); | ||||||
|     this.deltaSub = this.stateService.mempoolBlockDelta$.subscribe((delta) => { | 
 | ||||||
|       this.updateBlock(delta); |         const subId = this.subId; | ||||||
|  | 
 | ||||||
|  |         // If time since last event is less than X seconds, delay this event
 | ||||||
|  |         if (timeSinceLastEvent < this.rateLimit) { | ||||||
|  |           return timer(this.rateLimit - timeSinceLastEvent).pipe( | ||||||
|  |             // Emit the event after the timer
 | ||||||
|  |             map(() => ({ update, subId })) | ||||||
|  |           ); | ||||||
|  |         } else { | ||||||
|  |           // If enough time has passed, emit the event immediately
 | ||||||
|  |           return of({ update, subId }); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |     ).subscribe(({ update, subId }) => { | ||||||
|  |       // discard stale updates after a block transition
 | ||||||
|  |       if (subId !== this.subId) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       // process update
 | ||||||
|  |       if (update['added']) { | ||||||
|  |         // delta
 | ||||||
|  |         this.updateBlock(update as MempoolBlockDelta); | ||||||
|  |       } else { | ||||||
|  |         const transactionsStripped = update as TransactionStripped[]; | ||||||
|  |         // new transactions
 | ||||||
|  |         if (this.firstLoad) { | ||||||
|  |           this.replaceBlock(transactionsStripped); | ||||||
|  |         } else { | ||||||
|  |           const inOldBlock = {}; | ||||||
|  |           const inNewBlock = {}; | ||||||
|  |           const added: TransactionStripped[] = []; | ||||||
|  |           const changed: { txid: string, rate: number | undefined, acc: boolean | undefined }[] = []; | ||||||
|  |           const removed: string[] = []; | ||||||
|  |           for (const tx of transactionsStripped) { | ||||||
|  |             inNewBlock[tx.txid] = true; | ||||||
|  |           } | ||||||
|  |           for (const txid of Object.keys(this.blockGraph?.scene?.txs || {})) { | ||||||
|  |             inOldBlock[txid] = true; | ||||||
|  |             if (!inNewBlock[txid]) { | ||||||
|  |               removed.push(txid); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |           for (const tx of transactionsStripped) { | ||||||
|  |             if (!inOldBlock[tx.txid]) { | ||||||
|  |               added.push(tx); | ||||||
|  |             } else { | ||||||
|  |               changed.push({ | ||||||
|  |                 txid: tx.txid, | ||||||
|  |                 rate: tx.rate, | ||||||
|  |                 acc: tx.acc | ||||||
|  |               }); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |           this.updateBlock({ | ||||||
|  |             removed, | ||||||
|  |             changed, | ||||||
|  |             added | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   ngOnChanges(changes): void { |   ngOnChanges(changes): void { | ||||||
|     if (changes.index) { |     if (changes.index) { | ||||||
|  |       this.subId++; | ||||||
|  |       this.firstLoad = true; | ||||||
|       if (this.blockGraph) { |       if (this.blockGraph) { | ||||||
|         this.blockGraph.clear(changes.index.currentValue > changes.index.previousValue ? this.chainDirection : this.poolDirection); |         this.blockGraph.clear(changes.index.currentValue > changes.index.previousValue ? this.chainDirection : this.poolDirection); | ||||||
|       } |       } | ||||||
| @ -77,7 +142,6 @@ export class MempoolBlockOverviewComponent implements OnInit, OnDestroy, OnChang | |||||||
| 
 | 
 | ||||||
|   ngOnDestroy(): void { |   ngOnDestroy(): void { | ||||||
|     this.blockSub.unsubscribe(); |     this.blockSub.unsubscribe(); | ||||||
|     this.deltaSub.unsubscribe(); |  | ||||||
|     this.timeLtrSubscription.unsubscribe(); |     this.timeLtrSubscription.unsubscribe(); | ||||||
|     this.websocketService.stopTrackMempoolBlock(); |     this.websocketService.stopTrackMempoolBlock(); | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -75,6 +75,7 @@ export class MempoolBlockComponent implements OnInit, OnDestroy { | |||||||
| 
 | 
 | ||||||
|   ngOnDestroy(): void { |   ngOnDestroy(): void { | ||||||
|     this.stateService.markBlock$.next({}); |     this.stateService.markBlock$.next({}); | ||||||
|  |     this.websocketService.stopTrackMempoolBlock(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   getOrdinal(mempoolBlock: MempoolBlock): string { |   getOrdinal(mempoolBlock: MempoolBlock): string { | ||||||
|  | |||||||
| @ -62,6 +62,7 @@ | |||||||
|                 <tr><td>mempool.space</td></tr> |                 <tr><td>mempool.space</td></tr> | ||||||
|                 <tr><td>Be your own explorer</td></tr> |                 <tr><td>Be your own explorer</td></tr> | ||||||
|                 <tr><td>Explore the full Bitcoin ecosystem</td></tr> |                 <tr><td>Explore the full Bitcoin ecosystem</td></tr> | ||||||
|  |                 <tr><td>Mempool Goggles</td></tr> | ||||||
|               </tbody> |               </tbody> | ||||||
|             </table> |             </table> | ||||||
|           </div> |           </div> | ||||||
| @ -314,7 +315,7 @@ | |||||||
| 
 | 
 | ||||||
|           <p>Also, if you are using our Marks in a way described in the sections "Uses for Which We Are Granting a License," you must include the following trademark attribution at the foot of the webpage where you have used the Mark (or, if in a book, on the credits page), on any packaging or labeling, and on advertising or marketing materials:</p> |           <p>Also, if you are using our Marks in a way described in the sections "Uses for Which We Are Granting a License," you must include the following trademark attribution at the foot of the webpage where you have used the Mark (or, if in a book, on the credits page), on any packaging or labeling, and on advertising or marketing materials:</p> | ||||||
| 
 | 
 | ||||||
|           <p>"The Mempool Open Source Project®, Mempool Accelerator™, Mempool Enterprise®, Mempool Liquidity™, mempool.space®, Be your own explorer™, Explore the full Bitcoin ecosystem™, the mempool logo;, the mempool Square logo;, the mempool Blocks logo;, the mempool Blocks 3 | 2 logo;, the mempool.space Vertical Logo;, and the mempool.space Horizontal logo are either registered trademarks or trademarks of Mempool Space K.K in Japan, the United States, and/or other countries, and are used with permission. Mempool Space K.K. has no affiliation with and does not sponsor or endorse the information provided herein."</p>   |           <p>"The Mempool Open Source Project®, Mempool Accelerator™, Mempool Enterprise®, Mempool Liquidity™, mempool.space®, Be your own explorer™, Explore the full Bitcoin ecosystem™, Mempool Goggles™, the mempool logo;, the mempool Square logo;, the mempool Blocks logo;, the mempool Blocks 3 | 2 logo;, the mempool.space Vertical Logo;, and the mempool.space Horizontal logo are either registered trademarks or trademarks of Mempool Space K.K in Japan, the United States, and/or other countries, and are used with permission. Mempool Space K.K. has no affiliation with and does not sponsor or endorse the information provided herein."</p>   | ||||||
|           <li>What to Do When You See Abuse</li> |           <li>What to Do When You See Abuse</li> | ||||||
| 
 | 
 | ||||||
|           <br> |           <br> | ||||||
|  | |||||||
| @ -242,6 +242,7 @@ export class TransactionComponent implements OnInit, AfterViewInit, OnDestroy { | |||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     this.fetchAccelerationSubscription = this.fetchAcceleration$.pipe( |     this.fetchAccelerationSubscription = this.fetchAcceleration$.pipe( | ||||||
|  |       filter(() => this.stateService.env.ACCELERATOR === true), | ||||||
|       tap(() => { |       tap(() => { | ||||||
|         this.accelerationInfo = null; |         this.accelerationInfo = null; | ||||||
|       }), |       }), | ||||||
| @ -439,7 +440,11 @@ export class TransactionComponent implements OnInit, AfterViewInit, OnDestroy { | |||||||
|           block_time: block.timestamp, |           block_time: block.timestamp, | ||||||
|         }; |         }; | ||||||
|         this.stateService.markBlock$.next({ blockHeight: block.height }); |         this.stateService.markBlock$.next({ blockHeight: block.height }); | ||||||
|         this.audioService.playSound('magic'); |         if (this.tx.acceleration || (this.accelerationInfo && ['accelerating', 'mined', 'completed'].includes(this.accelerationInfo.status))) { | ||||||
|  |           this.audioService.playSound('wind-chimes-harp-ascend'); | ||||||
|  |         } else { | ||||||
|  |           this.audioService.playSound('magic'); | ||||||
|  |         } | ||||||
|         this.fetchAcceleration$.next(block.id); |         this.fetchAcceleration$.next(block.id); | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  | |||||||
| @ -157,7 +157,7 @@ ul.no-bull.block-audit code{ | |||||||
|   position: fixed; |   position: fixed; | ||||||
|   top: 80px; |   top: 80px; | ||||||
|   overflow-y: auto; |   overflow-y: auto; | ||||||
|   height: calc(100vh - 50px); |   height: calc(100vh - 75px); | ||||||
|   scrollbar-color: #2d3348 #11131f; |   scrollbar-color: #2d3348 #11131f; | ||||||
|   scrollbar-width: thin; |   scrollbar-width: thin; | ||||||
| } | } | ||||||
|  | |||||||
| @ -10,6 +10,7 @@ import { PushTransactionComponent } from '../components/push-transaction/push-tr | |||||||
| import { BlocksList } from '../components/blocks-list/blocks-list.component'; | import { BlocksList } from '../components/blocks-list/blocks-list.component'; | ||||||
| import { AssetGroupComponent } from '../components/assets/asset-group/asset-group.component'; | import { AssetGroupComponent } from '../components/assets/asset-group/asset-group.component'; | ||||||
| import { AssetsComponent } from '../components/assets/assets.component'; | import { AssetsComponent } from '../components/assets/assets.component'; | ||||||
|  | import { AssetsFeaturedComponent } from '../components/assets/assets-featured/assets-featured.component' | ||||||
| import { AssetComponent } from '../components/asset/asset.component'; | import { AssetComponent } from '../components/asset/asset.component'; | ||||||
| import { AssetsNavComponent } from '../components/assets/assets-nav/assets-nav.component'; | import { AssetsNavComponent } from '../components/assets/assets-nav/assets-nav.component'; | ||||||
| 
 | 
 | ||||||
| @ -73,6 +74,11 @@ const routes: Routes = [ | |||||||
|             data: { networks: ['liquid'] }, |             data: { networks: ['liquid'] }, | ||||||
|             component: AssetsComponent, |             component: AssetsComponent, | ||||||
|           }, |           }, | ||||||
|  |           { | ||||||
|  |             path: 'featured', | ||||||
|  |             data: { networks: ['liquid'] }, | ||||||
|  |             component: AssetsFeaturedComponent, | ||||||
|  |           }, | ||||||
|           { |           { | ||||||
|             path: 'asset/:id', |             path: 'asset/:id', | ||||||
|             data: { networkSpecific: true }, |             data: { networkSpecific: true }, | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ export class AudioService { | |||||||
|     } catch (e) {} |     } catch (e) {} | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   public playSound(name: 'magic' | 'chime' | 'cha-ching' | 'bright-harmony') { |   public playSound(name: 'magic' | 'chime' | 'cha-ching' | 'bright-harmony' | 'wind-chimes-harp-ascend' | 'ascend-chime-cartoon') { | ||||||
|     if (this.isPlaying || !this.audio) { |     if (this.isPlaying || !this.audio) { | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								frontend/src/resources/sounds/ascend-chime-cartoon.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/src/resources/sounds/ascend-chime-cartoon.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								frontend/src/resources/sounds/wind-chimes-harp-ascend.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								frontend/src/resources/sounds/wind-chimes-harp-ascend.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @ -106,7 +106,12 @@ | |||||||
|       "node206.tk7.mempool.space" |       "node206.tk7.mempool.space" | ||||||
|     ] |     ] | ||||||
|   }, |   }, | ||||||
|     "MEMPOOL_SERVICES": { |   "REDIS": { | ||||||
|  |     "ENABLED": true, | ||||||
|  |     "UNIX_SOCKET_PATH": "/tmp/redis.sock", | ||||||
|  |     "BATCH_QUERY_BASE_SIZE": 5000 | ||||||
|  |   }, | ||||||
|  |   "MEMPOOL_SERVICES": { | ||||||
|     "API": "https://mempool.space/api/v1/services", |     "API": "https://mempool.space/api/v1/services", | ||||||
|     "ACCELERATIONS": true |     "ACCELERATIONS": true | ||||||
|   } |   } | ||||||
|  | |||||||
							
								
								
									
										79
									
								
								production/redis.conf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								production/redis.conf
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,79 @@ | |||||||
|  | save 3600 1 300 100 60 10000 | ||||||
|  | unixsocket /tmp/redis.sock | ||||||
|  | unixsocketperm 666 | ||||||
|  | 
 | ||||||
|  | bind 127.0.0.1 -::1 | ||||||
|  | protected-mode yes | ||||||
|  | port 6379 | ||||||
|  | tcp-backlog 511 | ||||||
|  | timeout 0 | ||||||
|  | tcp-keepalive 300 | ||||||
|  | daemonize yes | ||||||
|  | pidfile /var/run/redis/redis.pid | ||||||
|  | loglevel notice | ||||||
|  | logfile /var/log/redis/redis.log | ||||||
|  | databases 16 | ||||||
|  | always-show-logo no | ||||||
|  | set-proc-title yes | ||||||
|  | proc-title-template "{title} {listen-addr} {server-mode}" | ||||||
|  | locale-collate "" | ||||||
|  | stop-writes-on-bgsave-error yes | ||||||
|  | rdbcompression yes | ||||||
|  | rdbchecksum yes | ||||||
|  | dbfilename dump.rdb | ||||||
|  | rdb-del-sync-files no | ||||||
|  | dir /var/db/redis/ | ||||||
|  | replica-serve-stale-data yes | ||||||
|  | replica-read-only yes | ||||||
|  | repl-diskless-sync yes | ||||||
|  | repl-diskless-sync-delay 5 | ||||||
|  | repl-diskless-sync-max-replicas 0 | ||||||
|  | repl-diskless-load disabled | ||||||
|  | repl-disable-tcp-nodelay no | ||||||
|  | replica-priority 100 | ||||||
|  | acllog-max-len 128 | ||||||
|  | lazyfree-lazy-eviction no | ||||||
|  | lazyfree-lazy-expire no | ||||||
|  | lazyfree-lazy-server-del no | ||||||
|  | replica-lazy-flush no | ||||||
|  | lazyfree-lazy-user-del no | ||||||
|  | lazyfree-lazy-user-flush no | ||||||
|  | oom-score-adj no | ||||||
|  | oom-score-adj-values 0 200 800 | ||||||
|  | disable-thp yes | ||||||
|  | appendonly no | ||||||
|  | appendfilename "appendonly.aof" | ||||||
|  | appenddirname "appendonlydir" | ||||||
|  | appendfsync everysec | ||||||
|  | no-appendfsync-on-rewrite no | ||||||
|  | auto-aof-rewrite-percentage 100 | ||||||
|  | auto-aof-rewrite-min-size 64mb | ||||||
|  | aof-load-truncated yes | ||||||
|  | aof-use-rdb-preamble yes | ||||||
|  | aof-timestamp-enabled no | ||||||
|  | 
 | ||||||
|  | slowlog-log-slower-than 10000 | ||||||
|  | slowlog-max-len 128 | ||||||
|  | latency-monitor-threshold 0 | ||||||
|  | notify-keyspace-events "" | ||||||
|  | hash-max-listpack-entries 512 | ||||||
|  | hash-max-listpack-value 64 | ||||||
|  | list-max-listpack-size -2 | ||||||
|  | list-compress-depth 0 | ||||||
|  | set-max-intset-entries 512 | ||||||
|  | set-max-listpack-entries 128 | ||||||
|  | set-max-listpack-value 64 | ||||||
|  | zset-max-listpack-entries 128 | ||||||
|  | zset-max-listpack-value 64 | ||||||
|  | hll-sparse-max-bytes 3000 | ||||||
|  | stream-node-max-bytes 4096 | ||||||
|  | stream-node-max-entries 100 | ||||||
|  | activerehashing yes | ||||||
|  | client-output-buffer-limit normal 0 0 0 | ||||||
|  | client-output-buffer-limit replica 256mb 64mb 60 | ||||||
|  | client-output-buffer-limit pubsub 32mb 8mb 60 | ||||||
|  | hz 10 | ||||||
|  | dynamic-hz yes | ||||||
|  | aof-rewrite-incremental-fsync yes | ||||||
|  | rdb-save-incremental-fsync yes | ||||||
|  | jemalloc-bg-thread yes | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user