2023-07-18 09:37:25 +00:00
|
|
|
import { act, render, screen, within, waitFor } from '@testing-library/react';
|
|
|
|
import { MemoryRouter } from 'react-router-dom';
|
2023-05-02 17:41:21 +00:00
|
|
|
import { Closed } from './closed';
|
2023-05-19 20:29:24 +00:00
|
|
|
import { MarketStateMapping, PropertyKeyType } from '@vegaprotocol/types';
|
2023-05-02 17:41:21 +00:00
|
|
|
import { MarketState } from '@vegaprotocol/types';
|
|
|
|
import { subDays } from 'date-fns';
|
|
|
|
import type { MockedResponse } from '@apollo/client/testing';
|
|
|
|
import { MockedProvider } from '@apollo/client/testing';
|
2023-05-18 11:22:54 +00:00
|
|
|
import type {
|
|
|
|
OracleSpecDataConnectionQuery,
|
|
|
|
MarketsDataQuery,
|
|
|
|
MarketsQuery,
|
2023-07-25 15:32:40 +00:00
|
|
|
SuccessorMarketIdsQuery,
|
2023-05-18 11:22:54 +00:00
|
|
|
} from '@vegaprotocol/markets';
|
|
|
|
import {
|
|
|
|
OracleSpecDataConnectionDocument,
|
|
|
|
MarketsDataDocument,
|
|
|
|
MarketsDocument,
|
2023-07-25 15:32:40 +00:00
|
|
|
SuccessorMarketIdsDocument,
|
2023-05-18 11:22:54 +00:00
|
|
|
} from '@vegaprotocol/markets';
|
2023-05-02 17:41:21 +00:00
|
|
|
import type { VegaWalletContextShape } from '@vegaprotocol/wallet';
|
|
|
|
import { VegaWalletContext } from '@vegaprotocol/wallet';
|
|
|
|
import { addDecimalsFormatNumber } from '@vegaprotocol/utils';
|
2023-07-25 15:32:40 +00:00
|
|
|
import { FLAGS } from '@vegaprotocol/environment';
|
2023-05-02 17:41:21 +00:00
|
|
|
import {
|
|
|
|
createMarketFragment,
|
|
|
|
marketsQuery,
|
|
|
|
marketsDataQuery,
|
|
|
|
createMarketsDataFragment,
|
|
|
|
} from '@vegaprotocol/mock';
|
2023-07-25 09:12:53 +00:00
|
|
|
import type { FeatureFlags } from '@vegaprotocol/environment';
|
|
|
|
|
|
|
|
jest.mock('@vegaprotocol/markets', () => ({
|
|
|
|
...jest.requireActual('@vegaprotocol/markets'),
|
|
|
|
useSuccessorMarket: (marketId: string) =>
|
|
|
|
marketId === 'include-0'
|
|
|
|
? {
|
|
|
|
data: {
|
|
|
|
id: 'successorMarketID',
|
|
|
|
state: 'STATE_ACTIVE',
|
|
|
|
tradableInstrument: {
|
|
|
|
instrument: {
|
|
|
|
name: 'Successor Market Name',
|
|
|
|
code: 'SuccessorCode',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
: { data: undefined },
|
|
|
|
}));
|
|
|
|
|
2023-07-25 15:32:40 +00:00
|
|
|
jest.mock('@vegaprotocol/environment', () => {
|
|
|
|
const actual = jest.requireActual('@vegaprotocol/environment');
|
|
|
|
return {
|
|
|
|
...actual,
|
|
|
|
FLAGS: {
|
|
|
|
...actual.FLAGS,
|
|
|
|
SUCCESSOR_MARKETS: true,
|
2023-07-26 12:52:12 +00:00
|
|
|
} as FeatureFlags,
|
2023-07-25 15:32:40 +00:00
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2023-05-02 17:41:21 +00:00
|
|
|
describe('Closed', () => {
|
|
|
|
let originalNow: typeof Date.now;
|
|
|
|
const mockNowTimestamp = 1672531200000;
|
|
|
|
const settlementDateMetaDate = subDays(
|
|
|
|
new Date(mockNowTimestamp),
|
|
|
|
3
|
|
|
|
).toISOString();
|
|
|
|
const settlementDateTag = `settlement-expiry-date:${settlementDateMetaDate}`;
|
|
|
|
const pubKey = 'pubKey';
|
|
|
|
const marketId = 'market-0';
|
|
|
|
const settlementDataProperty = 'spec-binding';
|
|
|
|
const settlementDataId = 'settlement-data-oracle-id';
|
|
|
|
|
|
|
|
const market = createMarketFragment({
|
|
|
|
id: marketId,
|
|
|
|
state: MarketState.STATE_SETTLED,
|
|
|
|
tradableInstrument: {
|
|
|
|
instrument: {
|
|
|
|
metadata: {
|
|
|
|
tags: [settlementDateTag],
|
|
|
|
},
|
|
|
|
product: {
|
|
|
|
dataSourceSpecForSettlementData: {
|
|
|
|
id: settlementDataId,
|
2023-05-19 20:29:24 +00:00
|
|
|
data: {
|
|
|
|
sourceType: {
|
|
|
|
sourceType: {
|
|
|
|
filters: [
|
|
|
|
{
|
|
|
|
__typename: 'Filter',
|
|
|
|
key: {
|
|
|
|
__typename: 'PropertyKey',
|
|
|
|
name: settlementDataProperty,
|
|
|
|
type: PropertyKeyType.TYPE_INTEGER,
|
|
|
|
numberDecimalPlaces: 5,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2023-05-02 17:41:21 +00:00
|
|
|
},
|
|
|
|
dataSourceSpecBinding: {
|
|
|
|
settlementDataProperty,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
const marketsMock: MockedResponse<MarketsQuery> = {
|
|
|
|
request: {
|
|
|
|
query: MarketsDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: marketsQuery({
|
|
|
|
marketsConnection: {
|
|
|
|
edges: [
|
|
|
|
{
|
|
|
|
node: market,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
const marketsData = createMarketsDataFragment({
|
|
|
|
__typename: 'MarketData',
|
|
|
|
market: {
|
|
|
|
__typename: 'Market',
|
|
|
|
id: marketId,
|
|
|
|
},
|
|
|
|
bestBidPrice: '1000',
|
|
|
|
bestOfferPrice: '2000',
|
|
|
|
markPrice: '1500',
|
|
|
|
});
|
|
|
|
const marketsDataMock: MockedResponse<MarketsDataQuery> = {
|
|
|
|
request: {
|
|
|
|
query: MarketsDataDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: marketsDataQuery({
|
|
|
|
marketsConnection: {
|
|
|
|
edges: [
|
|
|
|
{
|
|
|
|
node: {
|
|
|
|
data: marketsData,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
// Create mock oracle data
|
|
|
|
const property = {
|
|
|
|
__typename: 'Property' as const,
|
|
|
|
name: settlementDataProperty,
|
|
|
|
value: '12345',
|
|
|
|
};
|
|
|
|
const oracleDataMock: MockedResponse<OracleSpecDataConnectionQuery> = {
|
|
|
|
request: {
|
|
|
|
query: OracleSpecDataConnectionDocument,
|
|
|
|
variables: {
|
|
|
|
oracleSpecId: settlementDataId,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
oracleSpec: {
|
|
|
|
dataConnection: {
|
|
|
|
edges: [
|
|
|
|
{
|
|
|
|
node: {
|
|
|
|
externalData: {
|
|
|
|
data: {
|
|
|
|
data: [property],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
originalNow = Date.now;
|
|
|
|
Date.now = jest.fn().mockReturnValue(mockNowTimestamp);
|
|
|
|
});
|
|
|
|
|
|
|
|
afterAll(() => {
|
|
|
|
Date.now = originalNow;
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders correctly formatted and filtered rows', async () => {
|
|
|
|
await act(async () => {
|
|
|
|
render(
|
2023-07-18 09:37:25 +00:00
|
|
|
<MemoryRouter>
|
|
|
|
<MockedProvider
|
2023-07-20 11:52:13 +00:00
|
|
|
mocks={[marketsMock, marketsDataMock, oracleDataMock]}
|
2023-05-02 17:41:21 +00:00
|
|
|
>
|
2023-07-18 09:37:25 +00:00
|
|
|
<VegaWalletContext.Provider
|
|
|
|
value={{ pubKey } as VegaWalletContextShape}
|
|
|
|
>
|
|
|
|
<Closed />
|
|
|
|
</VegaWalletContext.Provider>
|
|
|
|
</MockedProvider>
|
|
|
|
</MemoryRouter>
|
2023-05-02 17:41:21 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
// screen.debug(document, Infinity);
|
|
|
|
|
|
|
|
const headers = screen.getAllByRole('columnheader');
|
|
|
|
const expectedHeaders = [
|
|
|
|
'Market',
|
|
|
|
'Description',
|
|
|
|
'Status',
|
|
|
|
'Settlement date',
|
2023-07-18 09:37:25 +00:00
|
|
|
'Successor market',
|
2023-05-02 17:41:21 +00:00
|
|
|
'Best bid',
|
|
|
|
'Best offer',
|
|
|
|
'Mark price',
|
|
|
|
'Settlement price',
|
|
|
|
'Settlement asset',
|
2023-05-18 04:05:53 +00:00
|
|
|
'', // actions row
|
2023-05-02 17:41:21 +00:00
|
|
|
];
|
|
|
|
expect(headers).toHaveLength(expectedHeaders.length);
|
|
|
|
expect(headers.map((h) => h.textContent?.trim())).toEqual(expectedHeaders);
|
|
|
|
|
|
|
|
const cells = screen.getAllByRole('gridcell');
|
|
|
|
const expectedValues = [
|
|
|
|
market.tradableInstrument.instrument.code,
|
|
|
|
market.tradableInstrument.instrument.name,
|
|
|
|
MarketStateMapping[market.state],
|
|
|
|
'3 days ago',
|
2023-07-18 09:37:25 +00:00
|
|
|
'-',
|
2023-05-02 17:41:21 +00:00
|
|
|
/* eslint-disable @typescript-eslint/no-non-null-assertion */
|
|
|
|
addDecimalsFormatNumber(marketsData.bestBidPrice, market.decimalPlaces),
|
|
|
|
addDecimalsFormatNumber(
|
|
|
|
marketsData!.bestOfferPrice,
|
|
|
|
market.decimalPlaces
|
|
|
|
),
|
|
|
|
addDecimalsFormatNumber(marketsData!.markPrice, market.decimalPlaces),
|
|
|
|
/* eslint-enable @typescript-eslint/no-non-null-assertion */
|
|
|
|
addDecimalsFormatNumber(property.value, market.decimalPlaces),
|
|
|
|
market.tradableInstrument.instrument.product.settlementAsset.symbol,
|
2023-05-18 04:05:53 +00:00
|
|
|
'', // actions row
|
2023-05-02 17:41:21 +00:00
|
|
|
];
|
|
|
|
cells.forEach((cell, i) => {
|
|
|
|
expect(cell).toHaveTextContent(expectedValues[i]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('only renders settled and terminated markets', async () => {
|
|
|
|
const mixedMarkets = [
|
|
|
|
{
|
|
|
|
// inlclude as settled
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: createMarketFragment({
|
|
|
|
id: 'include-0',
|
|
|
|
state: MarketState.STATE_SETTLED,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// omit this market
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: createMarketFragment({
|
|
|
|
id: 'discard-0',
|
|
|
|
state: MarketState.STATE_SUSPENDED,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// include as terminated
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: createMarketFragment({
|
|
|
|
id: 'include-1',
|
|
|
|
state: MarketState.STATE_TRADING_TERMINATED,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// omit this market
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: createMarketFragment({
|
|
|
|
id: 'discard-1',
|
|
|
|
state: MarketState.STATE_ACTIVE,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
];
|
|
|
|
const mixedMarketsMock: MockedResponse<MarketsQuery> = {
|
|
|
|
request: {
|
|
|
|
query: MarketsDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
marketsConnection: {
|
|
|
|
__typename: 'MarketConnection',
|
|
|
|
edges: mixedMarkets,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
await act(async () => {
|
|
|
|
render(
|
2023-07-18 09:37:25 +00:00
|
|
|
<MemoryRouter>
|
|
|
|
<MockedProvider
|
2023-07-20 11:52:13 +00:00
|
|
|
mocks={[mixedMarketsMock, marketsDataMock, oracleDataMock]}
|
2023-05-02 17:41:21 +00:00
|
|
|
>
|
2023-07-18 09:37:25 +00:00
|
|
|
<VegaWalletContext.Provider
|
|
|
|
value={{ pubKey } as VegaWalletContextShape}
|
|
|
|
>
|
|
|
|
<Closed />
|
|
|
|
</VegaWalletContext.Provider>
|
|
|
|
</MockedProvider>
|
|
|
|
</MemoryRouter>
|
2023-05-02 17:41:21 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
// check that the number of rows in datagrid is 2
|
|
|
|
const container = within(
|
|
|
|
document.querySelector('.ag-center-cols-container') as HTMLElement
|
|
|
|
);
|
|
|
|
const expectedRows = mixedMarkets.filter((m) => {
|
|
|
|
return [
|
|
|
|
MarketState.STATE_SETTLED,
|
|
|
|
MarketState.STATE_TRADING_TERMINATED,
|
|
|
|
].includes(m.node.state);
|
|
|
|
});
|
|
|
|
|
|
|
|
// check rows length is correct
|
|
|
|
const rows = container.getAllByRole('row');
|
|
|
|
expect(rows).toHaveLength(expectedRows.length);
|
|
|
|
|
|
|
|
// check that only included ids are shown
|
|
|
|
const cells = screen
|
|
|
|
.getAllByRole('gridcell')
|
2023-05-18 04:05:53 +00:00
|
|
|
.filter((cell) => cell.getAttribute('col-id') === 'code')
|
|
|
|
.map((cell) => {
|
|
|
|
const marketId = within(cell)
|
|
|
|
.getByTestId('market-code')
|
|
|
|
.getAttribute('data-market-id');
|
|
|
|
return marketId;
|
|
|
|
});
|
2023-05-02 17:41:21 +00:00
|
|
|
expect(cells).toEqual(expectedRows.map((m) => m.node.id));
|
|
|
|
});
|
2023-07-18 09:37:25 +00:00
|
|
|
|
|
|
|
it('successor marked should be visible', async () => {
|
|
|
|
const mixedMarkets = [
|
|
|
|
{
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: createMarketFragment({
|
|
|
|
id: 'include-0',
|
|
|
|
state: MarketState.STATE_SETTLED,
|
|
|
|
}),
|
|
|
|
},
|
2023-07-25 15:32:40 +00:00
|
|
|
{
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: {
|
|
|
|
...createMarketFragment({
|
|
|
|
id: 'successorMarketID',
|
|
|
|
state: MarketState.STATE_ACTIVE,
|
|
|
|
}),
|
|
|
|
tradableInstrument: {
|
|
|
|
...createMarketFragment().tradableInstrument,
|
|
|
|
instrument: {
|
|
|
|
...createMarketFragment().tradableInstrument.instrument,
|
|
|
|
id: 'successorAssset',
|
|
|
|
name: 'Successor Market Name',
|
|
|
|
code: 'SuccessorCode',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2023-07-18 09:37:25 +00:00
|
|
|
];
|
|
|
|
const mixedMarketsMock: MockedResponse<MarketsQuery> = {
|
|
|
|
request: {
|
|
|
|
query: MarketsDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
marketsConnection: {
|
|
|
|
__typename: 'MarketConnection',
|
|
|
|
edges: mixedMarkets,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
2023-07-25 15:32:40 +00:00
|
|
|
const successorMarketsMock: MockedResponse<SuccessorMarketIdsQuery> = {
|
|
|
|
request: {
|
|
|
|
query: SuccessorMarketIdsDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
marketsConnection: {
|
|
|
|
__typename: 'MarketConnection',
|
|
|
|
edges: [
|
|
|
|
{
|
|
|
|
node: {
|
|
|
|
id: 'include-0',
|
|
|
|
successorMarketID: 'successorMarketID',
|
|
|
|
parentMarketID: '',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
2023-08-10 11:03:53 +00:00
|
|
|
await act(() => {
|
|
|
|
render(
|
|
|
|
<MemoryRouter>
|
|
|
|
<MockedProvider
|
|
|
|
mocks={[
|
|
|
|
mixedMarketsMock,
|
|
|
|
marketsDataMock,
|
|
|
|
oracleDataMock,
|
|
|
|
successorMarketsMock,
|
|
|
|
]}
|
2023-07-18 09:37:25 +00:00
|
|
|
>
|
2023-08-10 11:03:53 +00:00
|
|
|
<VegaWalletContext.Provider
|
|
|
|
value={{ pubKey } as VegaWalletContextShape}
|
|
|
|
>
|
|
|
|
<Closed />
|
|
|
|
</VegaWalletContext.Provider>
|
|
|
|
</MockedProvider>
|
|
|
|
</MemoryRouter>
|
|
|
|
);
|
|
|
|
});
|
2023-07-18 09:37:25 +00:00
|
|
|
|
|
|
|
await waitFor(() => {
|
|
|
|
expect(
|
|
|
|
screen.getByRole('button', { name: 'SuccessorCode' })
|
|
|
|
).toBeInTheDocument();
|
|
|
|
});
|
2023-07-25 15:32:40 +00:00
|
|
|
expect(
|
|
|
|
screen.getByRole('columnheader', {
|
|
|
|
name: (_name, element) =>
|
|
|
|
element.getAttribute('col-id') === 'successorMarket',
|
|
|
|
})
|
|
|
|
).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('feature flag should hide successors', async () => {
|
|
|
|
const mockedFlags = jest.mocked(FLAGS);
|
|
|
|
mockedFlags.SUCCESSOR_MARKETS = false;
|
|
|
|
|
|
|
|
const mixedMarkets = [
|
|
|
|
{
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: createMarketFragment({
|
|
|
|
id: 'include-0',
|
|
|
|
state: MarketState.STATE_SETTLED,
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
__typename: 'MarketEdge' as const,
|
|
|
|
node: {
|
|
|
|
...createMarketFragment({
|
|
|
|
id: 'successorMarketID',
|
|
|
|
state: MarketState.STATE_ACTIVE,
|
|
|
|
}),
|
|
|
|
tradableInstrument: {
|
|
|
|
...createMarketFragment().tradableInstrument,
|
|
|
|
instrument: {
|
|
|
|
...createMarketFragment().tradableInstrument.instrument,
|
|
|
|
id: 'successorAssset',
|
|
|
|
name: 'Successor Market Name',
|
|
|
|
code: 'SuccessorCode',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
];
|
|
|
|
const mixedMarketsMock: MockedResponse<MarketsQuery> = {
|
|
|
|
request: {
|
|
|
|
query: MarketsDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
marketsConnection: {
|
|
|
|
__typename: 'MarketConnection',
|
|
|
|
edges: mixedMarkets,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const successorMarketsMock: MockedResponse<SuccessorMarketIdsQuery> = {
|
|
|
|
request: {
|
|
|
|
query: SuccessorMarketIdsDocument,
|
|
|
|
},
|
|
|
|
result: {
|
|
|
|
data: {
|
|
|
|
marketsConnection: {
|
|
|
|
__typename: 'MarketConnection',
|
|
|
|
edges: [
|
|
|
|
{
|
|
|
|
node: {
|
|
|
|
id: 'include-0',
|
|
|
|
successorMarketID: 'successorMarketID',
|
|
|
|
parentMarketID: '',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
render(
|
|
|
|
<MemoryRouter>
|
|
|
|
<MockedProvider
|
|
|
|
mocks={[
|
|
|
|
mixedMarketsMock,
|
|
|
|
marketsDataMock,
|
|
|
|
oracleDataMock,
|
|
|
|
successorMarketsMock,
|
|
|
|
]}
|
|
|
|
>
|
|
|
|
<VegaWalletContext.Provider
|
|
|
|
value={{ pubKey } as VegaWalletContextShape}
|
|
|
|
>
|
|
|
|
<Closed />
|
|
|
|
</VegaWalletContext.Provider>
|
|
|
|
</MockedProvider>
|
|
|
|
</MemoryRouter>
|
|
|
|
);
|
|
|
|
await waitFor(() => {
|
|
|
|
expect(
|
|
|
|
screen.getByRole('columnheader', {
|
|
|
|
name: (_name, element) =>
|
|
|
|
element.getAttribute('col-id') === 'settlementDate',
|
|
|
|
})
|
|
|
|
).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
screen.getAllByRole('columnheader').forEach((element) => {
|
|
|
|
expect(element.getAttribute('col-id')).not.toEqual('successorMarket');
|
|
|
|
});
|
2023-07-18 09:37:25 +00:00
|
|
|
});
|
2023-05-02 17:41:21 +00:00
|
|
|
});
|