Skip to content
代码片段 群组 项目
提交 7614b3e8 编辑于 作者: Avielle Wolfe's avatar Avielle Wolfe
浏览文件

Merge branch...

Merge branch '428949-frontend-make-catalog-resource-id-become-the-fullpath-of-a-resource' into 'master' 

Make Catalog resource ID become the fullpath of a resource

See merge request https://gitlab.com/gitlab-org/gitlab/-/merge_requests/137447



Merged-by: default avatarAvielle Wolfe <awolfe@gitlab.com>
Approved-by: default avatarAvielle Wolfe <awolfe@gitlab.com>
Approved-by: default avatarJose Ivan Vargas <jvargas@gitlab.com>
Reviewed-by: default avatarAvielle Wolfe <awolfe@gitlab.com>
Co-authored-by: default avatarFrédéric Caplette <fcaplette@gitlab.com>
No related branches found
No related tags found
无相关合并请求
显示
63 个添加43 个删除
...@@ -12,7 +12,7 @@ export default { ...@@ -12,7 +12,7 @@ export default {
GlTableLite, GlTableLite,
}, },
props: { props: {
resourceId: { resourcePath: {
type: String, type: String,
required: true, required: true,
}, },
...@@ -27,7 +27,7 @@ export default { ...@@ -27,7 +27,7 @@ export default {
query: getCiCatalogResourceComponents, query: getCiCatalogResourceComponents,
variables() { variables() {
return { return {
id: this.resourceId, fullPath: this.resourcePath,
}; };
}, },
update(data) { update(data) {
......
...@@ -14,7 +14,7 @@ export default { ...@@ -14,7 +14,7 @@ export default {
}, },
mixins: [glFeatureFlagsMixin()], mixins: [glFeatureFlagsMixin()],
props: { props: {
resourceId: { resourcePath: {
type: String, type: String,
required: true, required: true,
}, },
...@@ -31,10 +31,10 @@ export default { ...@@ -31,10 +31,10 @@ export default {
<template> <template>
<gl-tabs> <gl-tabs>
<gl-tab :title="$options.i18n.tabs.readme" lazy> <gl-tab :title="$options.i18n.tabs.readme" lazy>
<ci-resource-readme :resource-id="resourceId" /> <ci-resource-readme :resource-path="resourcePath" />
</gl-tab> </gl-tab>
<gl-tab v-if="glFeatures.ciCatalogComponentsTab" :title="$options.i18n.tabs.components" lazy> <gl-tab v-if="glFeatures.ciCatalogComponentsTab" :title="$options.i18n.tabs.components" lazy>
<ci-resource-components :resource-id="resourceId" <ci-resource-components :resource-path="resourcePath"
/></gl-tab> /></gl-tab>
</gl-tabs> </gl-tabs>
</template> </template>
......
...@@ -11,7 +11,7 @@ export default { ...@@ -11,7 +11,7 @@ export default {
}, },
directives: { SafeHtml }, directives: { SafeHtml },
props: { props: {
resourceId: { resourcePath: {
type: String, type: String,
required: true, required: true,
}, },
...@@ -26,7 +26,7 @@ export default { ...@@ -26,7 +26,7 @@ export default {
query: getCiCatalogResourceReadme, query: getCiCatalogResourceReadme,
variables() { variables() {
return { return {
id: this.resourceId, fullPath: this.resourcePath,
}; };
}, },
update(data) { update(data) {
......
...@@ -34,11 +34,17 @@ export default { ...@@ -34,11 +34,17 @@ export default {
authorProfileUrl() { authorProfileUrl() {
return this.latestVersion.author.webUrl; return this.latestVersion.author.webUrl;
}, },
detailsPageHref() { resourceId() {
return cleanLeadingSeparator(this.resource.webPath);
},
detailsPageResolved() {
return this.$router.resolve({ return this.$router.resolve({
name: CI_RESOURCE_DETAILS_PAGE_NAME, name: CI_RESOURCE_DETAILS_PAGE_NAME,
params: { id: this.entityId }, params: { id: this.resourceId },
}).href; });
},
detailsPageHref() {
return decodeURIComponent(this.detailsPageResolved.href);
}, },
entityId() { entityId() {
return getIdFromGraphQLId(this.resource.id); return getIdFromGraphQLId(this.resource.id);
...@@ -79,10 +85,8 @@ export default { ...@@ -79,10 +85,8 @@ export default {
// open a new tab. // open a new tab.
e.preventDefault(); e.preventDefault();
this.$router.push({ // Push to the decoded URL to avoid all the / being encoded
name: CI_RESOURCE_DETAILS_PAGE_NAME, this.$router.push({ path: decodeURIComponent(this.resourceId) });
params: { id: this.entityId },
});
}, },
}, },
}; };
......
...@@ -2,8 +2,7 @@ ...@@ -2,8 +2,7 @@
import { GlEmptyState } from '@gitlab/ui'; import { GlEmptyState } from '@gitlab/ui';
import { s__ } from '~/locale'; import { s__ } from '~/locale';
import { createAlert } from '~/alert'; import { createAlert } from '~/alert';
import { convertToGraphQLId } from '~/graphql_shared/utils'; import { cleanLeadingSeparator } from '~/lib/utils/url_utility';
import { CI_CATALOG_RESOURCE_TYPE } from '../../graphql/settings';
import getCatalogCiResourceDetails from '../../graphql/queries/get_ci_catalog_resource_details.query.graphql'; import getCatalogCiResourceDetails from '../../graphql/queries/get_ci_catalog_resource_details.query.graphql';
import getCatalogCiResourceSharedData from '../../graphql/queries/get_ci_catalog_resource_shared_data.query.graphql'; import getCatalogCiResourceSharedData from '../../graphql/queries/get_ci_catalog_resource_shared_data.query.graphql';
import CiResourceDetails from '../details/ci_resource_details.vue'; import CiResourceDetails from '../details/ci_resource_details.vue';
...@@ -28,7 +27,7 @@ export default { ...@@ -28,7 +27,7 @@ export default {
query: getCatalogCiResourceSharedData, query: getCatalogCiResourceSharedData,
variables() { variables() {
return { return {
id: this.graphQLId, fullPath: this.cleanFullPath,
}; };
}, },
update(data) { update(data) {
...@@ -43,7 +42,7 @@ export default { ...@@ -43,7 +42,7 @@ export default {
query: getCatalogCiResourceDetails, query: getCatalogCiResourceDetails,
variables() { variables() {
return { return {
id: this.graphQLId, fullPath: this.cleanFullPath,
}; };
}, },
update(data) { update(data) {
...@@ -56,8 +55,8 @@ export default { ...@@ -56,8 +55,8 @@ export default {
}, },
}, },
computed: { computed: {
graphQLId() { cleanFullPath() {
return convertToGraphQLId(CI_CATALOG_RESOURCE_TYPE, this.$route.params.id); return cleanLeadingSeparator(this.$route.params.id);
}, },
isLoadingDetails() { isLoadingDetails() {
return this.$apollo.queries.resourceAdditionalDetails.loading; return this.$apollo.queries.resourceAdditionalDetails.loading;
...@@ -103,7 +102,7 @@ export default { ...@@ -103,7 +102,7 @@ export default {
:pipeline-status="pipelineStatus" :pipeline-status="pipelineStatus"
:resource="resourceSharedData" :resource="resourceSharedData"
/> />
<ci-resource-details :resource-id="graphQLId" /> <ci-resource-details :resource-path="cleanFullPath" />
</div> </div>
</div> </div>
</template> </template>
fragment CatalogResourceFields on CiCatalogResource { fragment CatalogResourceFields on CiCatalogResource {
id id
webPath
icon icon
name name
description description
...@@ -15,5 +16,4 @@ fragment CatalogResourceFields on CiCatalogResource { ...@@ -15,5 +16,4 @@ fragment CatalogResourceFields on CiCatalogResource {
webUrl webUrl
} }
} }
webPath
} }
query getCiCatalogResourceComponents($id: CiCatalogResourceID!) { query getCiCatalogResourceComponents($fullPath: ID!) {
ciCatalogResource(id: $id) { ciCatalogResource(fullPath: $fullPath) {
id id
webPath
components @client { components @client {
nodes { nodes {
id id
......
query getCiCatalogResourceDetails($id: CiCatalogResourceID!) { query getCiCatalogResourceDetails($fullPath: ID!) {
ciCatalogResource(id: $id) { ciCatalogResource(fullPath: $fullPath) {
id id
webPath
openIssuesCount openIssuesCount
openMergeRequestsCount openMergeRequestsCount
versions(first: 1) { versions(first: 1) {
......
query getCiCatalogResourceReadme($id: CiCatalogResourceID!) { query getCiCatalogResourceReadme($fullPath: ID!) {
ciCatalogResource(id: $id) { ciCatalogResource(fullPath: $fullPath) {
id id
webPath
readmeHtml readmeHtml
} }
} }
#import "../fragments/catalog_resource.fragment.graphql" #import "../fragments/catalog_resource.fragment.graphql"
query getCiCatalogResourceSharedData($id: CiCatalogResourceID!) { query getCiCatalogResourceSharedData($fullPath: ID!) {
ciCatalogResource(id: $id) { ciCatalogResource(fullPath: $fullPath) {
...CatalogResourceFields ...CatalogResourceFields
} }
} }
...@@ -11,7 +11,8 @@ export const cacheConfig = { ...@@ -11,7 +11,8 @@ export const cacheConfig = {
ciCatalogResource(_, { args, toReference }) { ciCatalogResource(_, { args, toReference }) {
return toReference({ return toReference({
__typename: 'CiCatalogResource', __typename: 'CiCatalogResource',
id: args.id, // Webpath is the fullpath with a leading slash
webPath: `/${args.fullPath}`,
}); });
}, },
ciCatalogResources: { ciCatalogResources: {
...@@ -19,6 +20,9 @@ export const cacheConfig = { ...@@ -19,6 +20,9 @@ export const cacheConfig = {
}, },
}, },
}, },
CiCatalogResource: {
keyFields: ['webPath'],
},
}, },
}, },
}; };
......
...@@ -27,6 +27,9 @@ export const initCatalog = (selector = '#js-ci-cd-catalog') => { ...@@ -27,6 +27,9 @@ export const initCatalog = (selector = '#js-ci-cd-catalog') => {
name: 'GlobalCatalog', name: 'GlobalCatalog',
router: createRouter(ciCatalogPath, CiResourcesPage), router: createRouter(ciCatalogPath, CiResourcesPage),
apolloProvider, apolloProvider,
provide: {
ciCatalogPath,
},
render(h) { render(h) {
return h(GlobalCatalog); return h(GlobalCatalog);
}, },
......
...@@ -4,6 +4,6 @@ import { CI_RESOURCES_PAGE_NAME, CI_RESOURCE_DETAILS_PAGE_NAME } from './constan ...@@ -4,6 +4,6 @@ import { CI_RESOURCES_PAGE_NAME, CI_RESOURCE_DETAILS_PAGE_NAME } from './constan
export const createRoutes = (listComponent) => { export const createRoutes = (listComponent) => {
return [ return [
{ name: CI_RESOURCES_PAGE_NAME, path: '', component: listComponent }, { name: CI_RESOURCES_PAGE_NAME, path: '', component: listComponent },
{ name: CI_RESOURCE_DETAILS_PAGE_NAME, path: '/:id', component: CiResourceDetailsPage }, { name: CI_RESOURCE_DETAILS_PAGE_NAME, path: '/:id+', component: CiResourceDetailsPage },
]; ];
}; };
...@@ -23,7 +23,7 @@ def check_resource_access ...@@ -23,7 +23,7 @@ def check_resource_access
end end
def catalog_resource def catalog_resource
::Ci::Catalog::Listing.new(current_user).find_resource(id: params[:id]) ::Ci::Catalog::Listing.new(current_user).find_resource(full_path: params[:full_path])
end end
end end
end end
...@@ -30,8 +30,8 @@ def resources(namespace: nil, sort: nil, search: nil, scope: :all) ...@@ -30,8 +30,8 @@ def resources(namespace: nil, sort: nil, search: nil, scope: :all)
end end
end end
def find_resource(id:) def find_resource(id: nil, full_path: nil)
resource = Ci::Catalog::Resource.find_by_id(id) resource = id ? Ci::Catalog::Resource.find_by_id(id) : Project.find_by_full_path(full_path)&.catalog_resource
return unless resource.present? return unless resource.present?
return unless resource.published? return unless resource.published?
......
...@@ -31,12 +31,16 @@ class Resource < ::ApplicationRecord ...@@ -31,12 +31,16 @@ class Resource < ::ApplicationRecord
scope :order_by_latest_released_at_desc, -> { reorder(arel_table[:latest_released_at].desc.nulls_last) } scope :order_by_latest_released_at_desc, -> { reorder(arel_table[:latest_released_at].desc.nulls_last) }
scope :order_by_latest_released_at_asc, -> { reorder(arel_table[:latest_released_at].asc.nulls_last) } scope :order_by_latest_released_at_asc, -> { reorder(arel_table[:latest_released_at].asc.nulls_last) }
delegate :avatar_path, :star_count, to: :project delegate :avatar_path, :star_count, :full_path, to: :project
enum state: { draft: 0, published: 1 } enum state: { draft: 0, published: 1 }
before_create :sync_with_project before_create :sync_with_project
def to_param
full_path
end
def unpublish! def unpublish!
update!(state: :draft) update!(state: :draft)
end end
......
...@@ -11,7 +11,10 @@ ...@@ -11,7 +11,10 @@
end end
resources :groups, only: [:index] resources :groups, only: [:index]
resources :catalog, only: [:index, :show], constraints: { id: /\d+/ } scope :catalog do
get '/' => 'catalog#index', as: :catalog_index
get '/*full_path' => 'catalog#show', as: :catalog
end
resources :snippets, only: [:index] resources :snippets, only: [:index]
root to: 'projects#index' root to: 'projects#index'
end end
......
...@@ -114,11 +114,11 @@ ...@@ -114,11 +114,11 @@
let_it_be(:project) { create(:project, :repository, namespace: namespace) } let_it_be(:project) { create(:project, :repository, namespace: namespace) }
before do before do
visit explore_catalog_path(id: new_ci_resource["id"]) visit explore_catalog_path(new_ci_resource)
end end
context 'when the resource is published' do context 'when the resource is published' do
let_it_be(:new_ci_resource) { create(:ci_catalog_resource, :published, project: project) } let(:new_ci_resource) { create(:ci_catalog_resource, :published, project: project) }
it 'navigates to the details page' do it 'navigates to the details page' do
expect(page).to have_content('Go to the project') expect(page).to have_content('Go to the project')
...@@ -126,7 +126,7 @@ ...@@ -126,7 +126,7 @@
end end
context 'when the resource is not published' do context 'when the resource is not published' do
let_it_be(:new_ci_resource) { create(:ci_catalog_resource, project: project, state: :draft) } let(:new_ci_resource) { create(:ci_catalog_resource, project: project, state: :draft) }
it 'returns a 404' do it 'returns a 404' do
expect(page).to have_title('Not Found') expect(page).to have_title('Not Found')
......
...@@ -19,9 +19,9 @@ describe('CiResourceComponents', () => { ...@@ -19,9 +19,9 @@ describe('CiResourceComponents', () => {
const components = mockComponents.data.ciCatalogResource.components.nodes; const components = mockComponents.data.ciCatalogResource.components.nodes;
const resourceId = 'gid://gitlab/Ci::Catalog::Resource/1'; const resourcePath = 'twitter/project-1';
const defaultProps = { resourceId }; const defaultProps = { resourcePath };
const createComponent = async () => { const createComponent = async () => {
const handlers = [[getCiCatalogcomponentComponents, mockComponentsResponse]]; const handlers = [[getCiCatalogcomponentComponents, mockComponentsResponse]];
......
...@@ -8,7 +8,7 @@ describe('CiResourceDetails', () => { ...@@ -8,7 +8,7 @@ describe('CiResourceDetails', () => {
let wrapper; let wrapper;
const defaultProps = { const defaultProps = {
resourceId: 'gid://gitlab/Ci::Catalog::Resource/1', resourcePath: 'twitter/project-1',
}; };
const defaultProvide = { const defaultProvide = {
glFeatures: { ciCatalogComponentsTab: true }, glFeatures: { ciCatalogComponentsTab: true },
......
0% 加载中 .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册