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

feat: Implement text file upload functionality. 增加上传文本文件的按钮和功能 #5585

Open
wants to merge 17 commits into
base: main
Choose a base branch
from

Conversation

Dakai
Copy link
Contributor

@Dakai Dakai commented Oct 6, 2024

💻 变更类型 | Change Type

  • feat
  • fix
  • refactor
  • perf
  • style
  • test
  • docs
  • ci
  • chore
  • build

🔀 变更说明 | Description of Change

在工具栏处增加了一个上传文本文件的按钮,文件内容会复制在用户输入的内容之后,以这样的格式发出:

// .... User Input \n
Here are the files:\n
{file 1 name}\n
{file 1 content}\n
{file 2 name)\n
{file 2 content}\n
// ...etc.

📝 补充信息 | Additional Information

文件在上传后会显示一个粗略的 token 计数,以方便用户使用。
上传的文件会排在上传的图片右边,风格和大小和图片上传一致,过长的文件名部分会被隐藏。

2024-10-06_11-19
2024-10-06_11-19_1

Summary by CodeRabbit

Release Notes

  • New Features

    • Introduced support for file uploads in the chat interface, allowing users to attach text files alongside images.
    • Added new file handling capabilities, including reading file content and counting tokens.
  • Improvements

    • Enhanced styling for chat attachments, providing a clearer layout for images and files.
    • Updated authorization handling for API requests based on different service providers.
  • Bug Fixes

    • Refined logic for managing attached images and files in user messages.
  • Dependencies

    • Added new dependencies for file icon handling.

Copy link

vercel bot commented Oct 6, 2024

@Dakai is attempting to deploy a commit to the NextChat Team on Vercel.

A member of the Team first needs to authorize it.

Copy link
Contributor

coderabbitai bot commented Oct 6, 2024

Walkthrough

The pull request introduces significant updates to the handling of multimodal content and file uploads within the chat application. Key changes include the enhancement of the MultimodalContent interface to accommodate a new file_url type and the addition of a new UploadFile interface. The chat components have been modified to support file uploads, with new methods and state management for handling attachments. Styling updates in the SCSS file improve the organization of attachment elements, while utility functions for file handling have been added. Additionally, the package.json file reflects new dependencies related to file icons.

Changes

File Change Summary
app/client/api.ts - Updated MultimodalContent interface to include file_url.
- Added UploadFile interface.
- Updated getHeaders function for authorization handling.
app/components/chat.module.scss - Renamed .attach-images to .attachments.
- Added classes for file attachments: .attach-files, .attach-file, and variants for file names.
app/components/chat.tsx - Added uploadDocument method for file uploads.
- Updated ChatActions and state management for file attachments.
- Modified rendering logic for files.
app/store/chat.ts - Updated onUserInput method to handle attachFiles.
- Refined logic for concatenating attached images and files.
app/utils.ts - Added readFileContent, countTokens, and getMessageFiles functions for file handling.
- Updated imports for UploadFile.
package.json - Added dependencies: @types/react-file-icon and react-file-icon.
- Retained mermaid dependency.

Possibly related PRs

  • [Feature] Stable Diffusion #4983: The changes in the main PR regarding the MultimodalContent interface and the addition of the UploadFile interface relate to the modifications in app/client/api.ts in this PR, which also involves changes to the API structure.
  • Chore: Remove Unused Imports and Add ESLint Rules #5112: This PR modifies the app/client/api.ts file, which is relevant to the main PR as both involve changes to the API structure and functionality.
  • Feature/tencent #5157: The changes in this PR include modifications to the app/client/api.ts file, which is relevant to the main PR as both involve enhancements to API capabilities.
  • reduce cloudflare functions build size #5190: The changes in this PR include modifications to the app/utils.ts file, which may relate to the main PR's enhancements in handling multimodal content and file uploads.
  • feat: add a page to search chat history #5274: This PR introduces changes to the app/client/api.ts file, which is relevant to the main PR as both involve enhancements to API functionalities.
  • Feature plugin (GPTs like action based on function call) #5331: The changes in this PR involve modifications to the app/client/api.ts file, which is relevant to the main PR as both involve enhancements to API capabilities.
  • feat: add shortcut key #5396: The changes in this PR involve modifications to the app/components/chat.tsx file, which is relevant to the main PR as both involve enhancements to user interaction and functionality.
  • add tts #5459: The changes in this PR involve modifications to the app/client/api.ts file, which is relevant to the main PR as both involve enhancements to API functionalities.
  • feat: allow send image only #5599: The changes in this PR involve modifications to the app/components/chat.tsx file, which is relevant to the main PR as both involve enhancements to user interaction and functionality.
  • Feature/h0lly w00dz z updater #5632: The changes in this PR involve modifications to the app/components/settings.tsx file, which may relate to the main PR's enhancements in handling multimodal content and file uploads.

