mirror of
				https://github.com/Fabio286/antares.git
				synced 2025-06-05 21:59:22 +02:00 
			
		
		
		
	feat: cancel button when waiting to connect database, closes #830
This commit is contained in:
		| @@ -386,20 +386,35 @@ | ||||
|                </div> | ||||
|             </div> | ||||
|             <div class="panel-footer"> | ||||
|                <button | ||||
|                   id="connection-test" | ||||
|                   class="btn btn-gray mr-2 d-flex" | ||||
|                   :class="{'loading': isTesting}" | ||||
|                   :disabled="isBusy" | ||||
|                   @click="startTest" | ||||
|                <div | ||||
|                   @mouseenter="setCancelTestButtonVisibility(true)" | ||||
|                   @mouseleave="setCancelTestButtonVisibility(false)" | ||||
|                > | ||||
|                   <BaseIcon | ||||
|                      icon-name="mdiLightningBolt" | ||||
|                      :size="24" | ||||
|                      class="mr-1" | ||||
|                   /> | ||||
|                   {{ t('connection.testConnection') }} | ||||
|                </button> | ||||
|                   <button | ||||
|                      v-if="showTestCancel && isTesting" | ||||
|                      class="btn btn-gray mr-2 cancellable" | ||||
|                      :title="t('general.cancel')" | ||||
|                      @click="abortConnection()" | ||||
|                   > | ||||
|                      <BaseIcon icon-name="mdiWindowClose" :size="24" /> | ||||
|                      <span class="d-invisible pr-1">{{ t('connection.testConnection') }}</span> | ||||
|                   </button> | ||||
|                   <button | ||||
|                      v-else | ||||
|                      id="connection-test" | ||||
|                      class="btn btn-gray mr-2 d-flex" | ||||
|                      :class="{'loading': isTesting}" | ||||
|                      :disabled="isBusy" | ||||
|                      @click="startTest" | ||||
|                   > | ||||
|                      <BaseIcon | ||||
|                         icon-name="mdiLightningBolt" | ||||
|                         :size="24" | ||||
|                         class="mr-1" | ||||
|                      /> | ||||
|                      {{ t('connection.testConnection') }} | ||||
|                   </button> | ||||
|                </div> | ||||
|                <button | ||||
|                   id="connection-save" | ||||
|                   class="btn btn-primary mr-2 d-flex" | ||||
| @@ -494,6 +509,8 @@ const firstInput: Ref<HTMLInputElement> = ref(null); | ||||
| const isConnecting = ref(false); | ||||
| const isTesting = ref(false); | ||||
| const isAsking = ref(false); | ||||
| const showTestCancel = ref(false); | ||||
| const abortController: Ref<AbortController> = ref(new AbortController()); | ||||
| const selectedTab = ref('general'); | ||||
|  | ||||
| const clientCustomizations = computed(() => { | ||||
| @@ -516,6 +533,10 @@ const setDefaults = () => { | ||||
|    connection.value.database = clientCustomizations.value.defaultDatabase; | ||||
| }; | ||||
|  | ||||
| const setCancelTestButtonVisibility = (val: boolean) => { | ||||
|    showTestCancel.value = val; | ||||
| }; | ||||
|  | ||||
| const startTest = async () => { | ||||
|    isTesting.value = true; | ||||
|  | ||||
| @@ -526,7 +547,7 @@ const startTest = async () => { | ||||
|          const res = await Connection.makeTest(connection.value); | ||||
|          if (res.status === 'error') | ||||
|             addNotification({ status: 'error', message: res.response.message || res.response.toString() }); | ||||
|          else | ||||
|          else if (res.status === 'success') | ||||
|             addNotification({ status: 'success', message: t('connection.connectionSuccessfullyMade') }); | ||||
|       } | ||||
|       catch (err) { | ||||
| @@ -537,13 +558,21 @@ const startTest = async () => { | ||||
|    } | ||||
| }; | ||||
|  | ||||
| const abortConnection = (): void => { | ||||
|    abortController.value.abort(); | ||||
|    Connection.abortConnection(connection.value.uid); | ||||
|    isTesting.value = false; | ||||
|    isConnecting.value = false; | ||||
|    abortController.value = new AbortController(); | ||||
| }; | ||||
|  | ||||
| const continueTest = async (credentials: { user: string; password: string }) => { // if "Ask for credentials" is true | ||||
|    isAsking.value = false; | ||||
|    const params = Object.assign({}, connection.value, credentials); | ||||
|  | ||||
|    try { | ||||
|       if (isConnecting.value) { | ||||
|          await connectWorkspace(params); | ||||
|          await connectWorkspace(params, { signal: abortController.value.signal }).catch(() => undefined); | ||||
|          isConnecting.value = false; | ||||
|       } | ||||
|       else { | ||||
|   | ||||
| @@ -387,20 +387,35 @@ | ||||
|             </div> | ||||
|          </div> | ||||
|          <div class="panel-footer"> | ||||
|             <button | ||||
|                id="connection-test" | ||||
|                class="btn btn-gray mr-2 d-flex" | ||||
|                :class="{'loading': isTesting}" | ||||
|                :disabled="isBusy" | ||||
|                @click="startTest" | ||||
|             <div | ||||
|                @mouseenter="setCancelTestButtonVisibility(true)" | ||||
|                @mouseleave="setCancelTestButtonVisibility(false)" | ||||
|             > | ||||
|                <BaseIcon | ||||
|                   icon-name="mdiLightningBolt" | ||||
|                   :size="24" | ||||
|                   class="mr-1" | ||||
|                /> | ||||
|                {{ t('connection.testConnection') }} | ||||
|             </button> | ||||
|                <button | ||||
|                   v-if="showTestCancel && isTesting" | ||||
|                   class="btn btn-gray mr-2 cancellable" | ||||
|                   :title="t('general.cancel')" | ||||
|                   @click="abortConnection()" | ||||
|                > | ||||
|                   <BaseIcon icon-name="mdiWindowClose" :size="24" /> | ||||
|                   <span class="d-invisible pr-1">{{ t('connection.testConnection') }}</span> | ||||
|                </button> | ||||
|                <button | ||||
|                   v-else | ||||
|                   id="connection-test" | ||||
|                   class="btn btn-gray mr-2 d-flex" | ||||
|                   :class="{'loading': isTesting}" | ||||
|                   :disabled="isBusy" | ||||
|                   @click="startTest" | ||||
|                > | ||||
|                   <BaseIcon | ||||
|                      icon-name="mdiLightningBolt" | ||||
|                      :size="24" | ||||
|                      class="mr-1" | ||||
|                   /> | ||||
|                   {{ t('connection.testConnection') }} | ||||
|                </button> | ||||
|             </div> | ||||
|             <button | ||||
|                id="connection-save" | ||||
|                class="btn btn-primary mr-2 d-flex" | ||||
| @@ -414,20 +429,35 @@ | ||||
|                /> | ||||
|                {{ t('general.save') }} | ||||
|             </button> | ||||
|             <button | ||||
|                id="connection-connect" | ||||
|                class="btn btn-success d-flex" | ||||
|                :class="{'loading': isConnecting}" | ||||
|                :disabled="isBusy" | ||||
|                @click="startConnection" | ||||
|             <div | ||||
|                @mouseenter="setCancelConnectButtonVisibility(true)" | ||||
|                @mouseleave="setCancelConnectButtonVisibility(false)" | ||||
|             > | ||||
|                <BaseIcon | ||||
|                   icon-name="mdiConnection" | ||||
|                   :size="24" | ||||
|                   class="mr-1" | ||||
|                /> | ||||
|                {{ t('connection.connect') }} | ||||
|             </button> | ||||
|                <button | ||||
|                   v-if="showConnectCancel && isConnecting" | ||||
|                   class="btn btn-success cancellable" | ||||
|                   :title="t('general.cancel')" | ||||
|                   @click="abortConnection()" | ||||
|                > | ||||
|                   <BaseIcon icon-name="mdiWindowClose" :size="24" /> | ||||
|                   <span class="d-invisible pr-1">{{ t('connection.connect') }}</span> | ||||
|                </button> | ||||
|                <button | ||||
|                   v-else | ||||
|                   id="connection-connect" | ||||
|                   class="btn btn-success d-flex" | ||||
|                   :class="{'loading': isConnecting}" | ||||
|                   :disabled="isBusy" | ||||
|                   @click="startConnection" | ||||
|                > | ||||
|                   <BaseIcon | ||||
|                      icon-name="mdiConnection" | ||||
|                      :size="24" | ||||
|                      class="mr-1" | ||||
|                   /> | ||||
|                   {{ t('connection.connect') }} | ||||
|                </button> | ||||
|             </div> | ||||
|          </div> | ||||
|       </div> | ||||
|       <ModalAskCredentials | ||||
| @@ -476,6 +506,9 @@ const localConnection: Ref<ConnectionParams & { pgConnString: string }> = ref(nu | ||||
| const isConnecting = ref(false); | ||||
| const isTesting = ref(false); | ||||
| const isAsking = ref(false); | ||||
| const showTestCancel = ref(false); | ||||
| const showConnectCancel = ref(false); | ||||
| const abortController: Ref<AbortController> = ref(new AbortController()); | ||||
| const selectedTab = ref('general'); | ||||
|  | ||||
| const clientCustomizations = computed(() => { | ||||
| @@ -501,7 +534,7 @@ const startConnection = async () => { | ||||
|    if (localConnection.value.ask) | ||||
|       isAsking.value = true; | ||||
|    else { | ||||
|       await connectWorkspace(localConnection.value); | ||||
|       await connectWorkspace(localConnection.value, { signal: abortController.value.signal }).catch(() => undefined); | ||||
|       isConnecting.value = false; | ||||
|    } | ||||
| }; | ||||
| @@ -516,7 +549,7 @@ const startTest = async () => { | ||||
|          const res = await Connection.makeTest(localConnection.value); | ||||
|          if (res.status === 'error') | ||||
|             addNotification({ status: 'error', message: res.response.message || res.response.toString() }); | ||||
|          else | ||||
|          else if (res.status === 'success') | ||||
|             addNotification({ status: 'success', message: t('connection.connectionSuccessfullyMade') }); | ||||
|       } | ||||
|       catch (err) { | ||||
| @@ -527,20 +560,36 @@ const startTest = async () => { | ||||
|    } | ||||
| }; | ||||
|  | ||||
| const setCancelTestButtonVisibility = (val: boolean) => { | ||||
|    showTestCancel.value = val; | ||||
| }; | ||||
|  | ||||
| const setCancelConnectButtonVisibility = (val: boolean) => { | ||||
|    showConnectCancel.value = val; | ||||
| }; | ||||
|  | ||||
| const abortConnection = (): void => { | ||||
|    abortController.value.abort(); | ||||
|    Connection.abortConnection(localConnection.value.uid); | ||||
|    isTesting.value = false; | ||||
|    isConnecting.value = false; | ||||
|    abortController.value = new AbortController(); | ||||
| }; | ||||
|  | ||||
| const continueTest = async (credentials: {user: string; password: string }) => { // if "Ask for credentials" is true | ||||
|    isAsking.value = false; | ||||
|    const params = Object.assign({}, localConnection.value, credentials); | ||||
|    try { | ||||
|       if (isConnecting.value) { | ||||
|          const params = Object.assign({}, props.connection, credentials); | ||||
|          await connectWorkspace(params); | ||||
|          await connectWorkspace(params, { signal: abortController.value.signal }).catch(() => undefined); | ||||
|          isConnecting.value = false; | ||||
|       } | ||||
|       else { | ||||
|          const res = await Connection.makeTest(params); | ||||
|          if (res.status === 'error') | ||||
|             addNotification({ status: 'error', message: res.response.message || res.response.toString() }); | ||||
|          else | ||||
|          else if (res.status === 'success') | ||||
|             addNotification({ status: 'success', message: t('connection.connectionSuccessfullyMade') }); | ||||
|       } | ||||
|    } | ||||
|   | ||||
| @@ -15,6 +15,10 @@ export default class { | ||||
|       return ipcRenderer.invoke('connect', unproxify(newParams)); | ||||
|    } | ||||
|  | ||||
|    static abortConnection (uid: string): void { | ||||
|       ipcRenderer.send('abort-connection', uid); | ||||
|    } | ||||
|  | ||||
|    static checkConnection (uid: string): Promise<boolean> { | ||||
|       return ipcRenderer.invoke('check-connection', uid); | ||||
|    } | ||||
|   | ||||
| @@ -147,7 +147,7 @@ export const useWorkspacesStore = defineStore('workspaces', { | ||||
|          else | ||||
|             this.selectedWorkspace = uid; | ||||
|       }, | ||||
|       async connectWorkspace (connection: ConnectionParams & { pgConnString?: string }, mode?: string) { | ||||
|       async connectWorkspace (connection: ConnectionParams & { pgConnString?: string }, args?: {mode?: string; signal?: AbortSignal}) { | ||||
|          this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid | ||||
|             ? { | ||||
|                ...workspace, | ||||
| @@ -155,112 +155,136 @@ export const useWorkspacesStore = defineStore('workspaces', { | ||||
|                breadcrumbs: {}, | ||||
|                loadedSchemas: new Set(), | ||||
|                database: connection.database, | ||||
|                connectionStatus: mode === 'switch' ? 'connected' : 'connecting' | ||||
|                connectionStatus: args?.mode === 'switch' ? 'connected' : 'connecting' | ||||
|             } | ||||
|             : workspace); | ||||
|  | ||||
|          const connectionsStore = useConnectionsStore(); | ||||
|          const notificationsStore = useNotificationsStore(); | ||||
|          const settingsStore = useSettingsStore(); | ||||
|  | ||||
|          try { | ||||
|             const { status, response } = await Connection.connect(connection); | ||||
|  | ||||
|             if (status === 'error') { | ||||
|                notificationsStore.addNotification({ status, message: response }); | ||||
|          return new Promise((resolve, reject) => { | ||||
|             const abortHandler = () => { | ||||
|                this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid | ||||
|                   ? { | ||||
|                      ...workspace, | ||||
|                      structure: [], | ||||
|                      breadcrumbs: {}, | ||||
|                      loadedSchemas: new Set(), | ||||
|                      connectionStatus: 'failed' | ||||
|                      connectionStatus: 'disconnected' | ||||
|                   } | ||||
|                   : workspace); | ||||
|             } | ||||
|             else { | ||||
|                let clientCustomizations: Customizations; | ||||
|                const { updateLastConnection } = connectionsStore; | ||||
|                return reject(new Error('Connection aborted by user')); | ||||
|             }; | ||||
|  | ||||
|                updateLastConnection(connection.uid); | ||||
|             args?.signal?.addEventListener('abort', abortHandler); | ||||
|  | ||||
|                switch (connection.client) { | ||||
|                   case 'mysql': | ||||
|                   case 'maria': | ||||
|                      clientCustomizations = customizations.mysql; | ||||
|                      break; | ||||
|                   case 'pg': | ||||
|                      clientCustomizations = customizations.pg; | ||||
|                      break; | ||||
|                   case 'sqlite': | ||||
|                      clientCustomizations = customizations.sqlite; | ||||
|                      break; | ||||
|                   case 'firebird': | ||||
|                      clientCustomizations = customizations.firebird; | ||||
|                      break; | ||||
|                } | ||||
|                const dataTypes = clientCustomizations.dataTypes; | ||||
|                const indexTypes = clientCustomizations.indexTypes; | ||||
|             (async () => { | ||||
|                try { | ||||
|                   const { status, response } = await Connection.connect(connection); | ||||
|  | ||||
|                const { status, response: version } = await Schema.getVersion(connection.uid); | ||||
|                   if (status === 'error') { | ||||
|                      notificationsStore.addNotification({ status, message: response }); | ||||
|                      this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid | ||||
|                         ? { | ||||
|                            ...workspace, | ||||
|                            structure: [], | ||||
|                            breadcrumbs: {}, | ||||
|                            loadedSchemas: new Set(), | ||||
|                            connectionStatus: 'failed' | ||||
|                         } | ||||
|                         : workspace); | ||||
|  | ||||
|                if (status === 'error') | ||||
|                   notificationsStore.addNotification({ status, message: version }); | ||||
|  | ||||
|                // Check if Maria or MySQL | ||||
|                const isMySQL = version.name.includes('MySQL'); | ||||
|                const isMaria = version.name.includes('Maria'); | ||||
|  | ||||
|                if (isMySQL && connection.client !== 'mysql') { | ||||
|                   const connProxy = Object.assign({}, connection); | ||||
|                   connProxy.client = 'mysql'; | ||||
|                   connectionsStore.editConnection(connProxy); | ||||
|                } | ||||
|                else if (isMaria && connection.client === 'mysql') { | ||||
|                   const connProxy = Object.assign({}, connection); | ||||
|                   connProxy.client = 'maria'; | ||||
|                   connectionsStore.editConnection(connProxy); | ||||
|                } | ||||
|  | ||||
|                const cachedTabs: WorkspaceTab[] = settingsStore.restoreTabs ? persistentStore.get(connection.uid, []) as WorkspaceTab[] : []; | ||||
|  | ||||
|                if (cachedTabs.length) { | ||||
|                   tabIndex[connection.uid] = cachedTabs.reduce((acc: number, curr) => { | ||||
|                      if (curr.index > acc) acc = curr.index; | ||||
|                      return acc; | ||||
|                   }, null); | ||||
|                } | ||||
|  | ||||
|                const selectedTab = cachedTabs.length | ||||
|                   ? connection.database | ||||
|                      ? cachedTabs.filter(tab => tab.type === 'query' || tab.database === connection.database)[0]?.uid | ||||
|                      : cachedTabs[0].uid | ||||
|                   : null; | ||||
|  | ||||
|                this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid | ||||
|                   ? { | ||||
|                      ...workspace, | ||||
|                      client: connection.client, | ||||
|                      dataTypes, | ||||
|                      indexTypes, | ||||
|                      customizations: clientCustomizations, | ||||
|                      structure: response, | ||||
|                      connectionStatus: 'connected', | ||||
|                      tabs: cachedTabs, | ||||
|                      selectedTab, | ||||
|                      version | ||||
|                      return reject(new Error(response)); | ||||
|                   } | ||||
|                   : workspace); | ||||
|                   else if (status === 'abort') | ||||
|                      return reject(new Error('Connection aborted by user')); | ||||
|                   else { | ||||
|                      let clientCustomizations: Customizations; | ||||
|                      const { updateLastConnection } = connectionsStore; | ||||
|  | ||||
|                this.refreshCollations(connection.uid); | ||||
|                this.refreshVariables(connection.uid); | ||||
|                this.refreshEngines(connection.uid); | ||||
|                this.refreshUsers(connection.uid); | ||||
|             } | ||||
|          } | ||||
|          catch (err) { | ||||
|             notificationsStore.addNotification({ status: 'error', message: err.stack }); | ||||
|          } | ||||
|                      updateLastConnection(connection.uid); | ||||
|  | ||||
|                      switch (connection.client) { | ||||
|                         case 'mysql': | ||||
|                         case 'maria': | ||||
|                            clientCustomizations = customizations.mysql; | ||||
|                            break; | ||||
|                         case 'pg': | ||||
|                            clientCustomizations = customizations.pg; | ||||
|                            break; | ||||
|                         case 'sqlite': | ||||
|                            clientCustomizations = customizations.sqlite; | ||||
|                            break; | ||||
|                         case 'firebird': | ||||
|                            clientCustomizations = customizations.firebird; | ||||
|                            break; | ||||
|                      } | ||||
|                      const dataTypes = clientCustomizations.dataTypes; | ||||
|                      const indexTypes = clientCustomizations.indexTypes; | ||||
|  | ||||
|                      const { status, response: version } = await Schema.getVersion(connection.uid); | ||||
|  | ||||
|                      if (status === 'error') | ||||
|                         notificationsStore.addNotification({ status, message: version }); | ||||
|  | ||||
|                      // Check if Maria or MySQL | ||||
|                      const isMySQL = version.name.includes('MySQL'); | ||||
|                      const isMaria = version.name.includes('Maria'); | ||||
|  | ||||
|                      if (isMySQL && connection.client !== 'mysql') { | ||||
|                         const connProxy = Object.assign({}, connection); | ||||
|                         connProxy.client = 'mysql'; | ||||
|                         connectionsStore.editConnection(connProxy); | ||||
|                      } | ||||
|                      else if (isMaria && connection.client === 'mysql') { | ||||
|                         const connProxy = Object.assign({}, connection); | ||||
|                         connProxy.client = 'maria'; | ||||
|                         connectionsStore.editConnection(connProxy); | ||||
|                      } | ||||
|  | ||||
|                      const cachedTabs: WorkspaceTab[] = settingsStore.restoreTabs ? persistentStore.get(connection.uid, []) as WorkspaceTab[] : []; | ||||
|  | ||||
|                      if (cachedTabs.length) { | ||||
|                         tabIndex[connection.uid] = cachedTabs.reduce((acc: number, curr) => { | ||||
|                            if (curr.index > acc) acc = curr.index; | ||||
|                            return acc; | ||||
|                         }, null); | ||||
|                      } | ||||
|  | ||||
|                      const selectedTab = cachedTabs.length | ||||
|                         ? connection.database | ||||
|                            ? cachedTabs.filter(tab => tab.type === 'query' || tab.database === connection.database)[0]?.uid | ||||
|                            : cachedTabs[0].uid | ||||
|                         : null; | ||||
|  | ||||
|                      this.workspaces = (this.workspaces as Workspace[]).map(workspace => workspace.uid === connection.uid | ||||
|                         ? { | ||||
|                            ...workspace, | ||||
|                            client: connection.client, | ||||
|                            dataTypes, | ||||
|                            indexTypes, | ||||
|                            customizations: clientCustomizations, | ||||
|                            structure: response, | ||||
|                            connectionStatus: 'connected', | ||||
|                            tabs: cachedTabs, | ||||
|                            selectedTab, | ||||
|                            version | ||||
|                         } | ||||
|                         : workspace); | ||||
|  | ||||
|                      args?.signal?.removeEventListener('abort', abortHandler); | ||||
|                      this.refreshCollations(connection.uid); | ||||
|                      this.refreshVariables(connection.uid); | ||||
|                      this.refreshEngines(connection.uid); | ||||
|                      this.refreshUsers(connection.uid); | ||||
|                      resolve(true); | ||||
|                   } | ||||
|                } | ||||
|                catch (err) { | ||||
|                   notificationsStore.addNotification({ status: 'error', message: err.stack }); | ||||
|                } | ||||
|             })(); | ||||
|          }); | ||||
|       }, | ||||
|       async refreshStructure (uid: string) { | ||||
|          const notificationsStore = useNotificationsStore(); | ||||
| @@ -405,7 +429,7 @@ export const useWorkspacesStore = defineStore('workspaces', { | ||||
|       }, | ||||
|       async switchConnection (connection: ConnectionParams & { pgConnString?: string }) { | ||||
|          await Connection.disconnect(connection.uid); | ||||
|          return this.connectWorkspace(connection, 'switch'); | ||||
|          return this.connectWorkspace(connection, { mode: 'switch' }); | ||||
|       }, | ||||
|       addWorkspace (uid: string) { | ||||
|          const workspace: Workspace = { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user