Spaces:
Paused
Paused
Update flare-ui/src/app/services/conversation-manager.service.ts
Browse files
flare-ui/src/app/services/conversation-manager.service.ts
CHANGED
|
@@ -456,7 +456,10 @@ export class ConversationManagerService implements OnDestroy {
|
|
| 456 |
const audioData = this.audioQueue.join('');
|
| 457 |
|
| 458 |
try {
|
| 459 |
-
|
|
|
|
|
|
|
|
|
|
| 460 |
const audioUrl = URL.createObjectURL(audioBlob);
|
| 461 |
|
| 462 |
// Update last message with audio URL
|
|
@@ -476,7 +479,20 @@ export class ConversationManagerService implements OnDestroy {
|
|
| 476 |
} catch (error) {
|
| 477 |
console.error('Error creating audio blob:', error);
|
| 478 |
this.audioQueue = []; // Clear queue on error
|
| 479 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 480 |
}
|
| 481 |
} catch (error) {
|
| 482 |
console.error('Error handling TTS audio:', error);
|
|
@@ -487,21 +503,25 @@ export class ConversationManagerService implements OnDestroy {
|
|
| 487 |
private base64ToBlob(base64: string, mimeType: string): Blob {
|
| 488 |
try {
|
| 489 |
// Clean base64 string - remove any whitespace or newlines
|
| 490 |
-
const cleanBase64 = base64.replace(
|
| 491 |
|
| 492 |
// Validate base64 string
|
| 493 |
if (!cleanBase64 || cleanBase64.length === 0) {
|
| 494 |
throw new Error('Empty base64 string');
|
| 495 |
}
|
| 496 |
|
| 497 |
-
// Check if
|
| 498 |
-
|
| 499 |
-
|
| 500 |
-
|
| 501 |
}
|
| 502 |
|
|
|
|
|
|
|
|
|
|
|
|
|
| 503 |
// Decode base64
|
| 504 |
-
const byteCharacters = atob(
|
| 505 |
const byteNumbers = new Array(byteCharacters.length);
|
| 506 |
|
| 507 |
for (let i = 0; i < byteCharacters.length; i++) {
|
|
@@ -512,6 +532,7 @@ export class ConversationManagerService implements OnDestroy {
|
|
| 512 |
return new Blob([byteArray], { type: mimeType });
|
| 513 |
} catch (error) {
|
| 514 |
console.error('Error converting base64 to blob:', error);
|
|
|
|
| 515 |
console.error('Base64 preview:', base64.substring(0, 100) + '...');
|
| 516 |
throw new Error('Failed to convert audio data');
|
| 517 |
}
|
|
|
|
| 456 |
const audioData = this.audioQueue.join('');
|
| 457 |
|
| 458 |
try {
|
| 459 |
+
// Default to MP3 if mime_type not provided
|
| 460 |
+
const mimeType = message['mime_type'] || 'audio/mpeg';
|
| 461 |
+
|
| 462 |
+
const audioBlob = this.base64ToBlob(audioData, mimeType);
|
| 463 |
const audioUrl = URL.createObjectURL(audioBlob);
|
| 464 |
|
| 465 |
// Update last message with audio URL
|
|
|
|
| 479 |
} catch (error) {
|
| 480 |
console.error('Error creating audio blob:', error);
|
| 481 |
this.audioQueue = []; // Clear queue on error
|
| 482 |
+
|
| 483 |
+
// Try direct blob creation as fallback
|
| 484 |
+
try {
|
| 485 |
+
const fallbackBlob = new Blob([atob(audioData)], { type: 'audio/mpeg' });
|
| 486 |
+
const audioUrl = URL.createObjectURL(fallbackBlob);
|
| 487 |
+
|
| 488 |
+
const messages = this.messagesSubject.value;
|
| 489 |
+
if (messages.length > 0 && messages[messages.length - 1].role === 'assistant') {
|
| 490 |
+
messages[messages.length - 1].audioUrl = audioUrl;
|
| 491 |
+
this.messagesSubject.next([...messages]);
|
| 492 |
+
}
|
| 493 |
+
} catch (fallbackError) {
|
| 494 |
+
console.error('Fallback audio creation also failed:', fallbackError);
|
| 495 |
+
}
|
| 496 |
}
|
| 497 |
} catch (error) {
|
| 498 |
console.error('Error handling TTS audio:', error);
|
|
|
|
| 503 |
private base64ToBlob(base64: string, mimeType: string): Blob {
|
| 504 |
try {
|
| 505 |
// Clean base64 string - remove any whitespace or newlines
|
| 506 |
+
const cleanBase64 = base64.replace(/[\s\n\r]/g, '');
|
| 507 |
|
| 508 |
// Validate base64 string
|
| 509 |
if (!cleanBase64 || cleanBase64.length === 0) {
|
| 510 |
throw new Error('Empty base64 string');
|
| 511 |
}
|
| 512 |
|
| 513 |
+
// Check if string contains MP3 header (ID3 tag)
|
| 514 |
+
if (cleanBase64.startsWith('SUQz')) {
|
| 515 |
+
// This is raw MP3 data, not base64 - likely a backend issue
|
| 516 |
+
console.log('Detected raw MP3 data instead of base64');
|
| 517 |
}
|
| 518 |
|
| 519 |
+
// Pad base64 string if necessary
|
| 520 |
+
const padding = cleanBase64.length % 4;
|
| 521 |
+
const paddedBase64 = padding ? cleanBase64 + '='.repeat(4 - padding) : cleanBase64;
|
| 522 |
+
|
| 523 |
// Decode base64
|
| 524 |
+
const byteCharacters = atob(paddedBase64);
|
| 525 |
const byteNumbers = new Array(byteCharacters.length);
|
| 526 |
|
| 527 |
for (let i = 0; i < byteCharacters.length; i++) {
|
|
|
|
| 532 |
return new Blob([byteArray], { type: mimeType });
|
| 533 |
} catch (error) {
|
| 534 |
console.error('Error converting base64 to blob:', error);
|
| 535 |
+
console.error('Base64 length:', base64.length);
|
| 536 |
console.error('Base64 preview:', base64.substring(0, 100) + '...');
|
| 537 |
throw new Error('Failed to convert audio data');
|
| 538 |
}
|