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

Remove excess response fetching on /training/intents page #744

Merged
merged 44 commits into from
Dec 13, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
6316b3b
Merge relevant from old branch
IgorKrupenja Dec 3, 2024
47d942c
Fix guard
IgorKrupenja Dec 3, 2024
b10dc8d
Fix response logic
IgorKrupenja Dec 3, 2024
37dfc75
Fix deps
IgorKrupenja Dec 3, 2024
06ecd85
Remove deupe code
IgorKrupenja Dec 3, 2024
5f6230e
TODOs
IgorKrupenja Dec 3, 2024
2340c4c
Revert unrelated changes
IgorKrupenja Dec 3, 2024
26f57fc
Clean up
IgorKrupenja Dec 3, 2024
c9750b9
Clean up
IgorKrupenja Dec 3, 2024
9f73066
Clean up
IgorKrupenja Dec 3, 2024
e916819
Fix deletion
IgorKrupenja Dec 3, 2024
b73c29f
Clean up
IgorKrupenja Dec 3, 2024
6e6b23e
Move entities
IgorKrupenja Dec 3, 2024
4faea19
Clean up
IgorKrupenja Dec 3, 2024
3ebaaf8
Clean up
IgorKrupenja Dec 3, 2024
f1fefc2
Fix adding example
IgorKrupenja Dec 3, 2024
df075da
Fix examples
IgorKrupenja Dec 3, 2024
ec52131
Clean up
IgorKrupenja Dec 3, 2024
048941e
Clean up
IgorKrupenja Dec 3, 2024
4596ee3
Clean up
IgorKrupenja Dec 3, 2024
802eed2
Fix rule
IgorKrupenja Dec 4, 2024
efad763
Fix editing example
IgorKrupenja Dec 4, 2024
4b4bd07
gsa
IgorKrupenja Dec 4, 2024
8786af7
Clean up
IgorKrupenja Dec 4, 2024
f785f2f
Clean up
IgorKrupenja Dec 4, 2024
5a11419
Clean up
IgorKrupenja Dec 4, 2024
871fc7d
Fix the issue
IgorKrupenja Dec 5, 2024
ef68b39
PoC without data mapping
IgorKrupenja Dec 6, 2024
705cb8f
PoC with poor format
IgorKrupenja Dec 6, 2024
9fa9066
PoC with better format
IgorKrupenja Dec 6, 2024
8e97646
Clean up
IgorKrupenja Dec 6, 2024
96153ef
Cle
IgorKrupenja Dec 6, 2024
c41897e
PoC component
IgorKrupenja Dec 6, 2024
3d770c8
Clean up
IgorKrupenja Dec 6, 2024
fe8ce3f
Fix name
IgorKrupenja Dec 6, 2024
3c154c6
Fix query
IgorKrupenja Dec 6, 2024
265107c
Fix fetch
IgorKrupenja Dec 6, 2024
5def279
Revert unrelated changes
IgorKrupenja Dec 6, 2024
6921388
Clean up
IgorKrupenja Dec 6, 2024
4e33c26
Clean up
IgorKrupenja Dec 6, 2024
52534ec
Clean up
IgorKrupenja Dec 9, 2024
28d2195
Clean up
IgorKrupenja Dec 10, 2024
ffc49b3
Merge base
IgorKrupenja Dec 11, 2024
3d691ec
Fixes based on PR feedback
IgorKrupenja Dec 12, 2024
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
6 changes: 6 additions & 0 deletions DSL/DMapper/hbs/get_response_by_intent_id.handlebars
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
{
"response": {
"name": "{{response.[0]._id}}",
"text": {{#if response.[0]._source.response.[0].condition}}"{{{response.[0]._source.response.[1].text}}}"{{else}}"{{{response.[0]._source.response.[0].text}}}"{{/if}}
}
}
1 change: 1 addition & 0 deletions DSL/OpenSearch/deploy-opensearch.sh
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ curl -XDELETE "$URL/responses?ignore_unavailable=true" -u "$AUTH" --insecure
curl -H "Content-Type: application/x-ndjson" -X PUT "$URL/responses" -ku "$AUTH" --data-binary "@fieldMappings/responses.json"
if $MOCK_ALLOWED; then curl -H "Content-Type: application/x-ndjson" -X PUT "$URL/responses/_bulk" -ku "$AUTH" --data-binary "@mock/responses.json"; fi
curl -L -X POST "$URL/_scripts/response-with-name-and-text" -H 'Content-Type: application/json' -H 'Cookie: customJwtCookie=test' --data-binary "@templates/response-with-name-and-text.json"
curl -L -X POST "$URL/_scripts/response" -H 'Content-Type: application/json' -H 'Cookie: customJwtCookie=test' --data-binary "@templates/response.json"

# intents
curl -XDELETE "$URL/intents?ignore_unavailable=true" -u "$AUTH" --insecure
Expand Down
23 changes: 23 additions & 0 deletions DSL/OpenSearch/templates/response.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
{
"script": {
"lang": "mustache",
"source": {
"size": 10000,
"query": {
"bool": {
"must": [
{
"query_string": {
"query": "{{response_name}}",
"default_field": "name"
}
}
]
}
}
},
"params": {
"response_name": ""
}
}
}
1 change: 0 additions & 1 deletion DSL/Ruuter.private/GET/rasa/intents/by-id.yml
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,6 @@ getServiceIntentConnections:
url: "[#TRAINING_RESQL]/get-service-intent-connections"
result: getServiceIntentConnectionsResult

# todo if not using full, maybe modify to return one value
getIntentListLastChanged:
call: http.post
args:
Expand Down
42 changes: 42 additions & 0 deletions DSL/Ruuter.private/GET/rasa/response-by-intent-id.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
declaration:
call: declare
version: 0.1
description: "Get response by intent name"
method: get
accepts: json
returns: json
namespace: training
allowlist:
params:
- field: intent
type: string
description: "Intent ID"

assign_values:
assign:
response_name: ${"utter_" + incoming.params.intent}

getResponsesWithNameAndText:
call: http.post
args:
url: "[#TRAINING_OPENSEARCH]/responses/_search/template"
body:
id: "response"
params:
response_name: ${response_name}
result: getResponsesResult

mapResponsesData:
call: http.post
args:
url: "[#TRAINING_DMAPPER]/hbs/training/get_response_by_intent_id"
headers:
type: "json"
body:
response: ${getResponsesResult.response.body.hits.hits}
result: responsesData
next: returnSuccess

returnSuccess:
return: ${responsesData.response.body.response}
next: end
2 changes: 1 addition & 1 deletion DSL/Ruuter.private/GET/rasa/responses-list.yml
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ mapResponsesData:
args:
url: "[#TRAINING_DMAPPER]/hbs/training/get_responses_list"
headers:
type: 'json'
type: "json"
body:
response: ${getResponsesResult.response.body.hits.hits}
result: responsesData
Expand Down
79 changes: 37 additions & 42 deletions GUI/src/pages/Training/Intents/IntentDetails.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -28,14 +28,14 @@ import LoadingDialog from 'components/LoadingDialog';
import useDocumentEscapeListener from 'hooks/useDocumentEscapeListener';
import { IntentWithExamplesCount } from 'types/intentWithExampleCounts';

interface ResponsesResponse
extends Array<{
name: string;
response: {
name: string;
text: string;
}[];
}> {}
interface Response {
name: string;
text: string;
}

interface ResponseResponse {
response: Response;
}

interface IntentResponse {
response: Intent;
Expand All @@ -58,8 +58,7 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
const [refreshing, setRefreshing] = useState(false);
const [showConnectToServiceModal, setShowConnectToServiceModal] = useState(false);
const [showDeleteModal, setShowDeleteModal] = useState(false);
const [intentResponseText, setIntentResponseText] = useState<string>('');
const [intentResponseName, setIntentResponseName] = useState<string>('');
const [response, setResponse] = useState<Response | null>(null);
const [intentRule, setIntentRule] = useState<string>('');

const queryClient = useQueryClient();
Expand All @@ -74,29 +73,13 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
setIntent(intentResponse.response);
// Also reset form states on choosing another intent from list
setEditingIntentTitle(null);
setIntentResponseText('');
}
}, [intentResponse]);

