Json logic query builder fixes (#18819)

* minor workflow changes

* fix json logic fixes for extension field

* fix tests

* query builder ui changes

* minor add entity type in global search

* fix tests

* update certification ui

* update default icons

* minor add test id
This commit is contained in:
Karan Hotchandani 2024-11-28 13:12:46 +05:30 committed by GitHub
parent 609620e54f
commit b614594252
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 149 additions and 14 deletions

View File

@ -11,7 +11,7 @@
"description": "Bronze certified Data Asset.",
"style": {
"color": "#C08329",
"iconURL": ""
"iconURL": ""
}
},
{
@ -19,7 +19,7 @@
"description": "Silver certified Data Asset.",
"style": {
"color": "#ADADAD",
"iconURL": ""
"iconURL": ""
}
},
{
@ -27,7 +27,7 @@
"description": "Gold certified Data Asset.",
"style": {
"color": "#FFCE00",
"iconURL":""
"iconURL":""
}
}
]

View File

@ -10,10 +10,11 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { Tag } from 'antd';
import { Tag, Tooltip } from 'antd';
import React from 'react';
import { AssetCertification } from '../../../generated/entity/data/table';
import { getEntityName } from '../../../utils/EntityUtils';
import { getTagTooltip } from '../../../utils/TagsUtils';
import './certification-tag.less';
const CertificationTag = ({
@ -21,19 +22,34 @@ const CertificationTag = ({
}: {
certification: AssetCertification;
}) => {
if (certification.tagLabel.style?.iconURL) {
const name = getEntityName(certification.tagLabel);
return (
<Tooltip
className="cursor-pointer"
title={getTagTooltip(name, certification.tagLabel.description)}
trigger="hover">
<div data-testid={`certification-${certification.tagLabel.tagFQN}`}>
<img
alt="certification"
src={certification.tagLabel.style?.iconURL}
/>
</div>
</Tooltip>
);
}
return (
<Tag
className="certification-tag"
data-testid={`certification-${certification.tagLabel.tagFQN}`}
style={{
borderColor: certification.tagLabel.style?.color,
backgroundColor: certification.tagLabel.style?.color
? `${certification.tagLabel.style.color}33`
: undefined, // Assuming 33 is the hex transparency for lighter shade
}}>
{certification.tagLabel.style?.iconURL && (
<img alt="certification" src={certification.tagLabel.style?.iconURL} />
)}
{getEntityName(certification.tagLabel)}
</Tag>
);

View File

@ -12,7 +12,16 @@
*/
import { InfoCircleOutlined } from '@ant-design/icons';
import { WidgetProps } from '@rjsf/utils';
import { Alert, Button, Card, Col, Row, Skeleton, Typography } from 'antd';
import {
Alert,
Button,
Card,
Col,
Divider,
Row,
Skeleton,
Typography,
} from 'antd';
import classNames from 'classnames';
import { t } from 'i18next';
import { debounce, isEmpty, isUndefined } from 'lodash';
@ -29,6 +38,10 @@ import {
import { getExplorePath } from '../../../../../../constants/constants';
import { EntityType } from '../../../../../../enums/entity.enum';
import { SearchIndex } from '../../../../../../enums/search.enum';
import {
EsBoolQuery,
QueryFieldInterface,
} from '../../../../../../pages/ExplorePage/ExplorePage.interface';
import { searchQuery } from '../../../../../../rest/searchAPI';
import {
elasticSearchFormat,
@ -124,6 +137,22 @@ const QueryBuilderWidget: FC<WidgetProps> = ({
query: data,
};
if (data) {
if (entityType !== EntityType.ALL) {
// Scope the search to the passed entity type
if (
Array.isArray(
((qFilter.query as QueryFieldInterface)?.bool as EsBoolQuery)
?.must
)
) {
(
(qFilter.query as QueryFieldInterface)?.bool
?.must as QueryFieldInterface[]
)?.push({
term: { entityType: entityType },
});
}
}
debouncedFetchEntityCount(qFilter);
}
@ -208,7 +237,19 @@ const QueryBuilderWidget: FC<WidgetProps> = ({
data-testid="query-builder-form-field">
<Card className={classNames('query-builder-card', outputType)}>
<Row gutter={[8, 8]}>
<Col className="p-t-sm" span={24}>
<Col
className={classNames({
'p-t-sm': outputType === SearchOutputType.ElasticSearch,
})}
span={24}>
{outputType === SearchOutputType.JSONLogic && (
<>
<Typography.Text className="query-filter-label text-grey-muted">
{props.label}
</Typography.Text>
<Divider className="m-y-sm" />
</>
)}
<Query
{...config}
renderBuilder={(props) => (

View File

@ -26,6 +26,29 @@
}
}
.query-builder-form-field
.query-builder-card.jsonlogic
.query-builder-container {
.group-or-rule-container.group-container {
& > .group.group-or-rule {
& > .group--header {
order: 0;
.action.action--ADD-RULE {
position: absolute !important;
margin-top: 0;
right: 0;
top: -48px;
}
}
}
}
}
.query-filter-label {
text-transform: capitalize;
}
.query-builder-form-field {
.hide--line.one--child {
margin-top: 0;
@ -176,6 +199,10 @@
}
}
}
.rule {
align-items: center;
}
}
.query-builder-card.elasticsearch {

View File

@ -223,7 +223,7 @@ const AppInstall = () => {
);
case 3:
return (
<div className="w-500 p-md border rounded-4">
<div className="w-3/5 p-md border rounded-4">
<Typography.Title level={5}>{t('label.schedule')}</Typography.Title>
<ScheduleInterval
defaultSchedule={defaultValue}

View File

@ -46,6 +46,7 @@ describe('AdvancedSearchClassBase', () => {
EntityFields.TIER,
'extension',
'descriptionStatus',
'entityType',
]);
});
});

View File

@ -610,6 +610,19 @@ class AdvancedSearchClassBase {
},
},
},
[EntityFields.ENTITY_TYPE]: {
label: t('label.entity-type-plural', { entity: t('label.entity') }),
type: 'select',
mainWidgetProps: this.mainWidgetProps,
fieldSettings: {
asyncFetch: this.autocomplete({
searchIndex: entitySearchIndex,
entityField: EntityFields.ENTITY_TYPE,
}),
useAsyncSearch: true,
},
},
};
}

