Query.tsx 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. import { FC, useEffect, useState, useRef, useMemo, useContext } from 'react';
  2. import { TextField } from '@material-ui/core';
  3. import { useTranslation } from 'react-i18next';
  4. import { Parser } from '@json2csv/plainjs';
  5. import { rootContext } from '../../context/Root';
  6. import EmptyCard from '../../components/cards/EmptyCard';
  7. import icons from '../../components/icons/Icons';
  8. import CustomButton from '../../components/customButton/CustomButton';
  9. import AttuGrid from '../../components/grid/Grid';
  10. import { ToolBarConfig } from '../../components/grid/Types';
  11. import { getQueryStyles } from './Styles';
  12. import Filter from '../../components/advancedSearch';
  13. import { CollectionHttp } from '../../http/Collection';
  14. import { FieldHttp } from '../../http/Field';
  15. import { usePaginationHook } from '../../hooks/Pagination';
  16. // import { useTimeTravelHook } from '../../hooks/TimeTravel';
  17. import CopyButton from '../../components/advancedSearch/CopyButton';
  18. import DeleteTemplate from '../../components/customDialog/DeleteDialogTemplate';
  19. import CustomToolBar from '../../components/grid/ToolBar';
  20. // import { CustomDatePicker } from '../../components/customDatePicker/CustomDatePicker';
  21. import { saveAs } from 'file-saver';
  22. import { DataTypeStringEnum } from '../collections/Types';
  23. const Query: FC<{
  24. collectionName: string;
  25. }> = ({ collectionName }) => {
  26. const [fields, setFields] = useState<any[]>([]);
  27. const [expression, setExpression] = useState('');
  28. const [tableLoading, setTableLoading] = useState<any>();
  29. const [queryResult, setQueryResult] = useState<any>();
  30. const [selectedData, setSelectedData] = useState<any[]>([]);
  31. const [primaryKey, setPrimaryKey] = useState<{ value: string; type: string }>(
  32. { value: '', type: DataTypeStringEnum.Int64 }
  33. );
  34. const { setDialog, handleCloseDialog, openSnackBar } =
  35. useContext(rootContext);
  36. const VectorSearchIcon = icons.vectorSearch;
  37. const ResetIcon = icons.refresh;
  38. const { t: dialogTrans } = useTranslation('dialog');
  39. const { t: successTrans } = useTranslation('success');
  40. const { t: searchTrans } = useTranslation('search');
  41. const { t: collectionTrans } = useTranslation('collection');
  42. const { t: btnTrans } = useTranslation('btn');
  43. const { t: commonTrans } = useTranslation();
  44. const copyTrans = commonTrans('copy');
  45. const classes = getQueryStyles();
  46. // Format result list
  47. const queryResultMemo = useMemo(
  48. () =>
  49. queryResult?.map((resultItem: { [key: string]: any }) => {
  50. // Iterate resultItem keys, then format vector(array) items.
  51. const tmp = Object.keys(resultItem).reduce(
  52. (prev: { [key: string]: any }, item: string) => {
  53. switch (item) {
  54. case 'json':
  55. prev[item] = <div>{JSON.stringify(resultItem[item])}</div>;
  56. break;
  57. case 'vector':
  58. const list2Str = JSON.stringify(resultItem[item]);
  59. prev[item] = (
  60. <div className={classes.vectorTableCell}>
  61. <div>{list2Str}</div>
  62. <CopyButton
  63. label={copyTrans.label}
  64. value={list2Str}
  65. className={classes.copyBtn}
  66. />
  67. </div>
  68. );
  69. break;
  70. default:
  71. prev[item] = `${resultItem[item]}`;
  72. }
  73. return prev;
  74. },
  75. {}
  76. );
  77. return tmp;
  78. }),
  79. [queryResult, classes.vectorTableCell, classes.copyBtn, copyTrans.label]
  80. );
  81. const {
  82. pageSize,
  83. handlePageSize,
  84. currentPage,
  85. handleCurrentPage,
  86. total,
  87. data: result,
  88. order,
  89. orderBy,
  90. handleGridSort,
  91. } = usePaginationHook(queryResultMemo || []);
  92. const handlePageChange = (e: any, page: number) => {
  93. handleCurrentPage(page);
  94. };
  95. const getFields = async (collectionName: string) => {
  96. const schemaList = await FieldHttp.getFields(collectionName);
  97. const nameList = schemaList.map(v => ({
  98. name: v.name,
  99. type: v.data_type,
  100. }));
  101. const primaryKey = schemaList.find(v => v._isPrimaryKey === true)!;
  102. setPrimaryKey({ value: primaryKey['name'], type: primaryKey['data_type'] });
  103. setFields(nameList);
  104. };
  105. // Get fields at first or collection name changed.
  106. useEffect(() => {
  107. collectionName && getFields(collectionName);
  108. }, [collectionName]);
  109. const filterRef = useRef();
  110. const handleFilterReset = () => {
  111. const currentFilter: any = filterRef.current;
  112. currentFilter?.getReset();
  113. setExpression('');
  114. setTableLoading(null);
  115. setQueryResult(null);
  116. handleCurrentPage(0);
  117. };
  118. const handleFilterSubmit = (expression: string) => {
  119. setExpression(expression);
  120. handleQuery(expression);
  121. };
  122. const handleQuery = async (expr: string = '') => {
  123. setTableLoading(true);
  124. if (expr === '') {
  125. handleFilterReset();
  126. return;
  127. }
  128. try {
  129. const res = await CollectionHttp.queryData(collectionName, {
  130. expr: expr,
  131. output_fields: fields.map(i => i.name),
  132. // travel_timestamp: timeTravelInfo.timestamp,
  133. });
  134. const result = res.data;
  135. setQueryResult(result);
  136. } catch (err) {
  137. setQueryResult([]);
  138. } finally {
  139. setTableLoading(false);
  140. }
  141. };
  142. const handleSelectChange = (value: any) => {
  143. setSelectedData(value);
  144. };
  145. const handleDelete = async () => {
  146. await CollectionHttp.deleteEntities(collectionName, {
  147. expr: `${primaryKey.value} in [${selectedData
  148. .map(v =>
  149. primaryKey.type === DataTypeStringEnum.VarChar
  150. ? `"${v[primaryKey.value]}"`
  151. : v[primaryKey.value]
  152. )
  153. .join(',')}]`,
  154. });
  155. handleCloseDialog();
  156. openSnackBar(successTrans('delete', { name: collectionTrans('entites') }));
  157. handleQuery(expression);
  158. };
  159. const toolbarConfigs: ToolBarConfig[] = [
  160. {
  161. type: 'iconBtn',
  162. onClick: () => {
  163. setDialog({
  164. open: true,
  165. type: 'custom',
  166. params: {
  167. component: (
  168. <DeleteTemplate
  169. label={btnTrans('drop')}
  170. title={dialogTrans('deleteTitle', {
  171. type: collectionTrans('entites'),
  172. })}
  173. text={collectionTrans('deleteDataWarning')}
  174. handleDelete={handleDelete}
  175. />
  176. ),
  177. },
  178. });
  179. },
  180. label: collectionTrans('delete'),
  181. icon: 'delete',
  182. // tooltip: collectionTrans('deleteTooltip'),
  183. disabledTooltip: collectionTrans('deleteTooltip'),
  184. disabled: () => selectedData.length === 0,
  185. },
  186. {
  187. type: 'iconBtn',
  188. onClick: () => {
  189. try {
  190. const opts = {};
  191. const parser = new Parser(opts);
  192. const csv = parser.parse(queryResult);
  193. const csvData = new Blob([csv], {
  194. type: 'text/csv;charset=utf-8',
  195. });
  196. saveAs(csvData, 'milvus_query_result.csv');
  197. } catch (err) {
  198. console.error(err);
  199. }
  200. },
  201. label: '',
  202. icon: 'download',
  203. tooltip: collectionTrans('downloadTooltip'),
  204. disabledTooltip: collectionTrans('downloadDisabledTooltip'),
  205. disabled: () => !queryResult?.length,
  206. },
  207. ];
  208. return (
  209. <div className={classes.root}>
  210. <CustomToolBar toolbarConfigs={toolbarConfigs} />
  211. <div className={classes.toolbar}>
  212. <div className="left">
  213. <TextField
  214. className="textarea"
  215. InputProps={{
  216. classes: {
  217. root: 'textfield',
  218. multiline: 'multiline',
  219. },
  220. }}
  221. placeholder={collectionTrans('exprPlaceHolder')}
  222. value={expression}
  223. onChange={(e: React.ChangeEvent<{ value: unknown }>) => {
  224. setExpression(e.target.value as string);
  225. }}
  226. onKeyDown={e => {
  227. if (e.key === 'Enter') {
  228. // Do code here
  229. handleQuery(expression);
  230. e.preventDefault();
  231. }
  232. }}
  233. />
  234. <Filter
  235. ref={filterRef}
  236. title="Advanced Filter"
  237. fields={fields.filter(
  238. i =>
  239. i.type !== DataTypeStringEnum.FloatVector &&
  240. i.type !== DataTypeStringEnum.BinaryVector
  241. )}
  242. filterDisabled={false}
  243. onSubmit={handleFilterSubmit}
  244. showTitle={false}
  245. showTooltip={false}
  246. />
  247. {/* </div> */}
  248. {/* <CustomDatePicker
  249. label={timeTravelInfo.label}
  250. onChange={handleDateTimeChange}
  251. date={timeTravel}
  252. setDate={setTimeTravel}
  253. /> */}
  254. </div>
  255. <div className="right">
  256. <CustomButton
  257. className="btn"
  258. onClick={handleFilterReset}
  259. disabled={!expression}
  260. >
  261. <ResetIcon classes={{ root: 'icon' }} />
  262. {btnTrans('reset')}
  263. </CustomButton>
  264. <CustomButton
  265. variant="contained"
  266. disabled={!expression}
  267. onClick={() => handleQuery(expression)}
  268. >
  269. {btnTrans('query')}
  270. </CustomButton>
  271. </div>
  272. </div>
  273. {tableLoading || queryResult?.length ? (
  274. <AttuGrid
  275. toolbarConfigs={[]}
  276. colDefinitions={fields.map(i => ({
  277. id: i.name,
  278. align: 'left',
  279. disablePadding: false,
  280. label: i.name,
  281. }))}
  282. primaryKey={primaryKey.value}
  283. openCheckBox={true}
  284. isLoading={!!tableLoading}
  285. rows={result}
  286. rowCount={total}
  287. selected={selectedData}
  288. setSelected={handleSelectChange}
  289. page={currentPage}
  290. onChangePage={handlePageChange}
  291. rowsPerPage={pageSize}
  292. setRowsPerPage={handlePageSize}
  293. orderBy={orderBy}
  294. order={order}
  295. handleSort={handleGridSort}
  296. />
  297. ) : (
  298. <EmptyCard
  299. wrapperClass={`page-empty-card ${classes.emptyCard}`}
  300. icon={<VectorSearchIcon />}
  301. text={
  302. queryResult?.length === 0
  303. ? searchTrans('empty')
  304. : collectionTrans('startTip')
  305. }
  306. />
  307. )}
  308. </div>
  309. );
  310. };
  311. export default Query;