/** * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ import React from 'react'; import configureStore from 'redux-mock-store'; import thunk from 'redux-thunk'; import fetchMock from 'fetch-mock'; import * as featureFlags from 'src/featureFlags'; import { shallow } from 'enzyme'; import * as utils from 'src/utils/common'; import ShareSqlLabQuery from 'src/SqlLab/components/ShareSqlLabQuery'; import CopyToClipboard from 'src/components/CopyToClipboard'; import { Tooltip } from 'src/common/components/Tooltip'; const mockStore = configureStore([thunk]); const store = mockStore(); let isFeatureEnabledMock; const clipboardSpy = jest.fn(); describe('ShareSqlLabQuery', () => { const storeQueryUrl = 'glob:*/kv/store/'; const storeQueryMockId = '123'; beforeEach(() => { fetchMock.post(storeQueryUrl, () => ({ id: storeQueryMockId }), { overwriteRoutes: true, }); fetchMock.resetHistory(); jest.clearAllMocks(); }); afterAll(fetchMock.reset); const defaultProps = { queryEditor: { dbId: 0, title: 'query title', schema: 'query_schema', autorun: false, sql: 'SELECT * FROM ...', remoteId: 999, }, }; const storedQueryAttributes = { dbId: 0, title: 'query title', schema: 'query_schema', autorun: false, sql: 'SELECT * FROM ...', }; function setup(overrideProps) { const wrapper = shallow( , ).dive(); // wrapped in withToasts HOC return wrapper; } describe('via /kv/store', () => { beforeAll(() => { isFeatureEnabledMock = jest .spyOn(featureFlags, 'isFeatureEnabled') .mockImplementation(() => true); }); afterAll(() => { isFeatureEnabledMock.restore(); }); it('calls storeQuery() with the query when getCopyUrl() is called', () => { expect.assertions(4); const storeQuerySpy = jest.spyOn(utils, 'storeQuery'); const wrapper = setup(); const instance = wrapper.instance(); return instance.getCopyUrl(clipboardSpy).then(() => { expect(storeQuerySpy.mock.calls).toHaveLength(1); expect(fetchMock.calls(storeQueryUrl)).toHaveLength(1); expect(storeQuerySpy.mock.calls[0][0]).toMatchObject( storedQueryAttributes, ); expect(clipboardSpy).toHaveBeenCalledWith( expect.stringContaining('?id='), ); storeQuerySpy.mockRestore(); return Promise.resolve(); }); }); it('dispatches an error toast upon fetching failure', () => { expect.assertions(3); const error = 'There was an error with your request'; const addDangerToastSpy = jest.fn(); fetchMock.post( storeQueryUrl, { throws: error }, { overwriteRoutes: true }, ); const wrapper = setup(); wrapper.setProps({ addDangerToast: addDangerToastSpy }); return wrapper .instance() .getCopyUrl(clipboardSpy) .then(() => { // Fails then retries thrice expect(fetchMock.calls(storeQueryUrl)).toHaveLength(4); expect(addDangerToastSpy.mock.calls).toHaveLength(1); expect(addDangerToastSpy.mock.calls[0][0]).toBe(error); return Promise.resolve(); }); }); }); describe('via saved query', () => { beforeAll(() => { isFeatureEnabledMock = jest .spyOn(featureFlags, 'isFeatureEnabled') .mockImplementation(() => false); }); afterAll(() => { isFeatureEnabledMock.restore(); }); it('does not call storeQuery() with the query when getCopyUrl() is called', () => { const storeQuerySpy = jest.spyOn(utils, 'storeQuery'); const wrapper = setup(); const instance = wrapper.instance(); instance.getCopyUrl(clipboardSpy); expect(storeQuerySpy.mock.calls).toHaveLength(0); expect(fetchMock.calls(storeQueryUrl)).toHaveLength(0); expect(clipboardSpy).toHaveBeenCalledWith( expect.stringContaining('savedQueryId'), ); storeQuerySpy.mockRestore(); }); it('shows a request to save the query when the query is not yet saved', () => { const wrapper = setup({ queryEditor: { ...defaultProps.queryEditor, remoteId: undefined, }, }); expect(wrapper.find(CopyToClipboard)).toHaveLength(0); expect(wrapper.find('.btn-disabled')).toHaveLength(1); expect(wrapper.find(Tooltip)).toHaveProp( 'title', expect.stringContaining('Save the query'), ); }); }); });