Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix the display and jump logic for recent assets #8136

Merged
merged 24 commits into from
Jan 13, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
4707fac
Hide global objects in recent assets when workspace enabled, and jump…
Kapian1234 Sep 11, 2024
66f2b68
Changeset file for PR #8136 created/updated
opensearch-changeset-bot[bot] Sep 11, 2024
5a07c11
Update recently-accessed-service
Kapian1234 Sep 12, 2024
8ff5a28
Merge branch 'recent_assets' of github.com:Kapian1234/OpenSearch-Dash…
Kapian1234 Sep 12, 2024
9a540d3
fix the boolean value
Kapian1234 Sep 12, 2024
3340ea8
add unit test and fix some bugs
Kapian1234 Sep 12, 2024
41a1ca4
rename test cases
Kapian1234 Sep 12, 2024
37db157
Pass workspaceEnabled to createRecentNavLink
Kapian1234 Sep 30, 2024
8fff3cf
resolve conflicts
Kapian1234 Oct 9, 2024
e4fe87c
Merge branch 'main' of https://github.com/opensearch-project/OpenSear…
Kapian1234 Oct 29, 2024
879917e
Merge branch 'main' into recent_assets
ruanyl Oct 30, 2024
ea1e2ab
Merge branch 'main' of https://github.com/opensearch-project/OpenSear…
Kapian1234 Nov 4, 2024
0d046b9
Update unit tests
Kapian1234 Nov 4, 2024
291e82d
Merge branch 'recent_assets' of github.com:Kapian1234/OpenSearch-Dash…
Kapian1234 Nov 4, 2024
07a6f74
Merge branch 'main' into recent_assets
SuZhou-Joe Nov 15, 2024
af72d86
Resolve conflicts
Kapian1234 Dec 19, 2024
9c0064e
Merge branch 'recent_assets' of github.com:Kapian1234/OpenSearch-Dash…
Kapian1234 Dec 19, 2024
7c28a46
Remove workspaceEnabled props in recent_items
Kapian1234 Dec 24, 2024
0c92d9e
Fix links at recent accessed items
Kapian1234 Dec 24, 2024
4c83a8c
Fix unit tests
Kapian1234 Dec 25, 2024
101c4b8
Update snapshots
Kapian1234 Dec 25, 2024
d6ee2f1
Merge branch 'main' into recent_assets
SuZhou-Joe Dec 27, 2024
4d375a1
Modify unit tests
Kapian1234 Jan 9, 2025
bef66a3
Merge branch 'recent_assets' of github.com:Kapian1234/OpenSearch-Dash…
Kapian1234 Jan 9, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions changelogs/fragments/8136.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
fix:
- Fix the display and jump logic for recent assets ([#8136](https://github.com/opensearch-project/OpenSearch-Dashboards/pull/8136))
2 changes: 1 addition & 1 deletion src/core/public/chrome/chrome_service.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -260,7 +260,7 @@ export class ChromeService {

const navControls = this.navControls.start();
const navLinks = this.navLinks.start({ application, http });
const recentlyAccessed = await this.recentlyAccessed.start({ http, workspaces });
const recentlyAccessed = await this.recentlyAccessed.start({ http, workspaces, application });
const docTitle = this.docTitle.start({ document: window.document });
const navGroup = await this.navGroup.start({
navLinks,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ class LocalStorageMock implements Storage {

describe('RecentlyAccessed#start()', () => {
let originalLocalStorage: Storage;
let workspaceEnabled = false;
beforeAll(() => {
originalLocalStorage = window.localStorage;

Expand All @@ -78,7 +79,19 @@ describe('RecentlyAccessed#start()', () => {
const getStart = async () => {
const http = httpServiceMock.createStartContract();
const workspaces = workspacesServiceMock.createStartContract();
const recentlyAccessed = await new RecentlyAccessedService().start({ http, workspaces });
const application = {
...jest.requireActual('../../application'),
capabilities: {
workspaces: {
enabled: workspaceEnabled,
},
},
};
const recentlyAccessed = await new RecentlyAccessedService().start({
http,
workspaces,
application,
});
return { http, recentlyAccessed, workspaces };
};

Expand Down Expand Up @@ -160,4 +173,23 @@ Array [
{ link: '/app/item1', label: 'Item 1', id: 'item1', workspaceId: 'foo' },
]);
});

it('should not get objects without related workspace when workspace enabled', async () => {
workspaceEnabled = true;

const { recentlyAccessed } = await getStart();
recentlyAccessed.add('/app/item1', 'Item 1', 'item1');
expect(recentlyAccessed.get()).toEqual([]);
});

it('should get objects with related workspace when workspace enabled', async () => {
workspaceEnabled = true;

const { recentlyAccessed, workspaces } = await getStart();
workspaces.currentWorkspaceId$.next('foo');
recentlyAccessed.add('/app/item1', 'Item 1', 'item1');
expect(recentlyAccessed.get()).toEqual([
{ link: '/app/item1', label: 'Item 1', id: 'item1', workspaceId: 'foo' },
]);
});
});
Original file line number Diff line number Diff line change
Expand Up @@ -29,11 +29,12 @@
*/

import { Observable } from 'rxjs';

import { map } from 'rxjs/operators';
import { PersistedLog } from './persisted_log';
import { createLogKey } from './create_log_key';
import { HttpSetup } from '../../http';
import { WorkspacesStart } from '../../workspace';
import { InternalApplicationStart } from '../../application';

/** @public */
export interface ChromeRecentlyAccessedHistoryItem {
Expand All @@ -50,16 +51,18 @@ export interface ChromeRecentlyAccessedHistoryItem {
interface StartDeps {
http: HttpSetup;
workspaces: WorkspacesStart;
application: InternalApplicationStart;
}

/** @internal */
export class RecentlyAccessedService {
async start({ http, workspaces }: StartDeps): Promise<ChromeRecentlyAccessed> {
async start({ http, workspaces, application }: StartDeps): Promise<ChromeRecentlyAccessed> {
const logKey = await createLogKey('recentlyAccessed', http.basePath.getBasePath());
const history = new PersistedLog<ChromeRecentlyAccessedHistoryItem>(logKey, {
maxLength: 20,
isEqual: (oldItem, newItem) => oldItem.id === newItem.id,
});
const workspaceEnabled = application.capabilities.workspaces.enabled;

return {
/** Adds a new item to the history. */
Expand All @@ -81,10 +84,16 @@ export class RecentlyAccessedService {
},

/** Gets the current array of history items. */
get: () => history.get(),
get: () => history.get().filter((item) => (workspaceEnabled ? !!item.workspaceId : true)),
Copy link
Member

@ruanyl ruanyl Sep 29, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm thinking, should the filtering logic be done in the component where the data is used?
@SuZhou-Joe @wanglam what do you think?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct me if I missing understanding. I remember we discuss this offline. All recent accessed items should applied the same filtering logic.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right, if we do the filtering logic in component, then we need to applied the same logic in multiple places.

But it occurs to me that we can use different storage key for workspace enabled / workspace disabled, then we do not need to use filter.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct me if I missing understanding. I remember we discuss this offline. All recent accessed items should applied the same filtering logic.

I see, I'm fine with it now.

But it occurs to me that we can use different storage key for workspace enabled / workspace disabled, then we do not need to use filter.

Sounds good to me, could a future improvement/refactor task


/** Gets an observable of the current array of history items. */
get$: () => history.get$(),
get$: () => {
return history.get$().pipe(
map((items) => {
return items.filter((item) => (workspaceEnabled ? !!item.workspaceId : true));
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we add some UT to test this filtering logic?

})
);
},
};
}
}
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,7 @@ function mockProps(branding = {}) {
customNavLink$: new BehaviorSubject(undefined),
branding,
logos: getLogos(branding, mockBasePath.serverBasePath),
workspaceEnabled: true,
};
}

Expand Down
5 changes: 4 additions & 1 deletion src/core/public/chrome/ui/header/collapsible_nav.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,7 @@ interface Props {
navigateToUrl: InternalApplicationStart['navigateToUrl'];
customNavLink$: Rx.Observable<ChromeNavLink | undefined>;
logos: Logos;
workspaceEnabled: boolean | undefined;
}

export function CollapsibleNav({
Expand All @@ -105,6 +106,7 @@ export function CollapsibleNav({
navigateToApp,
navigateToUrl,
logos,
workspaceEnabled,
...observables
}: Props) {
const navLinks = useObservable(observables.navLinks$, []).filter((link) => !link.hidden);
Expand Down Expand Up @@ -193,7 +195,8 @@ export function CollapsibleNav({
link,
navLinks,
basePath,
navigateToUrl
navigateToUrl,
!!workspaceEnabled
);

return {
Expand Down
1 change: 1 addition & 0 deletions src/core/public/chrome/ui/header/header.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -689,6 +689,7 @@ export function Header({
}}
customNavLink$={observables.customNavLink$}
logos={logos}
workspaceEnabled={application.capabilities.workspaces.enabled}
/>
)}
</header>
Expand Down
20 changes: 19 additions & 1 deletion src/core/public/chrome/ui/header/nav_link.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -100,9 +100,27 @@ describe('createRecentNavLink', () => {
},
mockNavLinks,
mockBasePath,
mockedNavigateToUrl
mockedNavigateToUrl,
true
);

expect(recentLink.href).toEqual('http://localhost/test/w/foo/app/foo');
});

it('create a recent link when workspace disabled', () => {
const recentLink = createRecentNavLink(
{
id: 'foo',
label: 'foo',
link: '/app/foo',
workspaceId: 'foo',
},
mockNavLinks,
mockBasePath,
mockedNavigateToUrl,
false
);

expect(recentLink.href).toEqual('http://localhost/test/app/foo');
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shall we need to add test case for workspace disabled?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added

});
});
12 changes: 8 additions & 4 deletions src/core/public/chrome/ui/header/nav_link.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -125,13 +125,17 @@ export function createRecentNavLink(
recentLink: ChromeRecentlyAccessedHistoryItem,
navLinks: ChromeNavLink[],
basePath: HttpStart['basePath'],
navigateToUrl: InternalApplicationStart['navigateToUrl']
navigateToUrl: InternalApplicationStart['navigateToUrl'],
workspaceEnabled: boolean = false
): RecentNavLink {
const { link, label, workspaceId } = recentLink;
const href = relativeToAbsolute(
basePath.prepend(formatUrlWithWorkspaceId(link, workspaceId || '', basePath), {
withoutClientBasePath: true,
})
basePath.prepend(
workspaceEnabled ? formatUrlWithWorkspaceId(link, workspaceId || '', basePath) : link,
{
withoutClientBasePath: true,
}
)
);
const navLink = navLinks.find((nl) => href.startsWith(nl.baseUrl));
let titleAndAriaLabel = label;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -241,7 +241,7 @@ export const RecentWork = (props: { core: CoreStart; workspaceEnabled?: boolean
...recentAccessItem.meta,
workspaceName: findWorkspace?.name,
updatedAt: moment(obj?.updated_at).valueOf(),
link: href,
link: workspaceEnabled ? href : link,
label: label || obj.meta.title,
};
})
Expand All @@ -254,7 +254,7 @@ export const RecentWork = (props: { core: CoreStart; workspaceEnabled?: boolean
} finally {
setIsLoading(false);
}
}, [core.http, currentWorkspace, recentAccessed, workspaceList]);
}, [core.http, currentWorkspace, recentAccessed, workspaceList, workspaceEnabled]);

useEffect(() => {
// reset to allOption
Expand Down
Loading