debian-mirror-gitlab/spec/frontend/diffs/components/diff_file_spec.js

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

557 lines
18 KiB
JavaScript
Raw Normal View History

2022-04-04 11:22:00 +05:30
import { shallowMount } from '@vue/test-utils';
2021-03-08 18:12:59 +05:30
import MockAdapter from 'axios-mock-adapter';
2022-04-04 11:22:00 +05:30
import Vue, { nextTick } from 'vue';
2021-03-11 19:13:27 +05:30
import Vuex from 'vuex';
2021-01-29 00:20:46 +05:30
2022-01-26 12:08:38 +05:30
import DiffContentComponent from 'jh_else_ce/diffs/components/diff_content.vue';
2021-01-29 00:20:46 +05:30
import DiffFileComponent from '~/diffs/components/diff_file.vue';
import DiffFileHeaderComponent from '~/diffs/components/diff_file_header.vue';
2018-11-08 19:23:39 +05:30
2021-01-29 00:20:46 +05:30
import {
EVT_EXPAND_ALL_FILES,
EVT_PERF_MARK_DIFF_FILES_END,
EVT_PERF_MARK_FIRST_DIFF_FILE_SHOWN,
} from '~/diffs/constants';
2021-03-11 19:13:27 +05:30
import eventHub from '~/diffs/event_hub';
import createDiffsStore from '~/diffs/store/modules';
2021-01-29 00:20:46 +05:30
import { diffViewerModes, diffViewerErrors } from '~/ide/constants';
2021-03-11 19:13:27 +05:30
import axios from '~/lib/utils/axios_utils';
2021-06-08 01:23:25 +05:30
import { scrollToElement } from '~/lib/utils/common_utils';
2023-03-17 16:20:25 +05:30
import { HTTP_STATUS_OK } from '~/lib/utils/http_status';
2021-03-11 19:13:27 +05:30
import createNotesStore from '~/notes/stores/modules';
2022-08-13 15:12:31 +05:30
import { getDiffFileMock } from '../mock_data/diff_file';
2021-03-11 19:13:27 +05:30
import diffFileMockDataUnreadable from '../mock_data/diff_file_unreadable';
2021-01-29 00:20:46 +05:30
2021-06-08 01:23:25 +05:30
jest.mock('~/lib/utils/common_utils');
2021-01-29 00:20:46 +05:30
function changeViewer(store, index, { automaticallyCollapsed, manuallyCollapsed, name }) {
const file = store.state.diffs.diffFiles[index];
const newViewer = {
...file.viewer,
};
if (automaticallyCollapsed !== undefined) {
newViewer.automaticallyCollapsed = automaticallyCollapsed;
}
if (manuallyCollapsed !== undefined) {
newViewer.manuallyCollapsed = manuallyCollapsed;
}
if (name !== undefined) {
newViewer.name = name;
}
Object.assign(file, {
viewer: newViewer,
});
}
function forceHasDiff({ store, index = 0, inlineLines, parallelLines, readableText }) {
const file = store.state.diffs.diffFiles[index];
Object.assign(file, {
highlighted_diff_lines: inlineLines,
parallel_diff_lines: parallelLines,
blob: {
...file.blob,
readable_text: readableText,
},
});
}
function markFileToBeRendered(store, index = 0) {
const file = store.state.diffs.diffFiles[index];
Object.assign(file, {
renderIt: true,
});
}
2021-03-11 19:13:27 +05:30
function createComponent({ file, first = false, last = false, options = {}, props = {} }) {
2022-04-04 11:22:00 +05:30
Vue.use(Vuex);
2021-01-29 00:20:46 +05:30
const store = new Vuex.Store({
...createNotesStore(),
modules: {
diffs: createDiffsStore(),
},
});
store.state.diffs.diffFiles = [file];
const wrapper = shallowMount(DiffFileComponent, {
store,
propsData: {
file,
2018-11-08 19:23:39 +05:30
canCurrentUserFork: false,
2020-07-28 23:09:34 +05:30
viewDiffsFileByFile: false,
2021-01-29 00:20:46 +05:30
isFirstFile: first,
isLastFile: last,
2021-03-11 19:13:27 +05:30
...props,
2021-01-29 00:20:46 +05:30
},
2021-03-11 19:13:27 +05:30
...options,
2021-01-29 00:20:46 +05:30
});
return {
wrapper,
store,
};
}
2022-10-11 01:57:18 +05:30
const findDiffHeader = (wrapper) => wrapper.findComponent(DiffFileHeaderComponent);
2021-03-08 18:12:59 +05:30
const findDiffContentArea = (wrapper) => wrapper.find('[data-testid="content-area"]');
const findLoader = (wrapper) => wrapper.find('[data-testid="loader-icon"]');
const findToggleButton = (wrapper) => wrapper.find('[data-testid="expand-button"]');
2021-01-29 00:20:46 +05:30
2021-03-08 18:12:59 +05:30
const toggleFile = (wrapper) => findDiffHeader(wrapper).vm.$emit('toggleFile');
2022-08-13 15:12:31 +05:30
const getReadableFile = () => getDiffFileMock();
2021-01-29 00:20:46 +05:30
const getUnreadableFile = () => JSON.parse(JSON.stringify(diffFileMockDataUnreadable));
const makeFileAutomaticallyCollapsed = (store, index = 0) =>
changeViewer(store, index, { automaticallyCollapsed: true, manuallyCollapsed: null });
const makeFileOpenByDefault = (store, index = 0) =>
changeViewer(store, index, { automaticallyCollapsed: false, manuallyCollapsed: null });
const makeFileManuallyCollapsed = (store, index = 0) =>
changeViewer(store, index, { automaticallyCollapsed: false, manuallyCollapsed: true });
const changeViewerType = (store, newType, index = 0) =>
changeViewer(store, index, { name: diffViewerModes[newType] });
describe('DiffFile', () => {
let wrapper;
let store;
2021-03-08 18:12:59 +05:30
let axiosMock;
2021-01-29 00:20:46 +05:30
beforeEach(() => {
2021-03-08 18:12:59 +05:30
axiosMock = new MockAdapter(axios);
2021-01-29 00:20:46 +05:30
({ wrapper, store } = createComponent({ file: getReadableFile() }));
2018-11-08 19:23:39 +05:30
});
2019-10-12 21:52:04 +05:30
afterEach(() => {
2021-03-08 18:12:59 +05:30
axiosMock.restore();
2019-10-12 21:52:04 +05:30
});
2021-01-29 00:20:46 +05:30
describe('bus events', () => {
beforeEach(() => {
jest.spyOn(eventHub, '$emit').mockImplementation(() => {});
});
describe('during mount', () => {
it.each`
first | last | events | file
${false} | ${false} | ${[]} | ${{ inlineLines: [], parallelLines: [], readableText: true }}
${true} | ${true} | ${[]} | ${{ inlineLines: [], parallelLines: [], readableText: true }}
${true} | ${false} | ${[EVT_PERF_MARK_FIRST_DIFF_FILE_SHOWN]} | ${false}
${false} | ${true} | ${[EVT_PERF_MARK_DIFF_FILES_END]} | ${false}
${true} | ${true} | ${[EVT_PERF_MARK_FIRST_DIFF_FILE_SHOWN, EVT_PERF_MARK_DIFF_FILES_END]} | ${false}
`(
'emits the events $events based on the file and its position ({ first: $first, last: $last }) among all files',
async ({ file, first, last, events }) => {
if (file) {
forceHasDiff({ store, ...file });
}
({ wrapper, store } = createComponent({
file: store.state.diffs.diffFiles[0],
first,
last,
}));
2022-04-04 11:22:00 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
expect(eventHub.$emit).toHaveBeenCalledTimes(events.length);
2021-03-08 18:12:59 +05:30
events.forEach((event) => {
2021-01-29 00:20:46 +05:30
expect(eventHub.$emit).toHaveBeenCalledWith(event);
});
},
);
2023-07-09 08:55:56 +05:30
it('emits the "first file shown" and "files end" events when in File-by-File mode', async () => {
({ wrapper, store } = createComponent({
file: getReadableFile(),
first: false,
last: false,
props: {
viewDiffsFileByFile: true,
},
}));
await nextTick();
expect(eventHub.$emit).toHaveBeenCalledTimes(2);
expect(eventHub.$emit).toHaveBeenCalledWith(EVT_PERF_MARK_FIRST_DIFF_FILE_SHOWN);
expect(eventHub.$emit).toHaveBeenCalledWith(EVT_PERF_MARK_DIFF_FILES_END);
});
2021-01-29 00:20:46 +05:30
});
describe('after loading the diff', () => {
it('indicates that it loaded the file', async () => {
forceHasDiff({ store, inlineLines: [], parallelLines: [], readableText: true });
({ wrapper, store } = createComponent({
file: store.state.diffs.diffFiles[0],
first: true,
last: true,
}));
jest.spyOn(wrapper.vm, 'loadCollapsedDiff').mockResolvedValue(getReadableFile());
2021-03-08 18:12:59 +05:30
jest.spyOn(window, 'requestIdleCallback').mockImplementation((fn) => fn());
2021-01-29 00:20:46 +05:30
makeFileAutomaticallyCollapsed(store);
2022-04-04 11:22:00 +05:30
await nextTick(); // Wait for store updates to flow into the component
2021-01-29 00:20:46 +05:30
toggleFile(wrapper);
2022-04-04 11:22:00 +05:30
await nextTick(); // Wait for the load to resolve
await nextTick(); // Wait for the idleCallback
await nextTick(); // Wait for nextTick inside postRender
2021-01-29 00:20:46 +05:30
expect(eventHub.$emit).toHaveBeenCalledTimes(2);
expect(eventHub.$emit).toHaveBeenCalledWith(EVT_PERF_MARK_FIRST_DIFF_FILE_SHOWN);
expect(eventHub.$emit).toHaveBeenCalledWith(EVT_PERF_MARK_DIFF_FILES_END);
});
});
});
2020-04-08 14:13:33 +05:30
2018-11-08 19:23:39 +05:30
describe('template', () => {
2021-01-29 00:20:46 +05:30
it('should render component with file header, file content components', async () => {
const el = wrapper.vm.$el;
const { file_hash } = wrapper.vm.file;
2018-11-08 19:23:39 +05:30
2019-02-15 15:39:39 +05:30
expect(el.id).toEqual(file_hash);
2018-11-08 19:23:39 +05:30
expect(el.classList.contains('diff-file')).toEqual(true);
2018-11-20 20:47:30 +05:30
2018-11-08 19:23:39 +05:30
expect(el.querySelectorAll('.diff-content.hidden').length).toEqual(0);
expect(el.querySelector('.js-file-title')).toBeDefined();
2022-10-11 01:57:18 +05:30
expect(wrapper.findComponent(DiffFileHeaderComponent).exists()).toBe(true);
2018-11-08 19:23:39 +05:30
expect(el.querySelector('.js-syntax-highlight')).toBeDefined();
2018-11-20 20:47:30 +05:30
2021-01-29 00:20:46 +05:30
markFileToBeRendered(store);
2022-04-04 11:22:00 +05:30
await nextTick();
2018-11-20 20:47:30 +05:30
2022-10-11 01:57:18 +05:30
expect(wrapper.findComponent(DiffContentComponent).exists()).toBe(true);
2020-03-13 15:44:24 +05:30
});
2021-01-29 00:20:46 +05:30
});
2020-03-13 15:44:24 +05:30
2021-03-11 19:13:27 +05:30
describe('computed', () => {
describe('showLocalFileReviews', () => {
function setLoggedIn(bool) {
window.gon.current_user_id = bool;
}
it.each`
2021-11-11 11:23:49 +05:30
loggedIn | bool
${true} | ${true}
${false} | ${false}
`('should be $bool when { userIsLoggedIn: $loggedIn }', ({ loggedIn, bool }) => {
setLoggedIn(loggedIn);
({ wrapper } = createComponent({
props: {
file: store.state.diffs.diffFiles[0],
},
}));
2021-03-11 19:13:27 +05:30
2021-11-11 11:23:49 +05:30
expect(wrapper.vm.showLocalFileReviews).toBe(bool);
});
2021-03-11 19:13:27 +05:30
});
});
2021-01-29 00:20:46 +05:30
describe('collapsing', () => {
describe(`\`${EVT_EXPAND_ALL_FILES}\` event`, () => {
beforeEach(() => {
jest.spyOn(wrapper.vm, 'handleToggle').mockImplementation(() => {});
});
2020-03-13 15:44:24 +05:30
2021-01-29 00:20:46 +05:30
it('performs the normal file toggle when the file is collapsed', async () => {
makeFileAutomaticallyCollapsed(store);
2020-03-13 15:44:24 +05:30
2022-04-04 11:22:00 +05:30
await nextTick();
2018-11-08 19:23:39 +05:30
2021-01-29 00:20:46 +05:30
eventHub.$emit(EVT_EXPAND_ALL_FILES);
2018-11-08 19:23:39 +05:30
2021-01-29 00:20:46 +05:30
expect(wrapper.vm.handleToggle).toHaveBeenCalledTimes(1);
2018-11-08 19:23:39 +05:30
});
2021-01-29 00:20:46 +05:30
it('does nothing when the file is not collapsed', async () => {
eventHub.$emit(EVT_EXPAND_ALL_FILES);
2018-11-20 20:47:30 +05:30
2022-04-04 11:22:00 +05:30
await nextTick();
2018-11-20 20:47:30 +05:30
2021-01-29 00:20:46 +05:30
expect(wrapper.vm.handleToggle).not.toHaveBeenCalled();
2018-11-20 20:47:30 +05:30
});
2021-01-29 00:20:46 +05:30
});
2018-11-20 20:47:30 +05:30
2021-01-29 00:20:46 +05:30
describe('user collapsed', () => {
beforeEach(() => {
makeFileManuallyCollapsed(store);
});
2018-11-08 19:23:39 +05:30
2021-01-29 00:20:46 +05:30
it('should not have any content at all', async () => {
2022-04-04 11:22:00 +05:30
await nextTick();
2018-11-08 19:23:39 +05:30
2021-09-30 23:02:18 +05:30
expect(findDiffContentArea(wrapper).element.children.length).toBe(0);
2018-11-08 19:23:39 +05:30
});
2021-01-29 00:20:46 +05:30
it('should not have the class `has-body` to present the header differently', () => {
expect(wrapper.classes('has-body')).toBe(false);
});
});
2019-12-26 22:10:19 +05:30
2021-01-29 00:20:46 +05:30
describe('automatically collapsed', () => {
beforeEach(() => {
makeFileAutomaticallyCollapsed(store);
});
2019-12-26 22:10:19 +05:30
2021-01-29 00:20:46 +05:30
it('should show the collapsed file warning with expansion button', () => {
expect(findDiffContentArea(wrapper).html()).toContain(
'Files with large changes are collapsed by default.',
);
expect(findToggleButton(wrapper).exists()).toBe(true);
});
2019-12-26 22:10:19 +05:30
2021-01-29 00:20:46 +05:30
it('should style the component so that it `.has-body` for layout purposes', () => {
expect(wrapper.classes('has-body')).toBe(true);
2019-12-26 22:10:19 +05:30
});
2021-01-29 00:20:46 +05:30
});
2019-12-26 22:10:19 +05:30
2021-01-29 00:20:46 +05:30
describe('not collapsed', () => {
beforeEach(() => {
makeFileOpenByDefault(store);
markFileToBeRendered(store);
});
2019-02-15 15:39:39 +05:30
2023-06-20 00:43:36 +05:30
it('should have the file content', () => {
2022-10-11 01:57:18 +05:30
expect(wrapper.findComponent(DiffContentComponent).exists()).toBe(true);
2021-01-29 00:20:46 +05:30
});
2019-02-15 15:39:39 +05:30
2021-01-29 00:20:46 +05:30
it('should style the component so that it `.has-body` for layout purposes', () => {
expect(wrapper.classes('has-body')).toBe(true);
2019-02-15 15:39:39 +05:30
});
2021-01-29 00:20:46 +05:30
});
2019-02-15 15:39:39 +05:30
2021-01-29 00:20:46 +05:30
describe('toggle', () => {
2023-06-20 00:43:36 +05:30
it('should update store state', () => {
2021-01-29 00:20:46 +05:30
jest.spyOn(wrapper.vm.$store, 'dispatch').mockImplementation(() => {});
2019-02-15 15:39:39 +05:30
2021-01-29 00:20:46 +05:30
toggleFile(wrapper);
2019-02-15 15:39:39 +05:30
2021-01-29 00:20:46 +05:30
expect(wrapper.vm.$store.dispatch).toHaveBeenCalledWith('diffs/setFileCollapsedByUser', {
filePath: wrapper.vm.file.file_path,
collapsed: true,
2019-02-15 15:39:39 +05:30
});
});
2021-06-08 01:23:25 +05:30
describe('scoll-to-top of file after collapse', () => {
beforeEach(() => {
jest.spyOn(wrapper.vm.$store, 'dispatch').mockImplementation(() => {});
});
it("scrolls to the top when the file is open, the users initiates the collapse, and there's a content block to scroll to", async () => {
makeFileOpenByDefault(store);
await nextTick();
toggleFile(wrapper);
expect(scrollToElement).toHaveBeenCalled();
});
it('does not scroll when the content block is missing', async () => {
makeFileOpenByDefault(store);
await nextTick();
findDiffContentArea(wrapper).element.remove();
toggleFile(wrapper);
expect(scrollToElement).not.toHaveBeenCalled();
});
it("does not scroll if the user doesn't initiate the file collapse", async () => {
makeFileOpenByDefault(store);
await nextTick();
wrapper.vm.handleToggle();
expect(scrollToElement).not.toHaveBeenCalled();
});
it('does not scroll if the file is already collapsed', async () => {
makeFileManuallyCollapsed(store);
await nextTick();
toggleFile(wrapper);
expect(scrollToElement).not.toHaveBeenCalled();
});
});
2021-01-29 00:20:46 +05:30
describe('fetch collapsed diff', () => {
const prepFile = async (inlineLines, parallelLines, readableText) => {
forceHasDiff({
store,
inlineLines,
parallelLines,
readableText,
});
2022-04-04 11:22:00 +05:30
await nextTick();
2018-11-08 19:23:39 +05:30
2021-01-29 00:20:46 +05:30
toggleFile(wrapper);
};
2018-11-08 19:23:39 +05:30
2021-01-29 00:20:46 +05:30
beforeEach(() => {
jest.spyOn(wrapper.vm, 'requestDiff').mockImplementation(() => {});
makeFileAutomaticallyCollapsed(store);
2018-11-08 19:23:39 +05:30
});
2019-07-07 11:18:12 +05:30
2021-01-29 00:20:46 +05:30
it.each`
inlineLines | parallelLines | readableText
${[1]} | ${[1]} | ${true}
${[]} | ${[1]} | ${true}
${[1]} | ${[]} | ${true}
${[1]} | ${[1]} | ${false}
${[]} | ${[]} | ${false}
`(
'does not make a request to fetch the diff for a diff file like { inline: $inlineLines, parallel: $parallelLines, readableText: $readableText }',
async ({ inlineLines, parallelLines, readableText }) => {
await prepFile(inlineLines, parallelLines, readableText);
expect(wrapper.vm.requestDiff).not.toHaveBeenCalled();
},
);
2019-07-07 11:18:12 +05:30
2021-01-29 00:20:46 +05:30
it.each`
inlineLines | parallelLines | readableText
${[]} | ${[]} | ${true}
`(
'makes a request to fetch the diff for a diff file like { inline: $inlineLines, parallel: $parallelLines, readableText: $readableText }',
async ({ inlineLines, parallelLines, readableText }) => {
await prepFile(inlineLines, parallelLines, readableText);
2019-07-07 11:18:12 +05:30
2021-01-29 00:20:46 +05:30
expect(wrapper.vm.requestDiff).toHaveBeenCalled();
},
);
});
});
2019-07-07 11:18:12 +05:30
2021-01-29 00:20:46 +05:30
describe('loading', () => {
it('should have loading icon while loading a collapsed diffs', async () => {
2021-03-08 18:12:59 +05:30
const { load_collapsed_diff_url } = store.state.diffs.diffFiles[0];
2023-03-17 16:20:25 +05:30
axiosMock.onGet(load_collapsed_diff_url).reply(HTTP_STATUS_OK, getReadableFile());
2021-01-29 00:20:46 +05:30
makeFileAutomaticallyCollapsed(store);
2021-03-08 18:12:59 +05:30
wrapper.vm.requestDiff();
2021-01-29 00:20:46 +05:30
2022-04-04 11:22:00 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
expect(findLoader(wrapper).exists()).toBe(true);
2019-07-07 11:18:12 +05:30
});
2021-01-29 00:20:46 +05:30
});
2019-07-07 11:18:12 +05:30
2021-01-29 00:20:46 +05:30
describe('general (other) collapsed', () => {
it('should be expandable for unreadable files', async () => {
({ wrapper, store } = createComponent({ file: getUnreadableFile() }));
makeFileAutomaticallyCollapsed(store);
2019-07-07 11:18:12 +05:30
2022-04-04 11:22:00 +05:30
await nextTick();
2019-07-07 11:18:12 +05:30
2021-01-29 00:20:46 +05:30
expect(findDiffContentArea(wrapper).html()).toContain(
'Files with large changes are collapsed by default.',
);
expect(findToggleButton(wrapper).exists()).toBe(true);
2019-07-07 11:18:12 +05:30
});
2021-01-29 00:20:46 +05:30
it.each`
mode
${'renamed'}
${'mode_changed'}
`(
'should render the DiffContent component for files whose mode is $mode',
async ({ mode }) => {
makeFileOpenByDefault(store);
markFileToBeRendered(store);
changeViewerType(store, mode);
2022-04-04 11:22:00 +05:30
await nextTick();
2021-01-29 00:20:46 +05:30
expect(wrapper.classes('has-body')).toBe(true);
2022-10-11 01:57:18 +05:30
expect(wrapper.findComponent(DiffContentComponent).exists()).toBe(true);
expect(wrapper.findComponent(DiffContentComponent).isVisible()).toBe(true);
2021-01-29 00:20:46 +05:30
},
);
2018-11-08 19:23:39 +05:30
});
});
describe('too large diff', () => {
2021-01-29 00:20:46 +05:30
it('should have too large warning and blob link', async () => {
const file = store.state.diffs.diffFiles[0];
2018-11-08 19:23:39 +05:30
const BLOB_LINK = '/file/view/path';
2018-12-13 13:39:08 +05:30
2021-01-29 00:20:46 +05:30
Object.assign(store.state.diffs.diffFiles[0], {
...file,
view_path: BLOB_LINK,
renderIt: true,
viewer: {
...file.viewer,
error: diffViewerErrors.too_large,
error_message: 'This source diff could not be displayed because it is too large',
},
2018-11-08 19:23:39 +05:30
});
2018-12-13 13:39:08 +05:30
2022-04-04 11:22:00 +05:30
await nextTick();
2019-12-26 22:10:19 +05:30
2021-03-11 19:13:27 +05:30
const button = wrapper.find('[data-testid="blob-button"]');
expect(wrapper.text()).toContain('Changes are too large to be shown.');
expect(button.html()).toContain('View file @');
expect(button.attributes('href')).toBe('/file/view/path');
2019-12-26 22:10:19 +05:30
});
2018-12-13 13:39:08 +05:30
});
2021-10-27 15:23:28 +05:30
it('loads collapsed file on mounted when single file mode is enabled', async () => {
const file = {
...getReadableFile(),
load_collapsed_diff_url: '/diff_for_path',
highlighted_diff_lines: [],
parallel_diff_lines: [],
viewer: { name: 'collapsed', automaticallyCollapsed: true },
};
2023-03-17 16:20:25 +05:30
axiosMock.onGet(file.load_collapsed_diff_url).reply(HTTP_STATUS_OK, getReadableFile());
2021-10-27 15:23:28 +05:30
({ wrapper, store } = createComponent({ file, props: { viewDiffsFileByFile: true } }));
2022-04-04 11:22:00 +05:30
await nextTick();
2021-10-27 15:23:28 +05:30
expect(findLoader(wrapper).exists()).toBe(true);
});
describe('merge conflicts', () => {
it('does not render conflict alert', () => {
const file = {
...getReadableFile(),
conflict_type: null,
renderIt: true,
};
({ wrapper, store } = createComponent({ file }));
expect(wrapper.find('[data-testid="conflictsAlert"]').exists()).toBe(false);
});
it('renders conflict alert when conflict_type is present', () => {
const file = {
...getReadableFile(),
conflict_type: 'both_modified',
renderIt: true,
};
({ wrapper, store } = createComponent({ file }));
expect(wrapper.find('[data-testid="conflictsAlert"]').exists()).toBe(true);
});
});
2018-11-08 19:23:39 +05:30
});