const { data: isPossibleToUpdateMark, refetch } = useQuery<boolean>({
queryKey: [`intents/is-marked-for-service?intent=${intentId}`],
});

const setIntentResponse = useCallback(
(responsesResponse: ResponsesResponse | undefined) => {
if (!responsesResponse) return;

const intentExistingResponse = responsesResponse[0].response.find(
(response: any) => `utter_${intentId}` === response.name
);
if (intentExistingResponse) {
setIntentResponseText(intentExistingResponse.text);
setIntentResponseName(intentExistingResponse.name);
}
},
[intentId]
);

const addIntentRule = useCallback(
(rulesResponse: RulesResponse | undefined) => {
if (!rulesResponse) return;
Expand All @@ -117,23 +100,27 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
setIntent(intentsResponse.response);
setSelectedIntent(intentsResponse.response);

const resonsesResponse = await queryClient.fetchQuery<ResponsesResponse>(['responses-list']);
setIntentResponse(resonsesResponse);
const responseResponse = await queryClient.fetchQuery<ResponseResponse>([
`response-by-intent-id?intent=${intentId}`,
]);
setResponse(responseResponse.response);

const rulesResponse = await queryClient.fetchQuery<RulesResponse>(['rules']);
addIntentRule(rulesResponse);
},
[addIntentRule, intentId, queryClient, setIntentResponse, setSelectedIntent]
[addIntentRule, intentId, queryClient, setResponse, setSelectedIntent]
);