Suggested labels

enhancement

Suggested reviewers

  • Dogtiti

Poem

In the chat where we share and play,
Files and images come out to stay.
With icons bright and styles anew,
Upload your docs, it's easy to do!
A hop, a skip, let's chat away,
Our files are here, hip-hip-hooray! 🐇✨


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

@Dakai Dakai changed the title 增加上传文本文件的按钮和功能 feat: Implement text file upload functionality. 增加上传文本文件的按钮和功能 Oct 6, 2024
Copy link
Contributor

github-actions bot commented Oct 6, 2024

Your build has completed!

Preview deployment

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 12

🧹 Outside diff range and nitpick comments (4)
app/components/chat.module.scss (2)

12-14: Remove unnecessary commented-out code

The lines 12-14 contain commented-out position styles. If these styles are no longer needed, it's better to remove them to keep the stylesheet clean and maintainable.


54-56: Clean up commented-out code in .attach-files class

The lines 54-56 and 59 have commented-out code. Removing these unused lines will improve readability and maintain the cleanliness of the stylesheet.

Also applies to: 59-59

app/components/chat.tsx (2)

1492-1502: Rename imagesData to uploadedFiles for clarity

The variable imagesData is used to store uploaded text files, which might be confusing. Rename it to uploadedFiles to better reflect its purpose and improve code readability.

Apply this diff:

- const imagesData: UploadFile[] = [];
+ const uploadedFiles: UploadFile[] = [];
...
- imagesData.push(fileData);
+ uploadedFiles.push(fileData);
...
- if (
-   imagesData.length === 3 ||
-   imagesData.length === files.length
- ) {
+ if (
+   uploadedFiles.length === 3 ||
+   uploadedFiles.length === selectedFiles.length
+ ) {

591-591: Use localized strings for button text

The button text is hardcoded as "Upload Plain Text File". To support localization and maintain consistency, please use the Locale object to fetch the string.

Apply this diff:

- text={"Upload Plain Text File"}
+ text={Locale.Chat.InputActions.UploadFile}

Also, add the corresponding entry in the localization files if it doesn't exist.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 05e6e4b and 1f2dea4.

⛔ Files ignored due to path filters (2)
  • app/icons/upload-doc.svg is excluded by !**/*.svg
  • yarn.lock is excluded by !**/yarn.lock, !**/*.lock
📒 Files selected for processing (6)
  • app/client/api.ts (1 hunks)
  • app/components/chat.module.scss (4 hunks)
  • app/components/chat.tsx (12 hunks)
  • app/store/chat.ts (5 hunks)
  • app/utils.ts (3 hunks)
  • package.json (3 hunks)
🔇 Additional comments (10)
package.json (4)

25-25: LGTM: Addition of "@types/react-file-icon" type definitions.

Adding TypeScript type definitions for the "react-file-icon" library is a good practice. It will improve type checking and developer experience when using the library.


Line range hint 25-43: Summary of package.json changes

The changes to package.json mostly align with the PR objectives of adding file upload functionality:

  1. Added "@types/react-file-icon" for improved TypeScript support.
  2. Added "react-file-icon" for displaying file icons.
  3. Retained "mermaid" library (clarification needed).

These additions support the implementation of the new file upload feature. However, please provide clarification on the retention of the "mermaid" library as requested in the previous comment.


36-36: Please clarify the retention of the "mermaid" dependency.

The "mermaid" library (version ^10.6.1) has been retained in this PR. However, its retention doesn't seem directly related to the PR objectives of adding file upload functionality. Could you please clarify why this dependency was removed and then re-added without any version change? Is it necessary for the current feature implementation or future plans?

Let's check the usage of mermaid in the codebase:


43-43: LGTM: Addition of "react-file-icon" library.

The addition of the "react-file-icon" library aligns with the PR objectives of implementing file upload functionality. This library will likely be used to display file icons for uploaded text files.

Let's verify the usage of this library in the codebase:

✅ Verification successful

Verified: "react-file-icon" library correctly added and used.

The "react-file-icon" library is successfully imported and utilized within app/components/chat.tsx for displaying file icons in the chat component.

  • File: app/components/chat.tsx
    • Import: import { FileIcon, defaultStyles } from "react-file-icon";
    • Usage: <FileIcon {...style} glyphColor="#303030" />
🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify the usage of react-file-icon in the codebase

# Test: Search for import statements of react-file-icon
echo "Searching for import statements of react-file-icon:"
rg --type typescript --type tsx 'from [\'"]react-file-icon[\'"]'

# Test: Search for FileIcon component usage
echo "Searching for FileIcon component usage:"
rg --type typescript --type tsx '<FileIcon'

Length of output: 207


Script:

#!/bin/bash
# Description: Re-verify the usage of react-file-icon in the codebase with corrected syntax

# Test: Search for import statements of react-file-icon
echo "Searching for import statements of react-file-icon:"
rg --type typescript --type tsx "from ['\"]react-file-icon['\"]"

# Test: Search for FileIcon component usage
echo "Searching for FileIcon component usage:"
rg --type typescript --type tsx "<FileIcon"

Length of output: 388


Script:

#!/bin/bash
# Description: Re-verify the usage of react-file-icon in the codebase using file extensions

# Test: Search for import statements of react-file-icon
echo "Searching for import statements of react-file-icon in .ts and .tsx files:"
fd --extension ts --extension tsx --type f | xargs rg 'from [\'"]react-file-icon[\'"]'

# Test: Search for FileIcon component usage
echo "Searching for FileIcon component usage in .ts and .tsx files:"
fd --extension ts --extension tsx --type f | xargs rg '<FileIcon'

Length of output: 251


Script:

#!/bin/bash
# Description: Re-verify the usage of react-file-icon in the codebase using corrected syntax

# Test: Search for import statements of react-file-icon
echo "Searching for import statements of react-file-icon in .ts and .tsx files:"
fd --extension ts --extension tsx --type f | xargs rg "from ['\"]react-file-icon['\"]"

# Test: Search for FileIcon component usage
echo "Searching for FileIcon component usage in .ts and .tsx files:"
fd --extension ts --extension tsx --type f | xargs rg "<FileIcon"

Length of output: 812

app/components/chat.module.scss (2)

110-112: Verify the hover effect on .attach-file-icon

Setting opacity: 0 on hover for .attach-file-icon will make the icon invisible when hovered over. Is this the intended behavior? If not, consider adjusting the hover effect to enhance user experience.


567-574: New styles for file attachments look good

The styles for .chat-message-item-files and its child elements are well-structured and should enhance the display of file attachments within chat messages.

app/store/chat.ts (3)

9-9: Import statement added correctly

The UploadFile type is correctly imported from the client API module.


22-22: Import includes readFileContent

The readFileContent function is properly imported from the utils module.


448-449: Verify content property type compatibility

You're assigning displayContent to savedUserMessage.content, but displayContent is of type string | MultimodalContent[]. Ensure that the content property of ChatMessage can accept this type without causing type errors or runtime issues. If necessary, update the ChatMessage type definition to accommodate this change.

app/components/chat.tsx (1)

104-105: Verify react-file-icon dependency is included

The code imports react-file-icon, but please ensure that react-file-icon is added to package.json under dependencies to prevent runtime errors.

Run the following script to verify:

✅ Verification successful

react-file-icon dependency is correctly included in package.json.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check if 'react-file-icon' is listed in dependencies in package.json

grep '"react-file-icon"' package.json

Length of output: 72

app/client/api.ts Outdated Show resolved Hide resolved
app/utils.ts Outdated Show resolved Hide resolved
app/utils.ts Outdated Show resolved Hide resolved
app/utils.ts Outdated Show resolved Hide resolved
app/components/chat.module.scss Outdated Show resolved Hide resolved
app/store/chat.ts Outdated Show resolved Hide resolved
app/components/chat.tsx Outdated Show resolved Hide resolved
app/components/chat.tsx Outdated Show resolved Hide resolved
app/components/chat.tsx Outdated Show resolved Hide resolved
app/components/chat.tsx Outdated Show resolved Hide resolved
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 5

🧹 Outside diff range and nitpick comments (4)
app/client/api.ts (1)

File upload functionality is not implemented as per the PR objectives

The expected implementations for the file upload button and displaying uploaded files were not found in the codebase. While the interfaces MultimodalContent and UploadFile are defined and used, the actual UI components and functionality for uploading and displaying files are missing.

  • File upload button is not present.
  • Displaying of uploaded files is not implemented.
  • Token count calculation exists but is unrelated to the upload feature.
🔗 Analysis chain

Line range hint 1-47: Verify implementation of file upload functionality

While the changes to the MultimodalContent interface and the addition of the UploadFile interface align with the PR objectives, the actual implementation of the file upload functionality (such as the upload button and displaying uploaded files) is not visible in this file.

Could you please verify if the implementation exists in other files? It's important to ensure that:

  1. The file upload button is implemented as described in the PR objectives.
  2. The uploaded files are correctly displayed to the right of any uploaded images.
  3. The token count is calculated and displayed for uploaded files.
  4. The new interfaces (MultimodalContent and UploadFile) are correctly used in the implementation.

To help verify the implementation, you can run the following script:

This script will help locate the relevant implementations across the codebase.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for file upload related implementations
echo "Searching for file upload button implementation:"
rg --type typescript --type tsx 'file.*upload.*button'

echo "\nSearching for file display implementation:"
rg --type typescript --type tsx 'display.*uploaded.*file'

echo "\nSearching for token count calculation:"
rg --type typescript --type tsx 'token.*count.*file'

echo "\nSearching for usage of MultimodalContent and UploadFile interfaces:"
rg --type typescript --type tsx 'MultimodalContent|UploadFile'

Length of output: 822


Script:

#!/bin/bash
# Search for file upload related implementations
echo "Searching for file upload button implementation:"
rg -g "*.ts" -g "*.tsx" 'file.*upload.*button'

echo "\nSearching for file display implementation:"
rg -g "*.ts" -g "*.tsx" 'display.*uploaded.*file'

echo "\nSearching for token count calculation:"
rg -g "*.ts" -g "*.tsx" 'token.*count.*file'

echo "\nSearching for usage of MultimodalContent and UploadFile interfaces:"
rg -g "*.ts" -g "*.tsx" 'MultimodalContent|UploadFile'

Length of output: 3120


Script:

#!/bin/bash
# Search for file upload button related implementations with alternative patterns
echo "Searching for file upload button implementation with alternative patterns:"
rg -g "*.ts" -g "*.tsx" 'file.*upload.*button|attach.*button|attachment.*button|handleUpload|uploadButton'

echo "\nSearching for file display implementation with alternative patterns:"
rg -g "*.ts" -g "*.tsx" 'display.*uploaded.*file|render.*uploaded.*file|UploadedFiles|FileList|showUploadedFiles'

echo "\nSearching for token count calculation:"
rg -g "*.ts" -g "*.tsx" 'token.*count.*file'

echo "\nSearching for usage of MultimodalContent and UploadFile interfaces:"
rg -g "*.ts" -g "*.tsx" 'MultimodalContent|UploadFile'

Length of output: 3348

app/store/chat.ts (1)

Line range hint 398-427: Suggestion: Refactor image handling to reduce duplication

The implementation correctly handles image attachments when there are no file attachments. However, there's duplication in the code for processing images.

Consider refactoring the image handling logic into a separate function to reduce duplication. Here's a suggested implementation:

function processImageAttachments(images: string[], content: string | MultimodalContent[]) {
  const imageContent = images.map((url) => ({
    type: "image_url",
    image_url: { url },
  }));

  const textContent = Array.isArray(content)
    ? content
    : [{ type: "text", text: content }];

  return textContent.concat(imageContent);
}

// Then in the onUserInput function:
if (attachImages && attachImages.length > 0) {
  mContent = processImageAttachments(attachImages, mContent);
  displayContent = processImageAttachments(attachImages, displayContent);
} else {
  mContent = userContent;
  displayContent = userContent;
}

This refactoring will make the code more maintainable and reduce the risk of inconsistencies when handling images in different scenarios.

app/components/chat.tsx (1)

Line range hint 1948-2126: Consider refactoring the attachment rendering code

The implementation for displaying attached files is consistent with the existing code for attached images and provides a good user experience. However, there are a couple of areas that could be improved:

  1. The code for rendering file attachments is quite similar to the code for rendering image attachments. Consider refactoring this to reduce code duplication.

  2. The conditional rendering based on the number of attached images (lines 2091-2110) seems overly complex. This could potentially be simplified using a more dynamic approach.

Here's a suggested refactoring to address these issues:

function renderAttachment(attachment: Image | UploadFile, index: number, onDelete: () => void) {
  const isImage = 'url' in attachment && !('tokenCount' in attachment);
  
  return (
    <div key={index} className={isImage ? styles["attach-image"] : styles["attach-file"]}>
      {isImage ? (
        <div style={{ backgroundImage: `url("${attachment.url}")` }} />
      ) : (
        <>
          <div className={styles["attach-file-icon"] + " no-dark"}>
            <FileIcon {...defaultStyles[(attachment as UploadFile).name.split('.').pop()?.toLowerCase() as DefaultExtensionType]} glyphColor="#303030" />
          </div>
          <div className={styles[`attach-file-name-${getNameClass(attachImages.length)}`]}>
            {(attachment as UploadFile).name} {(attachment as UploadFile).tokenCount}K
          </div>
        </>
      )}
      <div className={styles["attach-image-mask"]}>
        <DeleteImageButton deleteImage={onDelete} />
      </div>
    </div>
  );
}

function getNameClass(imageCount: number) {
  if (imageCount === 0) return 'full';
  if (imageCount === 1) return 'half';
  if (imageCount === 2) return 'less';
  return 'min';
}

// In the render method:
<div className={styles["attachments"]}>
  <div className={styles["attach-images"]}>
    {attachImages.map((image, index) => renderAttachment(image, index, () => setAttachImages(attachImages.filter((_, i) => i !== index))))}
  </div>
  <div className={styles["attach-files"]}>
    {attachFiles.map((file, index) => renderAttachment(file, index, () => setAttachFiles(attachFiles.filter((_, i) => i !== index))))}
  </div>
</div>

This refactoring creates a single renderAttachment function that handles both images and files, reducing code duplication. It also simplifies the conditional rendering of file names based on the number of attached images.

app/utils.ts (1)

32-34: Remove Commented-Out Code

The commented-out code can be removed to keep the codebase clean and maintainable.

-    //const content = await response.text();
-    //const result = file.name + "\n" + content;
-    //return result;
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 1f2dea4 and 32045e9.

📒 Files selected for processing (5)
  • app/client/api.ts (1 hunks)
  • app/components/chat.module.scss (4 hunks)
  • app/components/chat.tsx (13 hunks)
  • app/store/chat.ts (5 hunks)
  • app/utils.ts (3 hunks)
🧰 Additional context used
🔇 Additional comments (14)
app/client/api.ts (2)

32-32: LGTM: Changes align with PR objectives

The modifications to the MultimodalContent interface successfully implement the text file upload functionality as described in the PR objectives. The addition of the "file_url" type and the corresponding file_url property with url, name, and tokenCount provides the necessary structure to handle uploaded text files.

It's worth noting that the tokenCount is correctly typed as number, addressing a previous review comment.

Also applies to: 37-41


44-47: LGTM: New UploadFile interface added

The new UploadFile interface is well-structured and aligns with the PR objectives for handling uploaded text files. It includes the necessary properties (name, url, and optional tokenCount) to represent an uploaded file.

The tokenCount is correctly typed as number, maintaining consistency with the MultimodalContent interface.

app/store/chat.ts (3)

9-9: LGTM: Necessary imports added for file upload functionality

The new imports UploadFile and readFileContent are correctly added to support the text file upload feature.

Also applies to: 22-22


449-450: LGTM: Correct content saved for display

The change to use displayContent instead of mContent when saving the user message is correct. This ensures that the session stores the content as it should be displayed, including file and image attachments, which aligns with the PR objectives.


Line range hint 1-1180: Overall assessment: Implementation aligns with PR objectives, with minor improvements suggested

The changes successfully implement the text file upload functionality as described in the PR objectives. The code handles both file and image attachments, structures the content correctly for API requests and display, and updates the session with the appropriate content.

Key points:

  1. The function signature and imports have been correctly updated.
  2. File and image attachment handling is implemented, though there's a minor issue with the condition for handling images.
  3. There's some code duplication in image handling that could be refactored.
  4. The session update correctly uses the display content, ensuring proper representation of attachments.

Please address the suggested improvements, particularly the image handling condition issue, to enhance the overall quality and reliability of the implementation.

app/components/chat.tsx (3)

49-49: LGTM: New imports for file upload functionality

The new imports are consistent with the added functionality for uploading and displaying text files. They include necessary components and functions for file icons, file uploads, and token counting.

Also applies to: 80-81, 83-83, 77-77


453-456: LGTM: ChatActions component updated for file upload

The ChatActions component has been appropriately updated with new props (uploadDocument and setAttachFiles) to support the new file upload functionality.


Line range hint 1-2159: Overall, good implementation of text file upload functionality

The changes successfully implement the new feature for uploading and displaying text files. The code is well-integrated with the existing chat component and maintains consistency with the current codebase structure.

Main points for improvement:

  1. The uploadDocument function could be refactored into smaller, more manageable functions.
  2. Error handling could be improved in the file upload process.
  3. The rendering of attachments (both images and files) could be simplified to reduce code duplication.

These improvements would enhance the maintainability and robustness of the code. Great job on implementing this new feature!

app/utils.ts (3)

4-4: Import Statement Updated Correctly

The addition of UploadFile to the import statement is appropriate for the new functionality.


42-70: Use a Standard Tokenization Library for Accurate Token Counting

The previous suggestion to use a standard tokenizer is still applicable. Utilizing a standard library like tiktoken or gpt-3-encoder will provide more accurate token counts and simplify maintenance.


305-316: Issue Resolved: Correctly Handling Undefined file_url

The function now properly checks if c.file_url is defined before adding it to the files array, addressing the previous concern about potential issues with undefined values.

app/components/chat.module.scss (3)

74-99: Duplicated styles in .attach-file-name-* classes have been consolidated

Great job on using the %attach-file-name-common placeholder selector and extending it in the .attach-file-name-* classes. This effectively reduces code duplication and enhances maintainability.


121-128: Refactor .delete-image styles for consistency

The .delete-image class inside .attach-file has similar styles to the one within .attach-image, with differences in dimensions and float direction. To minimize code duplication and ensure consistency, it's advisable to use a mixin or extend a common class, allowing you to parameterize the differing properties.


562-569: Styles for .chat-message-item-files are well-structured

The implementation and styling for the .chat-message-item-files class effectively organize file attachments within chat messages. The use of flex layout and appropriate spacing enhances readability and maintains a clean UI.

app/store/chat.ts Show resolved Hide resolved
app/store/chat.ts Show resolved Hide resolved
app/components/chat.tsx Show resolved Hide resolved
app/utils.ts Outdated Show resolved Hide resolved
app/components/chat.module.scss Show resolved Hide resolved
@lloydzhou lloydzhou requested a review from ElricLiu October 8, 2024 08:48
@lloydzhou lloydzhou added help wanted Extra attention is needed good first issue Good for newcomers planned planned feature, will support in the future labels Oct 17, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 32045e9 and e86a390.

📒 Files selected for processing (1)
  • app/utils.ts (3 hunks)
🧰 Additional context used
🔇 Additional comments (2)
app/utils.ts (2)

42-70: 'countTokens' Function Implementation is Acceptable

The countTokens function provides a rough estimation of the token count based on character weights. Given the constraints and the unavailability of standard tokenization libraries due to dependency issues, the implementation is acceptable for its intended purpose.


305-316: 'getMessageFiles' Function Correctly Extracts File URLs

The function accurately retrieves files from the message content by checking c.type === "file_url" and ensuring c.file_url is defined. This prevents potential issues with undefined file URLs.

Comment on lines +22 to +24
if (!file.url.includes(host_url.host)) {
throw new Error(`The URL ${file.url} is not allowed to access.`);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Enhance URL Validation by Comparing Hostnames Directly

The current check using file.url.includes(host_url.host) might not securely verify that the file URL belongs to the same host. Partial matches could lead to false positives or security vulnerabilities. To ensure robust URL validation, consider parsing the URLs and comparing the hostnames directly.

Apply this diff to improve URL validation:

 export const readFileContent = async (file: UploadFile): Promise<string> => {
   const host_url = new URL(window.location.href);
+  const file_url = new URL(file.url);
-  if (!file.url.includes(host_url.host)) {
+  if (file_url.host !== host_url.host) {
     throw new Error(`The URL ${file.url} is not allowed to access.`);
   }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
if (!file.url.includes(host_url.host)) {
throw new Error(`The URL ${file.url} is not allowed to access.`);
}
const host_url = new URL(window.location.href);
const file_url = new URL(file.url);
if (file_url.host !== host_url.host) {
throw new Error(`The URL ${file.url} is not allowed to access.`);
}

@QAbot-zh
Copy link

有演示地址吗,我移植后无法上传文件,定位问题是到这里:

image

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


Is there a demo address? I cannot upload files after transplantation. The problem is here:

image

@Dakai
Copy link
Contributor Author

Dakai commented Jan 2, 2025

有演示地址吗,我移植后无法上传文件,定位问题是到这里:

image

有的:chat-gpt-next-web-demo-ramgi84mp-dakai1s-projects.vercel.app
请用自己的 key 测试,我刚试了上传文件功能是正常的。

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


Is there a demo address? I cannot upload files after transplantation. The problem is here:

![image](https://private-user-images.githubusercontent.com/40236765/398501935-4dfcb4c1-4acd-412a-9 ba0-461fa9dbb3e4.png?jwt=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJnaXRodWIuY29tIiwiYXVkIjoic mF3LmdpdGh1YnVzZXJjb250ZW50LmNvbSIsImtleSI6ImtleTUiLCJleHAiOjE3MzU4MTA5NDIsIm5iZiI6MTczNTgxMDY0Miw icGF0aCI6Ii80MDIzNjc2NS8zOTg1MDE5MzUtNGRmY2I0YzEtNGFjZC00MTJhLTliYTAtNDYxZmE5ZGJiM2U0LnBuZz9YLUFtei 1BbGdvcml0aG09QVdTNC1ITUFDLVNIQTI1NiZYLUFtei1DcmVkZW50aWFsPUFLSUFWQ09EWUxTQTUzUFFLNFpBJTJGMjAyNTAx MDIlMkZ1cy1lYXN0LTElMkZzMyUyRmF3czRfcmVxdWVzdCZYLUFtei1EYXRlPTIwMjUwMTAyVDA5MzcyMlomWC1BbXotRXhwaXJ lcz0zMDAmWC1BbXotU2lnbmF0dXJlPTRiZjUwYmU0NDMwNDJjNDM4NTM4MDJiNzE3MGY5ZTI4NjI4ZmM2NzBkNzM1YzFiYWNlO GE4YWM4NzdlZGI3ZjMmWC1BbXotU2lnbmVkSGVhZGVycz1ob3N0In0.WvxCa4I60mlCw3JG_inai9HVG3hoX44HgIgErKX7erk)

Yes: chat-gpt-next-web-demo-ramgi84mp-dakai1s-projects.vercel.app
Please use your own key to test. I just tried the file upload function and it works fine.

@QAbot-zh
Copy link

QAbot-zh commented Jan 2, 2025

有演示地址吗,我移植后无法上传文件,定位问题是到这里:
image

有的:chat-gpt-next-web-demo-ramgi84mp-dakai1s-projects.vercel.app 请用自己的 key 测试,我刚试了上传文件功能是正常的。

访问不了,这个好像算私域网站

image

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


Is there a demo address? I can't upload files after transplantation. The location problem is here:
![image](https://private-user-images.githubusercontent.com/40236765/398501935-4dfcb4c1-4acd-412a-9 ba0-461fa9dbb3e4.png?jwt=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJnaXRodWIuY29tIiwiYXVkIjoic mF3LmdpdGh1YnVzZXJjb250ZW50LmNvbSIsImtleSI6ImtleTUiLCJleHAiOjE3MzU4MTA5NDIsIm5iZiI6MTczNTgxMDY0Miw icGF0aCI6Ii80MDIzNjc2NS8zOTg1MDE5MzUtNGRmY2I0YzEtNGFjZC00MTJhLTliYTAtNDYxZmE5ZGJiM2U0LnBuZz9YLUFtei 1BbGdvcml0aG09QVdTNC1ITUFDLVNIQTI1NiZYLUFtei1DcmVkZW50aWFsPUFLSUFWQ09EWUxTQTUzUFFLNFpBJTJGMjAyNTAx MDIlMkZ1cy1lYXN0LTElMkZzMyUyRmF3czRfcmVxdWVzdCZYLUFtei1EYXRlPTIwMjUwMTAyVDA5MzcyMlomWC1BbXotRXhwaXJ lcz0zMDAmWC1BbXotU2lnbmF0dXJlPTRiZjUwYmU0NDMwNDJjNDM4NTM4MDJiNzE3MGY5ZTI4NjI4ZmM2NzBkNzM1YzFiYWNlO GE4YWM4NzdlZGI3ZjMmWC1BbXotU2lnbmVkSGVhZGVycz1ob3N0In0.WvxCa4I60mlCw3JG_inai9HVG3hoX44HgIgErKX7erk)

Yes: chat-gpt-next-web-demo-ramgi84mp-dakai1s-projects.vercel.app Please use your own key to test. I just tried the file upload function and it is normal.

Can't access it, this seems to be a private domain website

image

@Dakai
Copy link
Contributor Author

Dakai commented Jan 3, 2025

有演示地址吗,我移植后无法上传文件,定位问题是到这里:
image

有的:chat-gpt-next-web-demo-ramgi84mp-dakai1s-projects.vercel.app 请用自己的 key 测试,我刚试了上传文件功能是正常的。

访问不了,这个好像算私域网站

image

https://demo.answer42.xyz/ 这个链接

2025-01-03_12-33

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


Is there a demo address? I can't upload files after transplantation. The location problem is here:
![image](https://private-user-images.githubusercontent.com/40236765/398501935-4dfcb4c1-4acd-412a-9 ba0-461fa9dbb3e4.png?jwt=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJnaXRodWIuY29tIiwiYXVkIjoic mF3LmdpdGh1YnVzZXJjb250ZW50LmNvbSIsImtleSI6ImtleTUiLCJleHAiOjE3MzU4MTA5NDIsIm5iZiI6MTczNTgxMDY0Miw icGF0aCI6Ii80MDIzNjc2NS8zOTg1MDE5MzUtNGRmY2I0YzEtNGFjZC00MTJhLTliYTAtNDYxZmE5ZGJiM2U0LnBuZz9YLUFtei 1BbGdvcml0aG09QVdTNC1ITUFDLVNIQTI1NiZYLUFtei1DcmVkZW50aWFsPUFLSUFWQ09EWUxTQTUzUFFLNFpBJTJGMjAyNTAx MDIlMkZ1cy1lYXN0LTElMkZzMyUyRmF3czRfcmVxdWVzdCZYLUFtei1EYXRlPTIwMjUwMTAyVDA5MzcyMlomWC1BbXotRXhwaXJ lcz0zMDAmWC1BbXotU2lnbmF0dXJlPTRiZjUwYmU0NDMwNDJjNDM4NTM4MDJiNzE3MGY5ZTI4NjI4ZmM2NzBkNzM1YzFiYWNlO GE4YWM4NzdlZGI3ZjMmWC1BbXotU2lnbmVkSGVhZGVycz1ob3N0In0.WvxCa4I60mlCw3JG_inai9HVG3hoX44HgIgErKX7erk)

Yes: chat-gpt-next-web-demo-ramgi84mp-dakai1s-projects.vercel.app Please use your own key to test. I just tried the upload file function and it is normal.

Cannot access, this seems to be a private domain website

![image](https://private-user-images.githubusercontent.com/40236765/399712435-6d37b679-1ba2-41ec-9 224-c9bd0afb0d32.png?jwt=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJnaXRodWIuY29tIiwiYXVkIjoic mF3LmdpdGh1YnVzZXJjb250ZW50LmNvbSIsImtleSI6ImtleTUiLCJleHAiOjE3MzU4Nzg3NzgsIm5iZiI6MTczNTg3ODQ3OCw icGF0aCI6Ii80MDIzNjc2NS8zOTk3MTI0MzUtNmQzN2I2NzktMWJhMi00MWVjLTkyMjQtYzliZDBhZmIwZDMyLnBuZz9YLUFtei 1BbGdvcml0aG09QVdTNC1ITUFDLVNIQTI1NiZYLUFtei1DcmVkZW50aWFsPUFLSUFWQ09EWUxTQTUzUFFLNFpBJTJGMjAyNTAx MDMlMkZ1cy1lYXN0LTElMkZzMyUyRmF3czRfcmVxdWVzdCZYLUFtei1EYXRlPTIwMjUwMTAzVDA0Mjc1OFomWC1BbXotRXhwaXJ lcz0zMDAmWC1BbXotU2lnbmF0dXJlPTViMTllYWZmNDk0YTkzNDBmY2M3OWZmMWEwZDUyZmQwNmFhOTBiNWI2MzFmNGNjNzQ0N mJhNzZhNDJlOWQ5OWUmWC1BbXotU2lnbmVkSGVhZGVycz1ob3N0In0.NAUZxtue5ITJJbY5rxpkCymhztkWH7Z9653ilyTuuy4)

https://demo.answer42.xyz/ This link

2025-01-03_12-33

@QAbot-zh
Copy link

QAbot-zh commented Jan 3, 2025

@Dakai
奇怪,也还是 405 错误,已经配了自己的 api,可以对话了,文件上传还是失败:

image

image

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


@Dakai
Strange, it’s still a 405 error. I’ve configured my own API and can talk, but the file upload still fails:

image

image

@Dakai
Copy link
Contributor Author

Dakai commented Jan 3, 2025

@QAbot-zh 你试一下图片上传能否成功,如果也不行那就不是服务器那边程序的问题,我这边上传的 Header 是这样的:
2025-01-03_17-56
你图片里的 Header 有个内网的 IP 地址,正常来说不应该有的,应该是它触发了 CORS Policy 被服务器挡下了,试下用浏览器隐私模式关闭所有插件再试试?

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


@QAbot-zh Can you try to see if the image upload is successful? If it doesn't work, it's not a problem with the server's program. The header I uploaded here is like this:
2025-01-03_17-56
The Header in your picture has an IP address on the intranet, which normally shouldn't be there. It should be that it triggered the CORS Policy and was blocked by the server. Can you try using the browser privacy mode to close all plug-ins and try again?

@code-october
Copy link
Contributor

@Dakai 可能跟系统代理软件有关吧,我换个设备就一切正常,在原来的电脑上就图片也传不上去

image

@Issues-translate-bot
Copy link

Bot detected the issue body's language is not English, translate it automatically.


@Dakai Maybe it has something to do with the system proxy software. When I changed the device, everything was normal. Even on the original computer, pictures could not be uploaded.

image

@hdh999
Copy link

hdh999 commented Jan 14, 2025

还需要做什么这个功能才能合并吗?
Is there anything else needed for this feature to be merged?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
good first issue Good for newcomers help wanted Extra attention is needed planned planned feature, will support in the future
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants