Skip to content

Commit

Permalink
Support for user defined additional filters
Browse files Browse the repository at this point in the history
  • Loading branch information
SaachiNayyer committed Nov 19, 2024
1 parent 20ef70b commit a5eccf1
Show file tree
Hide file tree
Showing 7 changed files with 154 additions and 2 deletions.
8 changes: 8 additions & 0 deletions .changeset/two-students-care.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
'@axis-backstage/plugin-jira-dashboard-backend': minor
---

Support for user defined additional filters
Issue https://github.com/AxisCommunications/backstage-plugins/issues/210

Signed-off-by: enaysaa <[email protected]>
22 changes: 22 additions & 0 deletions plugins/jira-dashboard-backend/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,28 @@ metadata:
jira.com/project-key: separate-jira-instance/my-project-key
```

### Custom Jira Filters

You can now define custom Jira filters directly in your `app-config.yaml` file. This allows you to create and display filters beyond the ones provided by the plugin.

#### Configuration

To add custom filters, use the `defaultFilters` property within a Jira instance configuration:

```yaml
jiraDashboard:
instances:
- name: my-jira-instance
# ... other configuration ...
defaultFilters:
- name: 'Open Bugs'
shortName: 'Bugs'
query: 'type = bug AND resolution = Unresolved ORDER BY updated DESC, priority DESC'
- name: 'Epics'
shortName: 'Epics'
query: 'type = epic AND resolution = Unresolved ORDER BY updated DESC, priority DESC'
```

#### Authentication examples and trouble shooting

Either "Basic Auth" or "Personal Acccess Tokens" can be used.
Expand Down
2 changes: 2 additions & 0 deletions plugins/jira-dashboard-backend/api-report.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
```ts
import { BackendFeatureCompat } from '@backstage/backend-plugin-api';
import { Filter } from '@axis-backstage/plugin-jira-dashboard-common';
import { JiraQueryResults } from '@axis-backstage/plugin-jira-dashboard-common';

// @public
Expand All @@ -12,6 +13,7 @@ export type ConfigInstance = {
headers: Record<string, string>;
baseUrl: string;
userEmailSuffix?: string;
defaultFilters?: Filter[];
};

// @public
Expand Down
44 changes: 44 additions & 0 deletions plugins/jira-dashboard-backend/src/config.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -116,4 +116,48 @@ describe('config', () => {
expect(instance2.headers).toEqual({ 'Other-Header': 'other value' });
expect(instance2.userEmailSuffix).toBe('@backstage2.com');
});

it('should handle defaultFilters config', () => {
const mockConfig = mockServices.rootConfig({
data: {
jiraDashboard: {
instances: [
{
name: 'default',
baseUrl: 'http://jira.com',
token: 'token',
defaultFilters: [
{
name: 'My Open Bugs',
shortName: 'MyBugs',
query: 'type = Bug AND resolution = Unresolved',
},
{
name: 'High Priority Issues',
shortName: 'HighPrio',
query: 'priority = "High"',
},
],
},
],
},
},
});

const jiraConfig = JiraConfig.fromConfig(mockConfig);
const instance = jiraConfig.getInstance();

expect(instance.defaultFilters).toEqual([
{
name: 'My Open Bugs',
shortName: 'MyBugs',
query: 'type = Bug AND resolution = Unresolved',
},
{
name: 'High Priority Issues',
shortName: 'HighPrio',
query: 'priority = "High"',
},
]);
});
});
21 changes: 21 additions & 0 deletions plugins/jira-dashboard-backend/src/config.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { ConflictError, ServiceUnavailableError } from '@backstage/errors';
import { RootConfigService } from '@backstage/backend-plugin-api';
import { Filter } from '@axis-backstage/plugin-jira-dashboard-common';

type Config = ReturnType<RootConfigService['getConfig']>;

Expand All @@ -25,13 +26,15 @@ export type ConfigInstance = {
headers: Record<string, string>;
baseUrl: string;
userEmailSuffix?: string;
defaultFilters?: Filter[];
};

const JIRA_CONFIG_BASE_URL = 'baseUrl';
const JIRA_CONFIG_TOKEN = 'token';
const JIRA_CONFIG_HEADERS = 'headers';
const JIRA_CONFIG_USER_EMAIL_SUFFIX = 'userEmailSuffix';
const JIRA_CONFIG_ANNOTATION = 'annotationPrefix';
const JIRA_FILTERS = 'defaultFilters';

