Commit 21b082b0 authored by Natalia Tepluhina's avatar Natalia Tepluhina

Merge branch 'leipert-generalize-startup-js-for-graphql' into 'master'

Try to generalize GraphQL startupJS approach

See merge request gitlab-org/gitlab!45060
parents ceafd6c9 78e71f16
...@@ -5,6 +5,7 @@ import { ApolloLink } from 'apollo-link'; ...@@ -5,6 +5,7 @@ import { ApolloLink } from 'apollo-link';
import { BatchHttpLink } from 'apollo-link-batch-http'; import { BatchHttpLink } from 'apollo-link-batch-http';
import csrf from '~/lib/utils/csrf'; import csrf from '~/lib/utils/csrf';
import PerformanceBarService from '~/performance_bar/services/performance_bar_service'; import PerformanceBarService from '~/performance_bar/services/performance_bar_service';
import { StartupJSLink } from '~/lib/utils/apollo_startup_js_link';
export const fetchPolicies = { export const fetchPolicies = {
CACHE_FIRST: 'cache-first', CACHE_FIRST: 'cache-first',
...@@ -62,7 +63,7 @@ export default (resolvers = {}, config = {}) => { ...@@ -62,7 +63,7 @@ export default (resolvers = {}, config = {}) => {
return new ApolloClient({ return new ApolloClient({
typeDefs: config.typeDefs, typeDefs: config.typeDefs,
link: ApolloLink.from([performanceBarLink, uploadsLink]), link: ApolloLink.from([performanceBarLink, new StartupJSLink(), uploadsLink]),
cache: new InMemoryCache({ cache: new InMemoryCache({
...config.cacheConfig, ...config.cacheConfig,
freezeResults: config.assumeImmutableResults, freezeResults: config.assumeImmutableResults,
......
import { ApolloLink, Observable } from 'apollo-link';
import { parse } from 'graphql';
import { isEqual, pickBy } from 'lodash';
/**
* Remove undefined values from object
* @param obj
* @returns {Dictionary<unknown>}
*/
const pickDefinedValues = obj => pickBy(obj, x => x !== undefined);
/**
* Compares two set of variables, order independent
*
* Ignores undefined values (in the top level) and supports arrays etc.
*/
const variablesMatch = (var1 = {}, var2 = {}) => {
return isEqual(pickDefinedValues(var1), pickDefinedValues(var2));
};
export class StartupJSLink extends ApolloLink {
constructor() {
super();
this.startupCalls = new Map();
this.parseStartupCalls(window.gl?.startup_graphql_calls || []);
}
// Extract operationNames from the queries and ensure that we can
// match operationName => element from result array
parseStartupCalls(calls) {
calls.forEach(call => {
const { query, variables, fetchCall } = call;
const operationName = parse(query)?.definitions?.find(x => x.kind === 'OperationDefinition')
?.name?.value;
if (operationName) {
this.startupCalls.set(operationName, {
variables,
fetchCall,
});
}
});
}
static noopRequest = (operation, forward) => forward(operation);
disable() {
this.request = StartupJSLink.noopRequest;
this.startupCalls = null;
}
request(operation, forward) {
// Disable StartupJSLink in case all calls are done or none are set up
if (this.startupCalls && this.startupCalls.size === 0) {
this.disable();
return forward(operation);
}
const { operationName } = operation;
// Skip startup call if the operationName doesn't match
if (!this.startupCalls.has(operationName)) {
return forward(operation);
}
const { variables: startupVariables, fetchCall } = this.startupCalls.get(operationName);
this.startupCalls.delete(operationName);
// Skip startup call if the variables values do not match
if (!variablesMatch(startupVariables, operation.variables)) {
return forward(operation);
}
return new Observable(observer => {
fetchCall
.then(response => {
// Handle HTTP errors
if (!response.ok) {
throw new Error('fetchCall failed');
}
operation.setContext({ response });
return response.json();
})
.then(result => {
if (result && (result.errors || !result.data)) {
throw new Error('Received GraphQL error');
}
// we have data and can send it to back up the link chain
observer.next(result);
observer.complete();
})
.catch(() => {
forward(operation).subscribe({
next: result => {
observer.next(result);
},
error: error => {
observer.error(error);
},
complete: observer.complete.bind(observer),
});
});
});
}
}
...@@ -6,12 +6,12 @@ import { ...@@ -6,12 +6,12 @@ import {
GlDropdownItem, GlDropdownItem,
GlIcon, GlIcon,
} from '@gitlab/ui'; } from '@gitlab/ui';
import permissionsQuery from 'shared_queries/repository/permissions.query.graphql';
import { joinPaths, escapeFileUrl } from '~/lib/utils/url_utility'; import { joinPaths, escapeFileUrl } from '~/lib/utils/url_utility';
import { __ } from '../../locale'; import { __ } from '../../locale';
import getRefMixin from '../mixins/get_ref'; import getRefMixin from '../mixins/get_ref';
import projectShortPathQuery from '../queries/project_short_path.query.graphql'; import projectShortPathQuery from '../queries/project_short_path.query.graphql';
import projectPathQuery from '../queries/project_path.query.graphql'; import projectPathQuery from '../queries/project_path.query.graphql';
import permissionsQuery from '../queries/permissions.query.graphql';
const ROW_TYPES = { const ROW_TYPES = {
header: 'header', header: 'header',
......
<script> <script>
import filesQuery from 'shared_queries/repository/files.query.graphql';
import { deprecatedCreateFlash as createFlash } from '~/flash'; import { deprecatedCreateFlash as createFlash } from '~/flash';
import { __ } from '../../locale'; import { __ } from '../../locale';
import FileTable from './table/index.vue'; import FileTable from './table/index.vue';
import getRefMixin from '../mixins/get_ref'; import getRefMixin from '../mixins/get_ref';
import filesQuery from '../queries/files.query.graphql';
import projectPathQuery from '../queries/project_path.query.graphql'; import projectPathQuery from '../queries/project_path.query.graphql';
import FilePreview from './preview/index.vue'; import FilePreview from './preview/index.vue';
import { readmeFile } from '../utils/readme'; import { readmeFile } from '../utils/readme';
......
import Vue from 'vue'; import Vue from 'vue';
import PathLastCommitQuery from 'shared_queries/repository/path_last_commit.query.graphql';
import { escapeFileUrl } from '../lib/utils/url_utility'; import { escapeFileUrl } from '../lib/utils/url_utility';
import createRouter from './router'; import createRouter from './router';
import App from './components/app.vue'; import App from './components/app.vue';
...@@ -19,10 +18,6 @@ export default function setupVueRepositoryList() { ...@@ -19,10 +18,6 @@ export default function setupVueRepositoryList() {
const { dataset } = el; const { dataset } = el;
const { projectPath, projectShortPath, ref, escapedRef, fullName } = dataset; const { projectPath, projectShortPath, ref, escapedRef, fullName } = dataset;
const router = createRouter(projectPath, escapedRef); const router = createRouter(projectPath, escapedRef);
const pathRegex = /-\/tree\/[^/]+\/(.+$)/;
const matches = window.location.href.match(pathRegex);
const currentRoutePath = matches ? matches[1] : '';
apolloProvider.clients.defaultClient.cache.writeData({ apolloProvider.clients.defaultClient.cache.writeData({
data: { data: {
...@@ -48,28 +43,7 @@ export default function setupVueRepositoryList() { ...@@ -48,28 +43,7 @@ export default function setupVueRepositoryList() {
}, },
}); });
if (window.gl.startup_graphql_calls) { initLastCommitApp();
const query = window.gl.startup_graphql_calls.find(
call => call.operationName === 'pathLastCommit',
);
query.fetchCall
.then(res => res.json())
.then(res => {
apolloProvider.clients.defaultClient.writeQuery({
query: PathLastCommitQuery,
data: res.data,
variables: {
projectPath,
ref,
path: currentRoutePath,
},
});
})
.catch(() => {})
.finally(() => initLastCommitApp());
} else {
initLastCommitApp();
}
router.afterEach(({ params: { path } }) => { router.afterEach(({ params: { path } }) => {
setTitle(path, ref, fullName); setTitle(path, ref, fullName);
......
import filesQuery from '../queries/files.query.graphql'; import filesQuery from 'shared_queries/repository/files.query.graphql';
import getRefMixin from './get_ref'; import getRefMixin from './get_ref';
import projectPathQuery from '../queries/project_path.query.graphql'; import projectPathQuery from '../queries/project_path.query.graphql';
......
#import "~/graphql_shared/fragments/pageInfo.fragment.graphql" fragment PageInfo on PageInfo {
__typename
hasNextPage
hasPreviousPage
startCursor
endCursor
}
fragment TreeEntry on Entry { fragment TreeEntry on Entry {
__typename
id id
sha sha
name name
...@@ -16,10 +23,15 @@ query getFiles( ...@@ -16,10 +23,15 @@ query getFiles(
$nextPageCursor: String $nextPageCursor: String
) { ) {
project(fullPath: $projectPath) { project(fullPath: $projectPath) {
__typename
repository { repository {
__typename
tree(path: $path, ref: $ref) { tree(path: $path, ref: $ref) {
__typename
trees(first: $pageSize, after: $nextPageCursor) { trees(first: $pageSize, after: $nextPageCursor) {
__typename
edges { edges {
__typename
node { node {
...TreeEntry ...TreeEntry
webPath webPath
...@@ -30,7 +42,9 @@ query getFiles( ...@@ -30,7 +42,9 @@ query getFiles(
} }
} }
submodules(first: $pageSize, after: $nextPageCursor) { submodules(first: $pageSize, after: $nextPageCursor) {
__typename
edges { edges {
__typename
node { node {
...TreeEntry ...TreeEntry
webUrl webUrl
...@@ -42,7 +56,9 @@ query getFiles( ...@@ -42,7 +56,9 @@ query getFiles(
} }
} }
blobs(first: $pageSize, after: $nextPageCursor) { blobs(first: $pageSize, after: $nextPageCursor) {
__typename
edges { edges {
__typename
node { node {
...TreeEntry ...TreeEntry
mode mode
......
query getPermissions($projectPath: ID!) { query getPermissions($projectPath: ID!) {
project(fullPath: $projectPath) { project(fullPath: $projectPath) {
__typename
userPermissions { userPermissions {
__typename
pushCode pushCode
forkProject forkProject
createMergeRequestIn createMergeRequestIn
......
...@@ -25,7 +25,7 @@ ...@@ -25,7 +25,7 @@
}; };
gl.startup_graphql_calls = gl.startup_graphql_calls.map(call => ({ gl.startup_graphql_calls = gl.startup_graphql_calls.map(call => ({
operationName: call.query.match(/^query (.+)\(/)[1], ...call,
fetchCall: fetch(url, { fetchCall: fetch(url, {
...opts, ...opts,
credentials: 'same-origin', credentials: 'same-origin',
......
- current_route_path = request.fullpath.match(/-\/tree\/[^\/]+\/(.+$)/).to_a[1] - current_route_path = request.fullpath.match(/-\/tree\/[^\/]+\/(.+$)/).to_a[1]
- add_page_startup_graphql_call('repository/path_last_commit', { projectPath: @project.full_path, ref: current_ref, path: current_route_path }) - add_page_startup_graphql_call('repository/path_last_commit', { projectPath: @project.full_path, ref: current_ref, path: current_route_path || "" })
- add_page_startup_graphql_call('repository/permissions', { projectPath: @project.full_path })
- add_page_startup_graphql_call('repository/files', { nextPageCursor: "", pageSize: 100, projectPath: @project.full_path, ref: current_ref, path: current_route_path || "/"})
- breadcrumb_title _("Repository") - breadcrumb_title _("Repository")
- @content_class = "limit-container-width" unless fluid_layout - @content_class = "limit-container-width" unless fluid_layout
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment