413 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			413 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { emitMempoolInfo, dropWebSocket } from "../../support/websocket";
 | |
| 
 | |
| const baseModule = Cypress.env("BASE_MODULE");
 | |
| 
 | |
| describe('Mainnet', () => {
 | |
|     beforeEach(() => {
 | |
|         //cy.intercept('/sockjs-node/info*').as('socket');
 | |
|         cy.intercept('/api/block-height/*').as('block-height');
 | |
|         cy.intercept('/api/block/*').as('block');
 | |
|         cy.intercept('/api/block/*/txs/0').as('block-txs');
 | |
|         cy.intercept('/api/tx/*/outspends').as('tx-outspends');
 | |
|         cy.intercept('/resources/pools.json').as('pools');
 | |
| 
 | |
|         // Search Auto Complete
 | |
|         cy.intercept('/api/address-prefix/1wiz').as('search-1wiz');
 | |
|         cy.intercept('/api/address-prefix/1wizS').as('search-1wizS');
 | |
|         cy.intercept('/api/address-prefix/1wizSA').as('search-1wizSA');
 | |
| 
 | |
|         Cypress.Commands.add('waitForBlockData', () => {
 | |
|             cy.wait('@tx-outspends');
 | |
|             cy.wait('@pools');
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     if (baseModule === 'mempool') {
 | |
| 
 | |
|         it('loads the status screen', () => {
 | |
|             cy.visit('/status');
 | |
|             cy.get('#mempool-block-0').should('be.visible');
 | |
|             cy.get('[id^="bitcoin-block-"]').should('have.length', 8);
 | |
|             cy.get('.footer').should('be.visible');
 | |
|             cy.get('.row > :nth-child(1)').invoke('text').then((text) => {
 | |
|                 expect(text).to.match(/Tx vBytes per second:.* vB\/s/);
 | |
|             });
 | |
|             cy.get('.row > :nth-child(2)').invoke('text').then((text) => {
 | |
|                 expect(text).to.match(/Unconfirmed:(.*)/);
 | |
|             });
 | |
|             cy.get('.row > :nth-child(3)').invoke('text').then((text) => {
 | |
|                 expect(text).to.match(/Mempool size:(.*) (kB|MB) \((\d+) (block|blocks)\)/);
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         it('loads dashboard, drop websocket and reconnect', () => {
 | |
|             cy.viewport('macbook-16');
 | |
|             cy.mockMempoolSocket();
 | |
|             cy.visit('/');
 | |
|             cy.get('.badge').should('not.exist');
 | |
|             dropWebSocket();
 | |
|             cy.get('.badge').should('be.visible');
 | |
|             cy.get('.badge', {timeout: 25000}).should('not.exist');
 | |
|             emitMempoolInfo({
 | |
|                 'params': {
 | |
|                 loaded: true
 | |
|                 }
 | |
|             });
 | |
|             cy.get(':nth-child(1) > #bitcoin-block-0').should('not.exist');
 | |
|             cy.get(':nth-child(2) > #bitcoin-block-0').should('not.exist');
 | |
|             cy.get(':nth-child(3) > #bitcoin-block-0').should('not.exist');
 | |
|         });
 | |
| 
 | |
|         it('loads the dashboard', () => {
 | |
|             cy.visit('/');
 | |
|             cy.waitForSkeletonGone();
 | |
|         });
 | |
| 
 | |
|         describe('search', () => {
 | |
|             it('allows searching for partial Bitcoin addresses', () => {
 | |
|                 cy.visit('/');
 | |
|                 cy.get('.search-box-container > .form-control').type('1wiz').then(() => {
 | |
|                     cy.wait('@search-1wiz');
 | |
|                     cy.get('ngb-typeahead-window button.dropdown-item').should('have.length', 10);
 | |
|                 });
 | |
| 
 | |
|                 cy.get('.search-box-container > .form-control').type('S').then(() => {
 | |
|                     cy.wait('@search-1wizS');
 | |
|                     cy.get('ngb-typeahead-window button.dropdown-item').should('have.length', 5);
 | |
|                 });
 | |
| 
 | |
|                 cy.get('.search-box-container > .form-control').type('A').then(() => {
 | |
|                     cy.wait('@search-1wizSA');
 | |
|                     cy.get('ngb-typeahead-window button.dropdown-item').should('have.length', 1)
 | |
|                 });
 | |
| 
 | |
|                 cy.get('ngb-typeahead-window button.dropdown-item.active').click().then(() => {
 | |
|                     cy.url().should('include', '/address/1wizSAYSbuyXbt9d8JV8ytm5acqq2TorC');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.get('.text-center').should('not.have.text', 'Invalid Bitcoin address');
 | |
|                 });
 | |
|             });
 | |
| 
 | |
|             ['BC1PQYQSZQ', 'bc1PqYqSzQ'].forEach((searchTerm) => {
 | |
|                 it(`allows searching for partial case insensitive bech32m addresses: ${searchTerm}`, () => {
 | |
|                     cy.visit('/');
 | |
|                     cy.get('.search-box-container > .form-control').type(searchTerm).then(() => {
 | |
|                         cy.get('ngb-typeahead-window button.dropdown-item').should('have.length', 1);
 | |
|                         cy.get('ngb-typeahead-window button.dropdown-item.active').click().then(() => {
 | |
|                             cy.url().should('include', '/address/bc1pqyqszqgpqyqszqgpqyqszqgpqyqszqgpqyqszqgpqyqszqgpqyqsyjer9e');
 | |
|                             cy.waitForSkeletonGone();
 | |
|                             cy.get('.text-center').should('not.have.text', 'Invalid Bitcoin address');
 | |
|                         });
 | |
|                     });
 | |
|                 });
 | |
|             });
 | |
| 
 | |
|             ['BC1Q000375VXCU', 'bC1q000375vXcU'].forEach((searchTerm) => {
 | |
|                 it(`allows searching for partial case insensitive bech32 addresses: ${searchTerm}`, () => {
 | |
|                     cy.visit('/');
 | |
|                     cy.get('.search-box-container > .form-control').type(searchTerm).then(() => {
 | |
|                         cy.get('ngb-typeahead-window button.dropdown-item').should('have.length', 1);
 | |
|                         cy.get('ngb-typeahead-window button.dropdown-item.active').click().then(() => {
 | |
|                             cy.url().should('include', '/address/bc1q000375vxcuf5v04lmwy22vy2thvhqkxghgq7dy');
 | |
|                             cy.waitForSkeletonGone();
 | |
|                             cy.get('.text-center').should('not.have.text', 'Invalid Bitcoin address');
 | |
|                         });
 | |
|                     });
 | |
|                 });
 | |
|             });
 | |
| 
 | |
|         });
 | |
| 
 | |
|         describe('blocks navigation', () => {
 | |
| 
 | |
|             describe('keyboard events', () => {
 | |
|                 it('loads first blockchain blocks visible and keypress arrow right', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.get('.blockchain-blocks-0 > a').click().then(() => {
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('not.exist');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.waitForPageIdle();
 | |
|                         cy.document().right();
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                     });
 | |
|                 });
 | |
| 
 | |
|                 it('loads first blockchain blocks visible and keypress arrow left', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.get('.blockchain-blocks-0 > a').click().then(() => {
 | |
|                         cy.waitForPageIdle();
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('not.exist');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.document().left();
 | |
|                         cy.get('.title-block h1').invoke('text').should('equal', 'Next block');
 | |
|                     });
 | |
|                 });
 | |
| 
 | |
|                 it('loads last blockchain blocks and keypress arrow right', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.get('.blockchain-blocks-4 > a').click().then(() => {
 | |
|                         cy.waitForPageIdle();
 | |
| 
 | |
|                         // block 6
 | |
|                         cy.document().right();
 | |
|                         cy.wait(5000);
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
| 
 | |
|                         // block 7
 | |
|                         cy.document().right();
 | |
|                         cy.wait(5000);
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
| 
 | |
|                         // block 8 - last visible block
 | |
|                         cy.document().right();
 | |
|                         cy.wait(5000);
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
| 
 | |
|                         // block 9 - not visible at the blochchain blocks visible block
 | |
|                         cy.document().right();
 | |
|                         cy.wait(5000);
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
| 
 | |
|                     });
 | |
|                 });
 | |
| 
 | |
|                 it('loads genesis block and keypress arrow right', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/block/0');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.waitForPageIdle();
 | |
| 
 | |
|                     cy.document().right();
 | |
|                     cy.wait(5000);
 | |
|                     cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                     cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('not.exist');
 | |
|                 });
 | |
| 
 | |
|                 it('loads genesis block and keypress arrow left', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/block/0');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.waitForPageIdle();
 | |
| 
 | |
|                     cy.document().left();
 | |
|                     cy.wait(5000);
 | |
|                     cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                     cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                 });
 | |
|             });
 | |
|             describe('mouse events', () => {
 | |
|                 it('loads first blockchain blocks visible and click on the arrow right', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.get('.blockchain-blocks-0 > a').click().then(() => {
 | |
|                         cy.waitForPageIdle();
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('not.exist');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').click().then(() => {
 | |
|                             cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                             cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         });
 | |
|                     });
 | |
|                 });
 | |
| 
 | |
|                 it('loads genesis block and click on the arrow left', () => {
 | |
|                     cy.viewport('macbook-16');
 | |
|                     cy.visit('/block/0');
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.waitForPageIdle();
 | |
|                     cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                     cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('not.exist');
 | |
|                     cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').click().then(() => {
 | |
|                         cy.get('[ngbtooltip="Next Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                         cy.get('[ngbtooltip="Previous Block"] > .ng-fa-icon > .svg-inline--fa').should('be.visible');
 | |
|                     });
 | |
|                 });
 | |
|             });
 | |
|         });
 | |
| 
 | |
| 
 | |
|         it('loads skeleton when changes between networks', () => {
 | |
|             cy.visit('/');
 | |
|             cy.waitForSkeletonGone();
 | |
| 
 | |
|             cy.changeNetwork("testnet");
 | |
|             cy.changeNetwork("signet");
 | |
|             cy.changeNetwork("liquid");
 | |
|             cy.changeNetwork("mainnet");
 | |
|             cy.changeNetwork("bisq");
 | |
|         });
 | |
| 
 | |
|         it('loads the dashboard with the skeleton blocks', () => {
 | |
|             cy.mockMempoolSocket();
 | |
|             cy.visit("/");
 | |
|             cy.get(':nth-child(1) > #bitcoin-block-0').should('be.visible');
 | |
|             cy.get(':nth-child(2) > #bitcoin-block-0').should('be.visible');
 | |
|             cy.get(':nth-child(3) > #bitcoin-block-0').should('be.visible');
 | |
|             cy.get('#mempool-block-0').should('be.visible');
 | |
|             cy.get('#mempool-block-1').should('be.visible');
 | |
|             cy.get('#mempool-block-2').should('be.visible');
 | |
| 
 | |
|             emitMempoolInfo({
 | |
|                 'params': {
 | |
|                 loaded: true
 | |
|                 }
 | |
|             });
 | |
| 
 | |
|             cy.get(':nth-child(1) > #bitcoin-block-0').should('not.exist');
 | |
|             cy.get(':nth-child(2) > #bitcoin-block-0').should('not.exist');
 | |
|             cy.get(':nth-child(3) > #bitcoin-block-0').should('not.exist');
 | |
|         });
 | |
| 
 | |
|         it('loads the blocks screen', () => {
 | |
|             cy.visit('/');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('li:nth-of-type(2) > a').click().then(() => {
 | |
|                 cy.waitForPageIdle();
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         it('loads the graphs screen', () => {
 | |
|             cy.visit('/');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('li:nth-of-type(3) > a').click().then(() => {
 | |
|                 cy.wait(1000);
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         describe('graphs page', () => {
 | |
|           it('check buttons - mobile', () => {
 | |
|               cy.viewport('iphone-6');
 | |
|               cy.visit('/graphs');
 | |
|               cy.waitForSkeletonGone();
 | |
|               cy.get('.small-buttons > :nth-child(2)').should('be.visible');
 | |
|               cy.get('#dropdownFees').should('be.visible');
 | |
|               cy.get('.btn-group').should('be.visible');
 | |
|           });
 | |
|           it('check buttons - tablet', () => {
 | |
|             cy.viewport('ipad-2');
 | |
|             cy.visit('/graphs');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('.small-buttons > :nth-child(2)').should('be.visible');
 | |
|             cy.get('#dropdownFees').should('be.visible');
 | |
|             cy.get('.btn-group').should('be.visible');
 | |
|           });
 | |
|           it('check buttons - desktop', () => {
 | |
|             cy.viewport('macbook-16');
 | |
|             cy.visit('/graphs');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('.small-buttons > :nth-child(2)').should('be.visible');
 | |
|             cy.get('#dropdownFees').should('be.visible');
 | |
|             cy.get('.btn-group').should('be.visible');
 | |
|           });
 | |
|         });
 | |
| 
 | |
|         it('loads the tv screen - desktop', () => {
 | |
|             cy.viewport('macbook-16');
 | |
|             cy.visit('/');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('li:nth-of-type(4) > a').click().then(() => {
 | |
|                 cy.viewport('macbook-16');
 | |
|                 cy.get('.chart-holder');
 | |
|                 cy.get('.blockchain-wrapper').should('be.visible');
 | |
|                 cy.get('#mempool-block-0').should('be.visible');
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         it('loads the tv screen - mobile', () => {
 | |
|             cy.viewport('iphone-6');
 | |
|             cy.visit('/tv');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('.chart-holder');
 | |
|             cy.get('.blockchain-wrapper').should('not.visible');
 | |
|         });
 | |
| 
 | |
|         it('loads the api screen', () => {
 | |
|             cy.visit('/');
 | |
|             cy.waitForSkeletonGone();
 | |
|             cy.get('li:nth-of-type(5) > a').click().then(() => {
 | |
|                 cy.wait(1000);
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         describe('blocks', () => {
 | |
|             it('shows empty blocks properly', () => {
 | |
|                 cy.visit('/block/0000000000000000000bd14f744ef2e006e61c32214670de7eb891a5732ee775');
 | |
|                 cy.waitForSkeletonGone();
 | |
|                 cy.waitForPageIdle();
 | |
|                 cy.get('h2').invoke('text').should('equal', '1 transaction');
 | |
|             });
 | |
| 
 | |
|             it('expands and collapses the block details', () => {
 | |
|                 cy.visit('/block/0');
 | |
|                 cy.waitForSkeletonGone();
 | |
|                 cy.waitForPageIdle();
 | |
|                 cy.get('.btn.btn-outline-info').click().then(() => {
 | |
|                     cy.get('#details').should('be.visible');
 | |
|                 });
 | |
| 
 | |
|                 cy.get('.btn.btn-outline-info').click().then(() => {
 | |
|                     cy.get('#details').should('not.be.visible');
 | |
|                 });
 | |
|             });
 | |
|             it('shows blocks with no pagination', () => {
 | |
|                 cy.visit('/block/00000000000000000001ba40caf1ad4cec0ceb77692662315c151953bfd7c4c4');
 | |
|                 cy.waitForSkeletonGone();
 | |
|                 cy.waitForPageIdle();
 | |
|                 cy.get('.block-tx-title h2').invoke('text').should('equal', '19 transactions');
 | |
|                 cy.get('.pagination-container ul.pagination').first().children().should('have.length', 5);
 | |
|             });
 | |
| 
 | |
|             it('supports pagination on the block screen', () => {
 | |
|                 // 41 txs
 | |
|                 cy.visit('/block/00000000000000000009f9b7b0f63ad50053ad12ec3b7f5ca951332f134f83d8');
 | |
|                 cy.waitForSkeletonGone();
 | |
|                 cy.get('.pagination-container a').invoke('text').then((text1) => {
 | |
|                     cy.get('.active + li').first().click().then(() => {
 | |
|                         cy.waitForSkeletonGone();
 | |
|                         cy.waitForPageIdle();
 | |
|                         cy.get('.header-bg.box > a').invoke('text').then((text2) => {
 | |
|                             expect(text1).not.to.eq(text2);
 | |
|                         });
 | |
|                     });
 | |
|                 });
 | |
|             });
 | |
| 
 | |
|             it('shows blocks pagination with 5 pages (desktop)', () => {
 | |
|                 cy.viewport(760, 800);
 | |
|                 cy.visit('/block/000000000000000000049281946d26fcba7d99fdabc1feac524bc3a7003d69b3').then(() => {
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.waitForPageIdle();
 | |
|                 });
 | |
| 
 | |
|                 // 5 pages + 4 buttons = 9 buttons
 | |
|                 cy.get('.pagination-container ul.pagination').first().children().should('have.length', 9);
 | |
|             });
 | |
| 
 | |
|             it('shows blocks pagination with 3 pages (mobile)', () => {
 | |
|                 cy.viewport(669, 800);
 | |
|                 cy.visit('/block/000000000000000000049281946d26fcba7d99fdabc1feac524bc3a7003d69b3').then(() => {
 | |
|                     cy.waitForSkeletonGone();
 | |
|                     cy.waitForPageIdle();
 | |
|                 });
 | |
| 
 | |
|                 // 3 pages + 4 buttons = 7 buttons
 | |
|                 cy.get('.pagination-container ul.pagination').first().children().should('have.length', 7);
 | |
|             });
 | |
|         });
 | |
|     } else {
 | |
|         it.skip(`Tests cannot be run on the selected BASE_MODULE ${baseModule}`);
 | |
|     }
 | |
| });
 |