View File

@ -29,7 +29,7 @@ import {
import { SearchIndex } from '../enums/search.enum';
import { searchData } from '../rest/miscAPI';
import advancedSearchClassBase from './AdvancedSearchClassBase';
import { renderQueryBuilderFilterButtons } from './QueryBuilderUtils';
import { renderJSONLogicQueryBuilderButtons } from './QueryBuilderUtils';
class JSONLogicSearchClassBase {
baseConfig = AntdConfig as Config;
@ -370,7 +370,7 @@ class JSONLogicSearchClassBase {
operatorLabel: t('label.condition') + ':',
showNot: false,
valueLabel: t('label.criteria') + ':',
renderButton: renderQueryBuilderFilterButtons,
renderButton: renderJSONLogicQueryBuilderButtons,
},
};

View File

@ -10,7 +10,7 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import { CloseOutlined } from '@ant-design/icons';
import { CloseOutlined, PlusOutlined } from '@ant-design/icons';
import { Button } from 'antd';
import { t } from 'i18next';
import { isUndefined } from 'lodash';
@ -404,6 +404,43 @@ export const renderQueryBuilderFilterButtons: RenderSettings['renderButton'] = (
return <></>;
};
export const renderJSONLogicQueryBuilderButtons: RenderSettings['renderButton'] =
(props) => {
const type = props?.type;
if (type === 'delRule') {
return (
<Button
className="action action--DELETE ant-btn-sm"
data-testid="delete-condition-button"
icon={<CloseOutlined width={14} />}
onClick={props?.onClick}
/>
);
} else if (type === 'delRuleGroup') {
return (
<Button
className="action action--DELETE-GROUP ant-btn-sm"
data-testid="delete-group-condition-button"
icon={<CloseOutlined width={14} />}
onClick={props?.onClick}
/>
);
} else if (type === 'addRule') {
return (
<Button
className="action action--ADD-RULE ant-btn-sm"
data-testid="add-condition-button"
icon={<PlusOutlined width={14} />}
type="primary"
onClick={props?.onClick}
/>
);
}
return <></>;
};
interface ElasticsearchQuery {
bool?: {
must?: ElasticsearchQuery[];