// TODO: need to fetch response for the selected intent only
const { data: responsesResponse } = useQuery<ResponsesResponse>({
queryKey: ['responses-list'],
const { data: responseResponse } = useQuery<ResponseResponse>({
queryKey: [`response-by-intent-id?intent=${intentId}`],
});

useEffect(() => {
setIntentResponse(responsesResponse);
}, [responsesResponse, setIntentResponse]);
if (responseResponse?.response) setResponse(responseResponse.response);
}, [responseResponse]);

const responseText = response?.text ?? '';
const responseName = response?.name ?? '';

// TODO: need to fetch rules for the selected intent only
const { data: rulesResponse } = useQuery<RulesResponse>({
Expand Down Expand Up @@ -365,7 +352,10 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
setRefreshing(true);
},
onSuccess: async () => {
await queryClient.invalidateQueries({ queryKey: ['response-list'], refetchType: 'all' });
await queryClient.invalidateQueries({
queryKey: [`intents/is-marked-for-service?intent=${intentId}`],
refetchType: 'all',
});
toast.open({
type: 'success',
title: t('global.notification'),
Expand Down Expand Up @@ -409,17 +399,17 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
});

const handleIntentResponseSubmit = async () => {
if (intentResponseText === '' || !intent) return;
if (responseText === '' || !intent) return;

const intentId = intent.id;

addOrEditResponseMutation.mutate({
id: `utter_${intentId}`,
responseText: intentResponseText,
update: !!intentResponseName,
responseText,
update: !!responseName,
});

if (!intentResponseName) {
if (!responseName) {
addRuleMutation.mutate({
data: {
rule: `rule_${intentId}`,
Expand Down Expand Up @@ -471,7 +461,7 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
setRefreshing(true);
},
onSuccess: async () => {
await queryClient.invalidateQueries(['response-list']);
await queryClient.invalidateQueries([`intents/is-marked-for-service?intent=${intentId}`]);
await queryClient.invalidateQueries(['rules']);
toast.open({
type: 'success',
Expand Down Expand Up @@ -629,15 +619,20 @@ const IntentDetails: FC<IntentDetailsProps> = ({ intentId, setSelectedIntent, li
<h1>{t('training.intents.responseTitle')}</h1>
<FormTextarea
label={t('global.addNew')}
value={intentResponseText}
value={responseText}
name="intentResponse"
minRows={7}
maxRows={7}
placeholder={t('global.addNew') + '...' || ''}
hideLabel
maxLength={RESPONSE_TEXT_LENGTH}
showMaxLength
onChange={(e) => setIntentResponseText(e.target.value)}
onChange={(e) =>
setResponse({
name: response?.name ?? '',
text: e.target.value ?? '',
})
}
disableHeightResize
/>
</Track>
Expand Down