export class JiraConfig {
private instances: Record<string, ConfigInstance> = {};
Expand Down Expand Up @@ -61,6 +64,13 @@ export class JiraConfig {
userEmailSuffix: inst.getOptionalString(
JIRA_CONFIG_USER_EMAIL_SUFFIX,
),
defaultFilters: inst
.getOptionalConfigArray(JIRA_FILTERS)
?.map(filterConfig => ({
name: filterConfig.getString('name'),
shortName: filterConfig.getString('shortName'),
query: filterConfig.getString('query'),
})),
};
});
} else {
Expand All @@ -70,6 +80,13 @@ export class JiraConfig {
headers: parseHeaders(jira.getOptionalConfig(JIRA_CONFIG_HEADERS)),
baseUrl: jira.getString(JIRA_CONFIG_BASE_URL),
userEmailSuffix: jira.getOptionalString(JIRA_CONFIG_USER_EMAIL_SUFFIX),
defaultFilters: jira
.getOptionalConfigArray(JIRA_FILTERS)
?.map(filterConfig => ({
name: filterConfig.getString('name'),
shortName: filterConfig.getString('shortName'),
query: filterConfig.getString('query'),
})),
};
}
}
Expand Down Expand Up @@ -110,4 +127,8 @@ export class JiraConfig {
const instance = this.forInstance(instanceName);
return instance.userEmailSuffix;
}
resolveDefaultFilters(instanceName: string): Filter[] | undefined {
const instance = this.forInstance(instanceName);
return instance.defaultFilters;
}
}
56 changes: 55 additions & 1 deletion plugins/jira-dashboard-backend/src/filters.test.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { getDefaultFiltersForUser } from './filters';
import { getAssigneUser, getDefaultFiltersForUser } from './filters';
import { mockServices } from '@backstage/backend-test-utils';
import { UserEntity } from '@backstage/catalog-model';
import { ConfigInstance, JiraConfig } from './config';
Expand All @@ -14,6 +14,18 @@ describe('getDefaultFiltersForUser', () => {
},
});
const instance = JiraConfig.fromConfig(mockConfig).getInstance();
const mockInstance: ConfigInstance = {
token: 'mock_token',
headers: {},
baseUrl: 'http://jira.com',
defaultFilters: [
{
name: 'My Open Bugs',
shortName: 'MyBugs',
query: 'type = Bug AND resolution = Unresolved',
},
],
};

const mockUserEntity: UserEntity = {
apiVersion: 'backstage.io/v1beta1',
Expand Down Expand Up @@ -61,4 +73,46 @@ describe('getDefaultFiltersForUser', () => {
const filters = getDefaultFiltersForUser(instance);
expect(filters).toHaveLength(2);
});

it('should include defaultFilters from config', () => {
const filters = getDefaultFiltersForUser(mockInstance, mockUserEntity);
expect(filters).toContainEqual(
expect.objectContaining({
name: 'My Open Bugs',
shortName: 'MyBugs',
query: 'type = Bug AND resolution = Unresolved',
}),
);
});

it('should handle empty defaultFilters in config', () => {
const filters = getDefaultFiltersForUser(mockInstance, mockUserEntity);
expect(filters).toHaveLength(3);
});

it('should correctly apply filterOnUser logic', () => {
const filters = getDefaultFiltersForUser(mockInstance, mockUserEntity);
expect(filters).toContainEqual(
expect.objectContaining({
query: expect.stringContaining(
`(assignee = "${getAssigneUser(
mockInstance,
mockUserEntity,
)}" OR "Additional Assignees" in ("${getAssigneUser(
mockInstance,
mockUserEntity,
)}")) AND type = Bug AND resolution = Unresolved`,
),
}),
);
});

it('should not apply filterOnUser if userEntity is missing', () => {
const filters = getDefaultFiltersForUser(mockInstance);
expect(filters).toContainEqual(
expect.objectContaining({
query: 'type = Bug AND resolution = Unresolved',
}),
);
});
});
3 changes: 2 additions & 1 deletion plugins/jira-dashboard-backend/src/filters.ts
Original file line number Diff line number Diff line change
Expand Up @@ -51,10 +51,11 @@ export const getDefaultFiltersForUser = (
incomingStatus?: string,
): Filter[] => {
const incomingFilter = getIncomingFilter(incomingStatus ?? 'New');
const defaultFilters = instance.defaultFilters || [];

if (!userEntity) return [openFilter, incomingFilter];

const assigneeToMeFilter = getAssignedToMeFilter(userEntity, instance);

return [openFilter, incomingFilter, assigneeToMeFilter];
return [openFilter, incomingFilter, assigneeToMeFilter, ...defaultFilters];
};

0 comments on commit a5eccf1

Please sign in to comment.