Skip to content

Commit

Permalink
refactor(js/plugins/ollama): clean up parseMessage
Browse files Browse the repository at this point in the history
  • Loading branch information
cabljac committed Dec 12, 2024
1 parent 225aa6e commit 0f50c41
Showing 1 changed file with 36 additions and 63 deletions.
99 changes: 36 additions & 63 deletions js/plugins/ollama/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -195,82 +195,55 @@ function ollamaModel(
);
}

/**
* Parses the Ollama response into a standardized MessageData format.
* @param {ChatResponse | GenerateResponse} response - The raw response from Ollama
* @param {ApiType} type - The type of API used (chat or generate)
* @param {GenerateRequest} input - The original request input
* @returns {MessageData} The parsed message data
* @throws {GenkitError} If the response format is invalid or parsing fails
*/
function parseMessage(
response: ChatResponse | GenerateResponse,
type: ApiType,
_type: ApiType,
input: GenerateRequest
): MessageData {
function isErrorResponse(resp: any): resp is ErrorResponse {
return 'error' in resp && typeof resp.error === 'string';
}
// Type guards
const isErrorResponse = (resp: any): resp is ErrorResponse =>
'error' in resp && typeof resp.error === 'string';
const isChatResponse = (resp: any): resp is ChatResponse => 'message' in resp;
const isGenerateResponse = (resp: any): resp is GenerateResponse =>
'response' in resp;

// Handle error responses first
if (isErrorResponse(response)) {
throw new Error(response.error);
}

function isChatResponse(resp: any): resp is ChatResponse {
return 'message' in resp;
}
// Get the text content based on response type
const content = isChatResponse(response)
? response.message.content
: isGenerateResponse(response)
? response.response
: null;

function isGenerateResponse(resp: any): resp is GenerateResponse {
return 'response' in resp;
}

// Handle JSON format if requested
if (input.output?.format === 'json' && input.output.schema) {
let rawContent;
if (isChatResponse(response)) {
try {
// Parse the content string into an object
const parsedContent = extractJson(response.message.content);
// Validate against the schema
rawContent = parsedContent;
} catch (e) {
throw new GenkitError({
message: 'Failed to parse structured response from Ollama model',
status: 'FAILED_PRECONDITION',
});
}
} else if (isGenerateResponse(response)) {
try {
const parsedContent = extractJson(response.response);
rawContent = parsedContent;
} catch (e) {
throw new GenkitError({
message: 'Failed to parse structured response from Ollama model',
status: 'FAILED_PRECONDITION',
});
}
} else {
throw new Error('Invalid response format');
}

return {
role:
type === 'chat' && isChatResponse(response)
? toGenkitRole(response.message.role)
: 'model',
content: [{ text: JSON.stringify(rawContent) }],
};
if (content === null) {
throw new GenkitError({
message: 'Invalid response format from Ollama model',
status: 'FAILED_PRECONDITION',
});
}

// Handle regular output
if (isChatResponse(response)) {
return {
role: toGenkitRole(response.message.role),
content: [{ text: response.message.content }],
};
} else if (isGenerateResponse(response)) {
return {
role: 'model',
content: [{ text: response.response }],
};
}
// Determine role for chat responses, default to 'model'
const role =
isChatResponse(response) && !input.output?.format
? toGenkitRole(response.message.role)
: 'model';

throw new GenkitError({
message: 'Invalid response format from Ollama model',
status: 'FAILED_PRECONDITION',
});
return {
role,
content: [{ text: content }],
};
}

function toOllamaRequest(
Expand Down

0 comments on commit 0f50c41

Please sign in to comment.