-
Notifications
You must be signed in to change notification settings - Fork 60.1k
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
style: 代码规范 #5954
style: 代码规范 #5954
Conversation
@ocean-gao is attempting to deploy a commit to the NextChat Team on Vercel. A member of the Team first needs to authorize it. |
WalkthroughThis pull request introduces a comprehensive set of changes across the project, primarily focusing on code formatting, import organization, and minor logic refinements. The modifications span multiple files and components, with a consistent approach to standardizing string usage (transitioning from double to single quotes), reorganizing import statements, and making minor syntactical improvements. The changes do not fundamentally alter the functionality of the existing codebase but enhance code readability and consistency. Changes
Suggested labels
Suggested reviewers
Possibly related PRs
Poem
Tip CodeRabbit's docstrings feature is now available as part of our Early Access Program! Simply use the command 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? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
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)
Other keywords and placeholders
CodeRabbit Configuration File (
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 13
🔭 Outside diff range comments (3)
app/client/platforms/tencent.ts (1)
Line range hint
143-168
: Consider extracting animation logic to a separate function.The response animation logic is complex and would benefit from being extracted into a separate function for better maintainability.
Consider creating a separate class or utility for handling streaming animations:
class ResponseAnimator { private responseText = ''; private remainText = ''; private finished = false; constructor( private readonly onUpdate: (text: string, delta: string) => void, private readonly controller: AbortController ) {} animate() { if (this.finished || this.controller.signal.aborted) { this.responseText += this.remainText; return this.responseText; } if (this.remainText.length > 0) { const fetchCount = Math.max(1, Math.round(this.remainText.length / 60)); const fetchText = this.remainText.slice(0, fetchCount); this.responseText += fetchText; this.remainText = this.remainText.slice(fetchCount); this.onUpdate(this.responseText, fetchText); } requestAnimationFrame(() => this.animate()); } addText(text: string) { this.remainText += text; } finish() { this.finished = true; return this.responseText + this.remainText; } }app/client/platforms/alibaba.ts (1)
Line range hint
229-243
: Enhance error handling in onmessage.The error handling could be more informative by providing specific error context.
} catch (e) { - console.error('[Request] parse error', text, msg); + console.error('[Alibaba API Request] Failed to parse SSE message:', { text, msg, error: e }); }app/client/platforms/openai.ts (1)
Line range hint
189-260
: Consider breaking down the chat method for better maintainabilityThe chat method is handling multiple responsibilities including:
- Model configuration
- Payload construction for different model types (DALL-E, O1, Vision)
- Request handling
Consider extracting these into separate methods for better maintainability and testability.
Suggested refactor:
+ private constructDallePayload(options: ChatOptions): DalleRequestPayload { + const prompt = getMessageTextContent(options.messages.slice(-1)?.pop() as any); + return { + model: options.config.model, + prompt, + response_format: 'b64_json', + n: 1, + size: options.config?.size ?? '1024x1024', + quality: options.config?.quality ?? 'standard', + style: options.config?.style ?? 'vivid', + }; + } + private constructChatPayload(options: ChatOptions, modelConfig: ModelConfig): RequestPayload { + const visionModel = isVisionModel(options.config.model); + const isO1 = options.config.model.startsWith('o1'); + const messages = this.prepareMessages(options.messages, visionModel, isO1); + + const payload: RequestPayload = { + messages, + stream: options.config.stream, + model: modelConfig.model, + temperature: !isO1 ? modelConfig.temperature : 1, + presence_penalty: !isO1 ? modelConfig.presence_penalty : 0, + frequency_penalty: !isO1 ? modelConfig.frequency_penalty : 0, + top_p: !isO1 ? modelConfig.top_p : 1, + }; + + if (isO1) { + payload.max_completion_tokens = modelConfig.max_tokens; + } + + if (visionModel) { + payload.max_tokens = Math.max(modelConfig.max_tokens, 4000); + } + + return payload; + } async chat(options: ChatOptions) { const modelConfig = {/*...*/}; const isDalle3 = _isDalle3(options.config.model); const requestPayload = isDalle3 ? this.constructDallePayload(options) : this.constructChatPayload(options, modelConfig); // Continue with request handling... }
🧹 Nitpick comments (58)
.github/workflows/sync.yml (1)
19-20
: Consider upgrading to actions/checkout@v4The checkout action can be upgraded to v4 for improved performance and security features.
- uses: actions/checkout@v3 + uses: actions/checkout@v4🧰 Tools
🪛 actionlint (1.7.4)
20-20: the runner of "actions/checkout@v3" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
app/client/platforms/tencent.ts (5)
38-43
: Consider using a more specific type for Messages array.The Messages array in RequestPayload could benefit from a more specific type definition to improve type safety and readability.
Consider this improvement:
- Messages: { - Role: 'system' | 'user' | 'assistant'; - Content: string | MultimodalContent[]; - }[]; + Messages: Array<{ + Role: 'system' | 'user' | 'assistant'; + Content: string | MultimodalContent[]; + }>;
Line range hint
49-61
: Enhance type safety in capitalizeKeys function.While the function works correctly, it could benefit from more specific TypeScript types to improve maintainability and catch potential issues at compile time.
Consider this type-safe implementation:
-function capitalizeKeys(obj: any): any { +function capitalizeKeys<T>(obj: T): Record<string, unknown> { if (isArray(obj)) { return obj.map(capitalizeKeys); } else if (isObject(obj)) { return mapValues( - mapKeys(obj, (value: any, key: string) => + mapKeys(obj as Record<string, unknown>, (_, key: string) => key.replace(/(^|_)(\w)/g, (m, $1, $2) => $2.toUpperCase())), capitalizeKeys, ); } else { return obj; } }
Line range hint
67-82
: Simplify baseUrl construction logic.The baseUrl construction logic could be simplified to improve readability and maintainability.
Consider this cleaner approach:
- let baseUrl = ''; - - if (accessStore.useCustomConfig) { - baseUrl = accessStore.tencentUrl; - } - - if (baseUrl.length === 0) { - const isApp = !!getClientConfig()?.isApp; - baseUrl = isApp ? TENCENT_BASE_URL : ApiPath.Tencent; - } + const baseUrl = accessStore.useCustomConfig + ? accessStore.tencentUrl + : !!getClientConfig()?.isApp + ? TENCENT_BASE_URL + : ApiPath.Tencent;
193-203
: Simplify content type check condition.The nested condition checks for content type and response status could be simplified for better readability.
Consider this cleaner approach:
- if ( - !res.ok - || !res.headers - .get('content-type') - ?.startsWith(EventStreamContentType) - || res.status !== 200 - ) { + const contentType = res.headers.get('content-type'); + const isValidResponse = res.ok + && contentType?.startsWith(EventStreamContentType) + && res.status === 200; + + if (!isValidResponse) {
261-263
: Enhance error logging with more context.The error logging could be more descriptive to aid in debugging.
Consider this improvement:
- console.log('[Request] failed to make a chat request', e); + console.error('[Tencent API] Chat request failed', { + error: e, + model: options.config.model, + timestamp: new Date().toISOString() + });app/components/chat.tsx (1)
Line range hint
1-213
: Overall File Review: Style Improvements and Minor ConcernsThis file introduces a significant set of style and structural changes. Given that the PR focuses on code style, most of the transformations here are acceptable: the imports have been reordered, and the usage of single quotes, consistency in indentation, and improved readability align well with standard code style guidelines. However, a few items warrant attention:
Arrow Function Assignments in Expressions
In multiple places (e.g., lines 160, 186, 373, 492, 856, 874, 1052-1053, 1427-1428, 1452-1453, 1456-1457, 1458-1459), arrow functions are used in the form:
session => (session.memoryPrompt = '')
or:
config => (config.theme = nextTheme)
While this is not strictly incorrect, many style guides consider it clearer to use braces around the body, explicitly returning. For instance:
session => { session.memoryPrompt = '';}
Using curly braces can improve clarity by separating side-effectful assignments from expression returns.Unnecessary Fragment Usage (lines 605-619)
The static analysis flags a redundant fragment. If the fragment only encloses a single child or is otherwise unnecessary, directly return the child element.Input Field (lines ~868-877)
The static analysis warns that a void tag like should not contain children. In this snippet, ensure there are no extraneous child elements or text between and .These are mostly small improvements, and overall, the file aligns with the style-based objectives of this PR.
.github/ISSUE_TEMPLATE/1_bug_report.yml (1)
1-80
: LGTM! Minor suggestions for template clarityThe YAML formatting is consistent and maintains all functional aspects. Consider these optional improvements for better user experience:
- Add placeholder text in the input fields to guide users
- Add examples in the description fields for "Bug Description" and "Recurrence Steps"
Example improvement for the Bug Description:
- type: textarea attributes: label: 🐛 Bug Description description: A clear and concise description of the bug, if the above option is `Other`, please also explain in detail. + placeholder: | + Describe what happened: + - What you were trying to do + - What went wrong + - What you expected to happenapp/api/stability.ts (1)
72-74
: Consider consolidating header declarationsThe headers could be more maintainable using a headers object.
- headers: { - 'Content-Type': req.headers.get('Content-Type') || 'multipart/form-data', - 'Accept': req.headers.get('Accept') || 'application/json', - 'Authorization': `Bearer ${key}`, - }, + headers: { + ...Object.fromEntries( + ['Content-Type', 'Accept'].map(h => [ + h, + req.headers.get(h) || (h === 'Content-Type' ? 'multipart/form-data' : 'application/json') + ]) + ), + 'Authorization': `Bearer ${key}`, + },app/api/xai.ts (1)
Line range hint
82-103
: Consider extracting model validation logicThe model validation logic could be moved to a separate function for better maintainability.
+async function validateModelAccess(body: string, customModels: any) { + try { + const jsonBody = JSON.parse(body) as { model?: string }; + return !isModelAvailableInServer( + customModels, + jsonBody?.model as string, + ServiceProvider.XAI as string, + ); + } catch (e) { + console.error(`[XAI] filter`, e); + return false; + } +}Then use it in the request function:
if (serverConfig.customModels && req.body) { try { const clonedBody = await req.text(); fetchOptions.body = clonedBody; - // ... existing validation logic + if (await validateModelAccess(clonedBody, serverConfig.customModels)) { + return NextResponse.json( + { + error: true, + message: `you are not allowed to use this model`, + }, + { status: 403 } + ); + } } catch (e) { console.error(`[XAI] filter`, e); } }app/api/bytedance.ts (1)
45-45
: Remove or update the misleading commentThe comment
// alibaba use base url or just remove the path
appears to be copied from another file and should be removed or updated to reference ByteDance instead.app/api/iflytek.ts (1)
13-13
: Remove redundant commentThe comment
// iflytek
is redundant as it's already clear from the filename.app/api/moonshot.ts (1)
45-46
: Remove or update the misleading commentThe comment
// alibaba use base url or just remove the path
appears to be copied from another file and should be removed or updated to reference Moonshot instead.app/api/baidu.ts (1)
Line range hint
35-43
: Consider standardizing template literal quotesWhile most strings use single quotes, the error message uses backticks for a simple string concatenation. Consider using single quotes with string concatenation for consistency.
- message: `missing BAIDU_API_KEY or BAIDU_SECRET_KEY in server env vars`, + message: 'missing BAIDU_API_KEY or BAIDU_SECRET_KEY in server env vars',app/api/webdav/[...path]/route.ts (1)
37-48
: Consider simplifying the endpoint validation logicThe nested validation logic could be simplified for better readability.
- !endpoint - || !mergedAllowedWebDavEndpoints.some((allowedEndpoint) => { - const normalizedAllowedEndpoint = normalizeUrl(allowedEndpoint); - const normalizedEndpoint = normalizeUrl(endpoint as string); - - return ( - normalizedEndpoint - && normalizedEndpoint.hostname === normalizedAllowedEndpoint?.hostname - && normalizedEndpoint.pathname.startsWith( - normalizedAllowedEndpoint.pathname, - ) - ); - }) + !endpoint || !mergedAllowedWebDavEndpoints.some((allowedEndpoint) => { + const normalizedAllowedEndpoint = normalizeUrl(allowedEndpoint); + const normalizedEndpoint = normalizeUrl(endpoint as string); + return normalizedEndpoint && + normalizedEndpoint.hostname === normalizedAllowedEndpoint?.hostname && + normalizedEndpoint.pathname.startsWith(normalizedAllowedEndpoint.pathname); + })app/api/anthropic.ts (2)
64-69
: Consider using nullish coalescing for auth headerThe auth header fallback chain could be simplified using nullish coalescing operator.
- const authValue - = req.headers.get(authHeaderName) - || req.headers.get('Authorization')?.replaceAll('Bearer ', '').trim() - || serverConfig.anthropicApiKey - || ''; + const authValue = req.headers.get(authHeaderName) ?? + req.headers.get('Authorization')?.replaceAll('Bearer ', '').trim() ?? + serverConfig.anthropicApiKey ?? + '';
73-74
: Consider using nullish coalescing for base URLThe base URL fallback chain could be simplified.
- let baseUrl - = serverConfig.anthropicUrl || serverConfig.baseUrl || ANTHROPIC_BASE_URL; + let baseUrl = serverConfig.anthropicUrl ?? serverConfig.baseUrl ?? ANTHROPIC_BASE_URL;app/components/chat-list.tsx (2)
67-91
: Consider extracting narrow view into a separate componentThe conditional rendering of narrow view could be extracted into a separate component for better maintainability.
+ const NarrowView = ({ mask, count }: { mask: Mask; count: number }) => ( + <div className={styles['chat-item-narrow']}> + <div className={clsx(styles['chat-item-avatar'], 'no-dark')}> + <MaskAvatar + avatar={mask.avatar} + model={mask.modelConfig.model} + /> + </div> + <div className={styles['chat-item-narrow-count']}> + {count} + </div> + </div> + ); {props.narrow - ? ( - <div className={styles['chat-item-narrow']}> - <div className={clsx(styles['chat-item-avatar'], 'no-dark')}> - <MaskAvatar - avatar={props.mask.avatar} - model={props.mask.modelConfig.model} - /> - </div> - <div className={styles['chat-item-narrow-count']}> - {props.count} - </div> - </div> - ) + ? <NarrowView mask={props.mask} count={props.count} />
161-164
: Consider simplifying the delete confirmation conditionThe delete confirmation logic could be simplified.
- if ( - (!props.narrow && !isMobileScreen) - || (await showConfirm(Locale.Home.DeleteChat)) - ) { + if (!props.narrow && !isMobileScreen || await showConfirm(Locale.Home.DeleteChat)) {README_CN.md (2)
180-182
: Improve formatting of the webdav endpoints exampleThe formatting of the example could be improved for better readability.
- - 每一个地址必须是一个完整的 endpoint - > `https://xxxx/xxx` + - 每一个地址必须是一个完整的 endpoint,例如: + `https://xxxx/xxx`
193-193
: Improve consistency in Azure and ByteDance configuration examplesThe formatting style differs between Azure and ByteDance configuration examples.
- > 示例:`+gpt-3.5-turbo@Azure=gpt35`这个配置会在模型列表显示一个`gpt35(Azure)`的选项。 + > 示例: `+gpt-3.5-turbo@Azure=gpt35` 这个配置会在模型列表显示一个 `gpt35(Azure)` 的选项。Also applies to: 199-199
app/components/new-chat.tsx (2)
42-43
: Fix curly brace formattingThe curly brace placement is inconsistent with the rest of the codebase.
- if (!appBody || masks.length === 0) - { return; } + if (!appBody || masks.length === 0) { + return; + }
113-114
: Improve readability of scrollLeft calculationThe line break in the middle of the calculation makes it harder to read.
- maskRef.current.scrollLeft - = (maskRef.current.scrollWidth - maskRef.current.clientWidth) / 2; + maskRef.current.scrollLeft = (maskRef.current.scrollWidth - maskRef.current.clientWidth) / 2;app/client/platforms/glm.ts (2)
187-189
: Enhance error logging and handlingThe error logging could be more informative for debugging purposes.
- console.log('[Request] failed to make a chat request', e); + console.error('[Request] Chat request failed:', { + error: e, + path: chatPath, + model: requestPayload.model + });
67-67
: Implement speech method or provide more contextThe unimplemented method should either be implemented or documented with a reason for the placeholder.
- throw new Error('Method not implemented.'); + throw new Error('Speech functionality is not supported for ChatGLM API.');app/components/home.tsx (2)
129-142
: Consider error handling for font loadingThe Google Font loading implementation lacks error handling and doesn't account for network failures.
Consider adding error handling:
function loadAsyncGoogleFont() { const linkEl = document.createElement('link'); const proxyFontUrl = '/google-fonts'; const remoteFontUrl = 'https://fonts.googleapis.com'; const googleFontUrl = getClientConfig()?.buildMode === 'export' ? remoteFontUrl : proxyFontUrl; linkEl.rel = 'stylesheet'; linkEl.href = `${googleFontUrl }/css2?family=${ encodeURIComponent('Noto Sans:wght@300;400;700;900') }&display=swap`; + linkEl.onerror = () => { + console.warn('Failed to load Google Font, falling back to system fonts'); + document.head.removeChild(linkEl); + }; document.head.appendChild(linkEl); }
177-182
: Simplify conditional rendering logicThe current implementation uses explicit return statements with braces which adds unnecessary verbosity.
Consider using a more concise approach:
- if (isAuth) - { return <AuthPage />; } - if (isSd) - { return <Sd />; } - if (isSdNew) - { return <Sd />; } + if (isAuth) return <AuthPage />; + if (isSd || isSdNew) return <Sd />;app/client/platforms/bytedance.ts (1)
188-192
: Consider simplifying the condition formatting.The multi-line condition could be more concise while maintaining readability.
- if ( - !res.ok - || !res.headers - .get('content-type') - ?.startsWith(EventStreamContentType) - || res.status !== 200 - ) { + if (!res.ok || !res.headers.get('content-type')?.startsWith(EventStreamContentType) || res.status !== 200) {app/components/message-selector.tsx (2)
30-31
: Fix condition formatting style.The condition formatting is inconsistent with typical React/TypeScript style.
- if (e.key !== 'Shift') - { return; } + if (e.key !== 'Shift') return; - if (e.key !== 'Shift') - { return; } + if (e.key !== 'Shift') return;Also applies to: 35-36
Line range hint
111-115
: Simplify forEach loop.The forEach with ternary operation can be simplified for better readability.
- messages.forEach(m => - getMessageTextContent(m).includes(text) - ? searchResults.add(m.id!) - : null, - ); + messages.forEach(m => { + if (getMessageTextContent(m).includes(text)) { + searchResults.add(m.id!); + } + });app/client/platforms/alibaba.ts (1)
202-206
: Consider simplifying the condition formatting.Similar to bytedance.ts, the multi-line condition could be more concise while maintaining readability.
- if ( - !res.ok - || !res.headers - .get('content-type') - ?.startsWith(EventStreamContentType) - || res.status !== 200 - ) { + if (!res.ok || !res.headers.get('content-type')?.startsWith(EventStreamContentType) || res.status !== 200) {app/components/artifacts.tsx (3)
71-72
: Remove unnecessary braces in conditional blockThe braces around the return statement are unnecessary and can be removed for better readability.
- if (!props.autoHeight) - { return props.height || 600; } + if (!props.autoHeight) + return props.height || 600;
84-85
: Simplify string concatenation logicThe string concatenation can be simplified using template literals.
- if (props.code.includes('<!DOCTYPE html>')) { - props.code.replace('<!DOCTYPE html>', `<!DOCTYPE html>${script}`); + if (props.code.includes('<!DOCTYPE html>')) { + return props.code.replace('<!DOCTYPE html>', `<!DOCTYPE html>${script}`);
153-154
: Remove unnecessary braces in conditional blockThe braces around the return statement are unnecessary and can be removed for better readability.
- if (loading) - { return; } + if (loading) + return;app/client/platforms/baidu.ts (2)
73-74
: Enhance URL validation logicConsider using URL constructor for more robust URL validation and handling.
- if (!baseUrl.startsWith('http') && !baseUrl.startsWith(ApiPath.Baidu)) { - baseUrl = `https://${baseUrl}`; + try { + new URL(baseUrl); + } catch { + baseUrl = `https://${baseUrl}`; + }
95-98
: Replace empty space placeholder with meaningful contentUsing a single space as placeholder content might cause issues with some models. Consider using a more meaningful placeholder or documenting why this approach is necessary.
- role: 'assistant', - content: ' ', + role: 'assistant', + content: '[placeholder response]',app/client/platforms/google.ts (2)
79-79
: Remove unnecessary this aliasingArrow functions inherit
this
from their enclosing scope, making the aliasing unnecessary.- const apiClient = this;
Then update the reference at line 291 to use
this
directly:- const message = apiClient.extractMessage(resJson); + const message = this.extractMessage(resJson);🧰 Tools
🪛 Biome (1.9.4)
[error] 79-79: This aliasing of this is unnecessary.
Arrow functions inherits
this
from their enclosing scope.
Safe fix: Use this instead of an alias.(lint/complexity/noUselessThisAlias)
Line range hint
207-221
: Simplify deeply nested optional chainingConsider extracting the nested property access into a separate variable for better readability.
- const functionCall = chunkJson?.candidates - ?.at(0) - ?.content - .parts - .at(0) - ?.functionCall; + const candidate = chunkJson?.candidates?.at(0); + const functionCall = candidate?.content?.parts?.at(0)?.functionCall;app/components/model-config.tsx (1)
116-200
: Consider extracting conditional rendering logicThe large conditional block for Google provider check makes the component harder to maintain.
Consider extracting the conditional rendering logic into a separate component:
function ModelConfigExtras({ modelConfig, updateConfig }) { if (modelConfig?.providerName === ServiceProvider.Google) { return null; } return ( <> <ListItem /* ... */ /> <ListItem /* ... */ /> {/* ... other items ... */} </> ); }This would make the main component more readable and easier to maintain.
🧰 Tools
🪛 Biome (1.9.4)
[error] 133-136: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 156-159: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 177-178: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 194-194: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 170-181: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 188-197: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
app/client/api.ts (1)
312-313
: Use early return pattern for better readabilityThe if statement with curly braces on a new line deviates from the standard pattern.
- if (isBaidu && clientConfig?.isApp) - { return headers; } + if (isBaidu && clientConfig?.isApp) { + return headers; + }app/components/markdown.tsx (3)
42-44
: Enhance error handling in Mermaid componentThe error handling could be more informative for debugging purposes.
- console.error('[Mermaid] ', e.message); + console.error('[Mermaid] Failed to render diagram:', { + error: e.message, + code: props.code, + });
106-107
: Improve boolean expression readabilityThe boolean expression with multiple conditions and line breaks could be more readable.
- const enableArtifacts - = session.mask?.enableArtifacts !== false && config.enableArtifacts; + const enableArtifacts = ( + session.mask?.enableArtifacts !== false + && config.enableArtifacts + );
349-355
: Simplify conditional renderingThe conditional rendering with multiple lines and parentheses can be simplified.
- {props.loading - ? ( - <LoadingIcon /> - ) - : ( - <MarkdownContent content={props.content} /> - )} + {props.loading ? <LoadingIcon /> : <MarkdownContent content={props.content} />}app/components/realtime-chat/realtime-chat.tsx (4)
62-63
: Fix inconsistent curly brace placement.The curly braces are inconsistently placed on new lines after if conditions.
Apply this diff to maintain consistent style:
- if (isConnecting) - { return; } + if (isConnecting) { + return; + } - if (initRef.current) - { return; } + if (initRef.current) { + return; + }Also applies to: 266-267
74-75
: Improve type declaration readability.The type declaration and assignment are split across lines in a way that reduces readability.
Apply this diff to improve readability:
- const modalities: Modality[] - = modality === 'audio' ? ['text', 'audio'] : ['text']; + const modalities: Modality[] = modality === 'audio' + ? ['text', 'audio'] + : ['text'];
Line range hint
89-106
: Consider extracting TODO comments into separate tasks.There's a large commented-out code block with a TODO comment that should be tracked properly.
Would you like me to create a GitHub issue to track this TODO item for implementing message history in real-time chat?
Line range hint
234-259
: Enhance error handling in toggleRecording function.The error handling in toggleRecording could be more informative and user-friendly.
Consider adding user-facing error messages and specific error types:
} catch (error) { - console.error('Failed to start recording:', error); + const errorMessage = error instanceof Error + ? error.message + : 'Unknown error occurred'; + console.error('Failed to start recording:', errorMessage); + setStatus(`Recording failed: ${errorMessage}`); }app/client/platforms/anthropic.ts (1)
132-135
: Simplify content validation logic.The content validation logic can be simplified using optional chaining and nullish coalescing.
Apply this diff to improve readability:
- if (!v.content) - { return false; } - if (typeof v.content === 'string' && !v.content.trim()) - { return false; } + const content = typeof v.content === 'string' + ? v.content?.trim() + : v.content; + return !!content;app/client/platforms/openai.ts (3)
96-98
: Consider enhancing error message clarityThe error message could be more descriptive to help users understand exactly what Azure configuration is missing.
- 'incomplete azure config, please check it in your settings page', + 'Incomplete Azure configuration. Please verify your Azure URL and API key in settings.',
414-415
: Enhance error handling for the usage endpointThe error handling could be more comprehensive to handle different types of API errors and provide more specific error messages.
- throw new Error('Failed to query usage from openai'); + const errorMessage = used.status === 403 + ? 'Access denied. Please verify your API key permissions.' + : `Failed to query usage from OpenAI. Status: ${used.status}`; + throw new Error(errorMessage);
470-471
: Improve documentation for the models methodThe comment suggests this code path is not executed due to
disableListModels
being true by default. Consider adding proper documentation to explain this behavior.+ /** + * Fetches available models from OpenAI API. + * Note: Currently this method always returns DEFAULT_MODELS due to disableListModels being true. + * The model fetching logic is kept for future use when dynamic model loading is enabled. + */ async models(): Promise<LLMModel[]> {app/components/exporter.tsx (3)
447-451
: Consolidate error handling in ImagePreviewerThe error handling pattern of checking for null/undefined could be simplified using optional chaining and early returns.
- if (!dom) - { return; } - toBlob(dom).then((blob) => { - if (!blob) - { return; } + if (!dom) { + showToast(Locale.Export.Image.Failed); + return; + } + + toBlob(dom).then((blob) => { + if (!blob) { + showToast(Locale.Export.Image.Failed); + return; + }
324-325
: Improve error handling in share functionalityThe early return without user feedback could be improved to provide better UX.
- if (!res) - { return; } + if (!res) { + showToast(Locale.Export.Share.Failed); + return; + }
650-660
: Consider performance optimization for markdown generationThe markdown text generation could be memoized to prevent unnecessary re-renders.
+ const mdText = useMemo(() => { + return `# ${props.topic}\n\n${ + props.messages + .map((m) => { + return m.role === 'user' + ? `## ${Locale.Export.MessageFromYou}:\n${getMessageTextContent(m)}` + : `## ${Locale.Export.MessageFromChatGPT}:\n${getMessageTextContent( + m, + ).trim()}`; + }) + .join('\n\n')}`; + }, [props.topic, props.messages]);app/components/mask.tsx (3)
Line range hint
94-104
: Consider optimizing update handlers with useCallbackThe update handlers could be memoized to prevent unnecessary re-renders.
+ const updateConfig = useCallback((updater: (config: ModelConfig) => void) => { if (props.readonly) return; const config = { ...props.mask.modelConfig }; updater(config); props.updateMask((mask) => { mask.modelConfig = config; mask.syncGlobalConfig = false; }); + }, [props.readonly, props.mask.modelConfig, props.updateMask]);
Line range hint
356-367
: Enhance error handling for context updatesThe context update logic could be more robust with proper error handling and validation.
const updateContextPrompt = (i: number, prompt: ChatMessage) => { + if (i < 0 || i >= props.context.length) { + console.error('Invalid context index:', i); + return; + } props.updateContext((context) => { const images = getMessageImages(context[i]); context[i] = prompt; if (images.length > 0) { const text = getMessageTextContent(context[i]); const newContext: MultimodalContent[] = [{ type: 'text', text }]; for (const img of images) { newContext.push({ type: 'image_url', image_url: { url: img } }); } context[i].content = newContext; } }); };
Line range hint
475-484
: Optimize search functionalityThe search implementation could be optimized using debounce and memoization.
+ import { useMemo, useCallback } from 'react'; + import debounce from 'lodash/debounce'; + const debouncedSearch = useCallback( + debounce((text: string, masks: Mask[], setResults: (masks: Mask[]) => void) => { + if (text.length > 0) { + const result = masks.filter(m => + m.name.toLowerCase().includes(text.toLowerCase()), + ); + setResults(result); + } else { + setResults(masks); + } + }, 300), + [] + ); const onSearch = (text: string) => { setSearchText(text); - if (text.length > 0) { - const result = allMasks.filter(m => - m.name.toLowerCase().includes(text.toLowerCase()), - ); - setSearchMasks(result); - } else { - setSearchMasks(allMasks); - } + debouncedSearch(text, allMasks, setSearchMasks); };app/components/auth.tsx (1)
75-75
: Improve state update pattern for better readabilityThe current pattern of using assignments in arrow function expressions can be confusing. Consider using object spread for clearer intent.
Example refactor for one of the cases:
- access => (access.accessCode = e.currentTarget.value) + access => ({ ...access, accessCode: e.currentTarget.value })Also applies to: 93-93, 106-106
🧰 Tools
🪛 Biome (1.9.4)
[error] 75-75: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
⛔ Files ignored due to path filters (29)
docs/images/bt/bt-install-1.jpeg
is excluded by!**/*.jpeg
docs/images/bt/bt-install-2.jpeg
is excluded by!**/*.jpeg
docs/images/bt/bt-install-3.jpeg
is excluded by!**/*.jpeg
docs/images/bt/bt-install-4.jpeg
is excluded by!**/*.jpeg
docs/images/bt/bt-install-5.jpeg
is excluded by!**/*.jpeg
docs/images/bt/bt-install-6.jpeg
is excluded by!**/*.jpeg
docs/images/vercel/vercel-create-1.jpg
is excluded by!**/*.jpg
docs/images/vercel/vercel-create-2.jpg
is excluded by!**/*.jpg
docs/images/vercel/vercel-create-3.jpg
is excluded by!**/*.jpg
docs/images/vercel/vercel-env-edit.jpg
is excluded by!**/*.jpg
docs/images/vercel/vercel-redeploy.jpg
is excluded by!**/*.jpg
pnpm-lock.yaml
is excluded by!**/pnpm-lock.yaml
src-tauri/Cargo.lock
is excluded by!**/*.lock
src-tauri/icons/128x128.png
is excluded by!**/*.png
src-tauri/icons/128x128@2x.png
is excluded by!**/*.png
src-tauri/icons/32x32.png
is excluded by!**/*.png
src-tauri/icons/Square107x107Logo.png
is excluded by!**/*.png
src-tauri/icons/Square142x142Logo.png
is excluded by!**/*.png
src-tauri/icons/Square150x150Logo.png
is excluded by!**/*.png
src-tauri/icons/Square284x284Logo.png
is excluded by!**/*.png
src-tauri/icons/Square30x30Logo.png
is excluded by!**/*.png
src-tauri/icons/Square310x310Logo.png
is excluded by!**/*.png
src-tauri/icons/Square44x44Logo.png
is excluded by!**/*.png
src-tauri/icons/Square71x71Logo.png
is excluded by!**/*.png
src-tauri/icons/Square89x89Logo.png
is excluded by!**/*.png
src-tauri/icons/StoreLogo.png
is excluded by!**/*.png
src-tauri/icons/icon.ico
is excluded by!**/*.ico
src-tauri/icons/icon.png
is excluded by!**/*.png
yarn.lock
is excluded by!**/yarn.lock
,!**/*.lock
📒 Files selected for processing (82)
.eslintignore
(0 hunks).eslintrc.json
(0 hunks).github/ISSUE_TEMPLATE/1_bug_report.yml
(1 hunks).github/ISSUE_TEMPLATE/1_bug_report_cn.yml
(1 hunks).github/ISSUE_TEMPLATE/2_feature_request.yml
(1 hunks).github/ISSUE_TEMPLATE/2_feature_request_cn.yml
(1 hunks).github/PULL_REQUEST_TEMPLATE.md
(1 hunks).github/dependabot.yml
(1 hunks).github/workflows/app.yml
(0 hunks).github/workflows/deploy_preview.yml
(0 hunks).github/workflows/docker.yml
(1 hunks).github/workflows/issue-translator.yml
(1 hunks).github/workflows/remove_deploy_preview.yml
(0 hunks).github/workflows/sync.yml
(1 hunks).github/workflows/test.yml
(0 hunks).gitignore
(0 hunks).gitpod.yml
(0 hunks).lintstagedrc.json
(0 hunks).npmrc
(1 hunks).vscode/README.md
(1 hunks).vscode/settings.json
(1 hunks)CODE_OF_CONDUCT.md
(0 hunks)README.md
(2 hunks)README_CN.md
(4 hunks)README_JA.md
(0 hunks)app/api/[provider]/[...path]/route.ts
(2 hunks)app/api/alibaba.ts
(5 hunks)app/api/anthropic.ts
(5 hunks)app/api/artifacts/route.ts
(4 hunks)app/api/auth.ts
(3 hunks)app/api/azure.ts
(2 hunks)app/api/baidu.ts
(4 hunks)app/api/bytedance.ts
(4 hunks)app/api/common.ts
(4 hunks)app/api/config/route.ts
(2 hunks)app/api/glm.ts
(5 hunks)app/api/google.ts
(4 hunks)app/api/iflytek.ts
(6 hunks)app/api/moonshot.ts
(5 hunks)app/api/openai.ts
(4 hunks)app/api/proxy.ts
(2 hunks)app/api/stability.ts
(4 hunks)app/api/tencent/route.ts
(4 hunks)app/api/upstash/[action]/[...key]/route.ts
(3 hunks)app/api/webdav/[...path]/route.ts
(8 hunks)app/api/xai.ts
(5 hunks)app/client/api.ts
(10 hunks)app/client/controller.ts
(1 hunks)app/client/platforms/alibaba.ts
(11 hunks)app/client/platforms/anthropic.ts
(15 hunks)app/client/platforms/baidu.ts
(12 hunks)app/client/platforms/bytedance.ts
(10 hunks)app/client/platforms/glm.ts
(6 hunks)app/client/platforms/google.ts
(11 hunks)app/client/platforms/iflytek.ts
(9 hunks)app/client/platforms/moonshot.ts
(6 hunks)app/client/platforms/openai.ts
(17 hunks)app/client/platforms/tencent.ts
(12 hunks)app/client/platforms/xai.ts
(6 hunks)app/command.ts
(3 hunks)app/components/artifacts.tsx
(10 hunks)app/components/auth.tsx
(6 hunks)app/components/button.tsx
(4 hunks)app/components/chat-list.tsx
(7 hunks)app/components/chat.module.scss
(2 hunks)app/components/chat.tsx
(67 hunks)app/components/emoji.tsx
(3 hunks)app/components/error.tsx
(2 hunks)app/components/exporter.module.scss
(3 hunks)app/components/exporter.tsx
(28 hunks)app/components/home.tsx
(7 hunks)app/components/input-range.tsx
(3 hunks)app/components/markdown.tsx
(14 hunks)app/components/mask.tsx
(27 hunks)app/components/message-selector.tsx
(6 hunks)app/components/model-config.tsx
(7 hunks)app/components/new-chat.module.scss
(1 hunks)app/components/new-chat.tsx
(5 hunks)app/components/plugin.module.scss
(1 hunks)app/components/plugin.tsx
(12 hunks)app/components/realtime-chat/index.ts
(1 hunks)app/components/realtime-chat/realtime-chat.tsx
(10 hunks)
⛔ Files not processed due to max files limit (24)
- app/components/realtime-chat/realtime-config.tsx
- app/components/sd/index.tsx
- app/components/sd/sd-panel.tsx
- app/components/sd/sd-sidebar.tsx
- app/components/sd/sd.module.scss
- app/components/sd/sd.tsx
- app/components/search-chat.tsx
- app/components/settings.module.scss
- app/components/settings.tsx
- app/components/sidebar.tsx
- app/components/tts-config.tsx
- app/components/ui-lib.module.scss
- app/components/ui-lib.tsx
- app/components/voice-print/index.ts
- app/components/voice-print/voice-print.tsx
- app/config/build.ts
- app/config/client.ts
- app/config/server.ts
- app/constant.ts
- app/global.d.ts
- app/layout.tsx
- app/lib/audio.ts
- app/locales/ar.ts
- app/locales/bn.ts
💤 Files with no reviewable changes (11)
- .eslintignore
- .gitpod.yml
- .gitignore
- .eslintrc.json
- .lintstagedrc.json
- .github/workflows/app.yml
- CODE_OF_CONDUCT.md
- .github/workflows/remove_deploy_preview.yml
- README_JA.md
- .github/workflows/test.yml
- .github/workflows/deploy_preview.yml
✅ Files skipped from review due to trivial changes (26)
- .github/dependabot.yml
- app/api/config/route.ts
- app/command.ts
- app/client/controller.ts
- .npmrc
- app/api/google.ts
- app/api/azure.ts
- .github/ISSUE_TEMPLATE/2_feature_request.yml
- app/components/exporter.module.scss
- app/components/error.tsx
- .github/PULL_REQUEST_TEMPLATE.md
- .github/workflows/issue-translator.yml
- app/api/common.ts
- app/components/realtime-chat/index.ts
- app/api/tencent/route.ts
- app/api/auth.ts
- app/components/new-chat.module.scss
- .vscode/settings.json
- app/api/proxy.ts
- app/api/[provider]/[...path]/route.ts
- app/components/chat.module.scss
- app/components/button.tsx
- app/client/platforms/iflytek.ts
- .github/ISSUE_TEMPLATE/1_bug_report_cn.yml
- app/components/plugin.module.scss
- app/api/upstash/[action]/[...key]/route.ts
🧰 Additional context used
🪛 LanguageTool
.vscode/README.md
[uncategorized] ~5-~5: 您的意思是“"需要"”吗?
Context: ## VSCode ### 1. 扩展安装 > 以下为项目初始化必须要安装的扩展,若有冲突的扩展项,请先卸载掉或者去掉全局配置 1. `ESLint...
(XU7_XU8)
🪛 actionlint (1.7.4)
.github/workflows/docker.yml
14-14: the runner of "actions/checkout@v3" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
16-16: the runner of "docker/login-action@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
23-23: the runner of "docker/metadata-action@v4" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
31-31: the runner of "docker/setup-qemu-action@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
34-34: the runner of "docker/setup-buildx-action@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
37-37: the runner of "docker/build-push-action@v4" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
.github/workflows/sync.yml
20-20: the runner of "actions/checkout@v3" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
🪛 Biome (1.9.4)
app/components/model-config.tsx
[error] 66-68: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 87-89: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 108-110: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 133-136: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 156-159: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 177-178: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 194-194: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 170-181: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 188-197: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 214-214: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 233-234: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 246-246: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 240-249: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
app/components/plugin.tsx
[error] 328-333: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 354-356: Avoid passing both children and the dangerouslySetInnerHTML prop.
This is the source of the children prop
Setting HTML content will inadvertently override any passed children in React
(lint/security/noDangerouslySetInnerHtmlWithChildren)
[error] 354-354: Avoid passing content using the dangerouslySetInnerHTML prop.
Setting content using code can expose users to cross-site scripting (XSS) attacks
(lint/security/noDangerouslySetInnerHtml)
app/client/platforms/google.ts
[error] 79-79: This aliasing of this is unnecessary.
Arrow functions inherits this
from their enclosing scope.
Safe fix: Use this instead of an alias.
(lint/complexity/noUselessThisAlias)
app/components/message-selector.tsx
[error] 152-162: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
app/components/mask.tsx
[error] 121-121: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 131-131: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 154-163: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 169-179: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 187-197: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 241-262: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
app/client/platforms/openai.ts
[error] 430-431: Change to an optional chain.
Unsafe fix: Change to an optional chain.
(lint/complexity/useOptionalChain)
app/components/exporter.tsx
[error] 221-221: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 240-240: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 235-244: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 329-340: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
app/components/new-chat.tsx
[error] 134-134: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
app/components/chat.tsx
[error] 160-160: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 186-186: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 373-373: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 492-492: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 605-619: Avoid using unnecessary Fragment.
A fragment is redundant if it contains only one child, or if it is the child of a html element, and is not a keyed fragment.
Unsafe fix: Remove the Fragment
(lint/complexity/noUselessFragments)
[error] 856-856: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 874-874: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 868-877: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
[error] 1052-1053: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 1427-1428: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 1452-1453: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 1456-1457: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 1458-1459: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
app/components/auth.tsx
[error] 75-75: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 93-93: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 106-106: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
app/components/input-range.tsx
[error] 29-39: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
🔇 Additional comments (37)
.github/workflows/docker.yml (2)
4-6
: LGTM: Trigger configuration is properly formatted
The indentation of workflow triggers is consistent and follows YAML best practices.
9-12
: LGTM: Job configuration is well-structured
The job configuration follows proper YAML formatting and maintains good readability.
.github/workflows/sync.yml (4)
35-40
: Well-structured error handling
The error handling is well implemented with:
- Clear bilingual error messages
- Helpful documentation links
- Consistent formatting
7-9
: Well-configured workflow triggers
The combination of daily automated runs and manual trigger capability provides a good balance between automation and flexibility.
3-4
: Appropriate permissions configuration
The workflow uses the minimum required permissions (contents: write) needed for sync operations, following security best practices.
27-29
: Verify the impact of target branch change
The target sync branch has been changed from 'main' to 'sync-main'. This change could affect existing automation or PR processes.
✅ Verification successful
Let me gather more context about the workflow configurations and branch usage.
Let me check the content of other workflow files to understand if they have any dependencies on the branch names.
Based on the analysis of the workflow files and branch references, I can now provide the final response:
Target branch change is safe to proceed
The change from 'main' to 'sync-main' in the sync workflow is isolated and won't impact other processes because:
- The sync workflow is self-contained and only handles upstream synchronization
- Other workflow files (docker.yml and issue-translator.yml) don't have any dependencies on branch names
- The sync workflow is only triggered for forks and uses a dedicated branch for synchronization
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Check for any references to the old branch name in other workflow files
echo "Checking for 'main' branch references in other workflow files..."
rg -l "branch: main" .github/workflows/
Length of output: 213
Script:
#!/bin/bash
# Check the full content of sync.yml to understand the workflow context
echo "Checking sync.yml content..."
cat .github/workflows/sync.yml
# Check for any other workflows that might depend on branch names
echo -e "\nChecking for other workflow files..."
fd -e yml -e yaml . .github/workflows/
# Check for branch references in other configuration files
echo -e "\nChecking for branch references in other configuration files..."
rg -l "branch.*main" --type yaml --type json
Length of output: 2246
Script:
#!/bin/bash
# Check contents of other workflow files
echo "Checking docker.yml content..."
cat .github/workflows/docker.yml
echo -e "\nChecking issue-translator.yml content..."
cat .github/workflows/issue-translator.yml
# Check for any PRs or branch references in the codebase
echo -e "\nChecking for PR or branch references..."
rg "sync-main|branch: main" -A 3 -B 3
Length of output: 2219
app/client/platforms/tencent.ts (1)
1-27
: LGTM! Import statements are well-organized.
The import statements follow a consistent style with logical grouping and proper use of single quotes.
.vscode/README.md (1)
1-15
: Minor Linguistic and Formatting Improvements
The new section effectively lists the recommended VSCode extensions and provides brief, clear instructions. However, you could refine the text slightly for more precise language and to ensure it reads smoothly for all users. For instance, consider rewriting the note on uninstalling conflicting extensions to mention alternative choices or clarifications on how to adjust global settings.
🧰 Tools
🪛 LanguageTool
[uncategorized] ~5-~5: 您的意思是“"需要"”吗?
Context: ## VSCode ### 1. 扩展安装 > 以下为项目初始化必须要安装的扩展,若有冲突的扩展项,请先卸载掉或者去掉全局配置 1. `ESLint...
(XU7_XU8)
.github/ISSUE_TEMPLATE/2_feature_request_cn.yml (1)
1-21
: Good Structural Clarity and Formatting
These changes align well with a style-based PR: removing quotes around simpler YAML fields and improving indentation helps maintain a consistent, readable YAML-based issue template. Verify that these updated fields remain valid under GitHub’s issue template parsing. The changes do not appear to alter functionality or content.
README.md (1)
Line range hint 5-26
: Consistent Bilingual Support and Content Streamlining
The introduction of a direct link to the Chinese documentation while removing enterprise-specific content strikes a good balance for an open-source project. The overall formatting and bilingual clarity are improved. Ensure that any references to enterprise editions in other documentation or code comments are similarly removed or updated to maintain consistency across the project.
app/components/emoji.tsx (3)
1-4
: LGTM: Clean import organization
The explicit type imports and consistent use of single quotes improve code organization.
26-26
: LGTM: Simplified JSX syntax
Changed from {"100%"}
to "100%"
for cleaner JSX.
41-49
: LGTM: Improved conditional formatting
The reformatted conditions enhance readability while maintaining the same logic.
app/api/openai.ts (3)
1-8
: LGTM: Consistent import organization
Clean separation of type imports and regular imports with consistent quote style.
17-19
: LGTM: Improved filter readability
The filter condition formatting enhances readability while maintaining the same logic.
39-44
: LGTM: Consistent string formatting
Error message formatting now uses template literals consistently.
app/api/artifacts/route.ts (3)
1-4
: LGTM: Clean import organization
Consistent type imports and ordering.
25-27
: LGTM: Improved number parsing
Using Number.parseInt
is more explicit than parseInt
.
34-35
: LGTM: Consistent string formatting
Standardized use of single quotes for string literals across the file.
Also applies to: 49-50, 66-67
app/api/stability.ts (1)
89-91
: LGTM! Good security practice
Properly handling authentication headers and disabling buffering.
app/api/xai.ts (1)
117-119
: LGTM! Consistent with security practices
Header handling is consistent with the stability API implementation.
app/api/bytedance.ts (1)
1-12
: LGTM: Style changes are consistent
The changes standardize string literals to use single quotes throughout the file, which aligns with the PR's style objectives.
Also applies to: 20-24, 37-38, 71-72, 76-79, 118-120
app/api/iflytek.ts (1)
1-12
: LGTM: Style changes are consistent
The changes maintain consistency with the project's style guidelines by using single quotes for string literals.
Also applies to: 21-24, 38-38, 72-73, 77-79, 118-120
app/api/moonshot.ts (2)
1-12
: LGTM: Style changes are consistent
The changes successfully standardize string literals to use single quotes, maintaining consistency with the project's style guidelines.
Also applies to: 20-23, 37-37, 71-72, 76-78, 117-119
Line range hint 1-120
: Verify consistent style across API route handlers
The style changes are consistently applied across all API route handlers, successfully standardizing string literals to use single quotes. The changes maintain functional equivalence while improving code consistency.
app/api/glm.ts (2)
1-12
: LGTM: Import statements are well organized
The imports are logically grouped and consistently use single quotes, improving readability.
20-23
: LGTM: String literal standardization
Console logs and string literals have been consistently updated to use single quotes, maintaining style consistency across the file.
Also applies to: 37-37, 58-59, 69-69, 118-118, 120-120
app/api/alibaba.ts (2)
1-12
: LGTM: Import organization matches other files
Import statements follow the same organization pattern as glm.ts, maintaining consistency across the codebase.
71-73
: LGTM: Headers consistently use single quotes
Header definitions, including the Alibaba-specific 'X-DashScope-SSE', follow the single-quote convention.
app/api/baidu.ts (2)
1-13
: LGTM: Import organization maintains consistency
Import statements follow the same pattern as other API files, contributing to codebase consistency.
88-94
: LGTM: Fetch options maintain consistent style
Headers and options consistently use single quotes, matching the style in other API files.
app/api/webdav/[...path]/route.ts (1)
13-19
: LGTM: Function declaration style improvement
The change from arrow function to regular function declaration for normalizeUrl
improves readability and follows better practices for function declarations.
app/client/platforms/moonshot.ts (1)
159-159
: Consider adding type safety for tool calls handling
Similar to xai.ts, this implementation uses @ts-ignore
to bypass TypeScript checks.
Please apply the same type-safe solution as suggested for xai.ts.
app/components/input-range.tsx (1)
1-3
: LGTM on style changes
The import statement updates and className formatting changes align well with the PR's style standardization goals.
Also applies to: 27-27
app/components/auth.tsx (3)
1-20
: LGTM on import organization
The import statements have been well organized and consistently use single quotes, aligning with the PR's style standardization goals.
49-51
: LGTM on JSX style updates
The className and style attribute updates maintain consistency with single quotes and proper formatting throughout the component.
Also applies to: 59-59, 63-64, 80-112, 114-114, 166-170, 171-171, 182-182, 187-187
137-142
: LGTM on localStorage interactions
String literal updates in localStorage operations align with the overall style standardization.
Also applies to: 158-158
- name: Check out the repo | ||
uses: actions/checkout@v3 | ||
- name: Log in to Docker Hub | ||
uses: docker/login-action@v2 | ||
with: | ||
username: ${{ secrets.DOCKER_USERNAME }} | ||
password: ${{ secrets.DOCKER_PASSWORD }} | ||
|
||
- | ||
name: Set up Docker Buildx | ||
uses: docker/setup-buildx-action@v2 | ||
|
||
- | ||
name: Build and push Docker image | ||
uses: docker/build-push-action@v4 | ||
with: | ||
context: . | ||
platforms: linux/amd64,linux/arm64 | ||
push: true | ||
tags: ${{ steps.meta.outputs.tags }} | ||
labels: ${{ steps.meta.outputs.labels }} | ||
cache-from: type=gha | ||
cache-to: type=gha,mode=max | ||
|
||
- name: Extract metadata (tags, labels) for Docker | ||
id: meta | ||
uses: docker/metadata-action@v4 | ||
with: | ||
images: yidadaa/chatgpt-next-web | ||
tags: | | ||
type=raw,value=latest | ||
type=ref,event=tag | ||
|
||
- name: Set up QEMU | ||
uses: docker/setup-qemu-action@v2 | ||
|
||
- name: Set up Docker Buildx | ||
uses: docker/setup-buildx-action@v2 | ||
|
||
- name: Build and push Docker image | ||
uses: docker/build-push-action@v4 | ||
with: | ||
context: . | ||
platforms: linux/amd64,linux/arm64 | ||
push: true | ||
tags: ${{ steps.meta.outputs.tags }} | ||
labels: ${{ steps.meta.outputs.labels }} | ||
cache-from: type=gha | ||
cache-to: type=gha,mode=max |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
💡 Codebase verification
🛠️ Refactor suggestion
Update GitHub Action versions to their latest releases
Several actions in the workflow are using outdated versions. Update them to the latest stable versions for enhanced security, compatibility, and features:
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
- uses: docker/login-action@v2
+ uses: docker/login-action@v3
- uses: docker/metadata-action@v4
+ uses: docker/metadata-action@v5
- uses: docker/setup-qemu-action@v2
+ uses: docker/setup-qemu-action@v3
- uses: docker/setup-buildx-action@v2
+ uses: docker/setup-buildx-action@v3
- uses: docker/build-push-action@v4
+ uses: docker/build-push-action@v6
🔗 Analysis chain
Update GitHub Action versions to their latest releases
Several actions in the workflow are using older versions. It's recommended to update them to their latest stable versions for better security, compatibility, and features.
Apply these version updates:
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
- uses: docker/login-action@v2
+ uses: docker/login-action@v3
- uses: docker/metadata-action@v4
+ uses: docker/metadata-action@v5
- uses: docker/setup-qemu-action@v2
+ uses: docker/setup-qemu-action@v3
- uses: docker/setup-buildx-action@v2
+ uses: docker/setup-buildx-action@v3
- uses: docker/build-push-action@v4
+ uses: docker/build-push-action@v5
🏁 Scripts executed
The following scripts were executed for the analysis:
Script:
#!/bin/bash
# Verify latest versions of the actions
gh api repos/actions/checkout/releases/latest | jq .tag_name
gh api repos/docker/login-action/releases/latest | jq .tag_name
gh api repos/docker/metadata-action/releases/latest | jq .tag_name
gh api repos/docker/setup-qemu-action/releases/latest | jq .tag_name
gh api repos/docker/setup-buildx-action/releases/latest | jq .tag_name
gh api repos/docker/build-push-action/releases/latest | jq .tag_name
Length of output: 467
🧰 Tools
🪛 actionlint (1.7.4)
14-14: the runner of "actions/checkout@v3" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
16-16: the runner of "docker/login-action@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
23-23: the runner of "docker/metadata-action@v4" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
31-31: the runner of "docker/setup-qemu-action@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
34-34: the runner of "docker/setup-buildx-action@v2" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
37-37: the runner of "docker/build-push-action@v4" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
{!state?.fromHome && ( | ||
<IconButton | ||
text={Locale.NewChat.NotShow} | ||
onClick={async () => { | ||
if (await showConfirm(Locale.NewChat.ConfirmNoShow)) { | ||
startChat(); | ||
config.update( | ||
(config) => (config.dontShowMaskSplashScreen = true), | ||
config => (config.dontShowMaskSplashScreen = true), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Fix assignment in expression
The assignment within the arrow function could be clearer.
- config => (config.dontShowMaskSplashScreen = true),
+ config => {
+ config.dontShowMaskSplashScreen = true;
+ return config;
+ }
📝 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.
config => (config.dontShowMaskSplashScreen = true), | |
config => { | |
config.dontShowMaskSplashScreen = true; | |
return config; | |
} |
🧰 Tools
🪛 Biome (1.9.4)
[error] 134-134: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
@@ -149,7 +153,7 @@ export class ChatGLMApi implements LLMApi { | |||
}); | |||
} else { | |||
// @ts-ignore | |||
runTools[index]["function"]["arguments"] += args; | |||
runTools[index].function.arguments += args; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove @ts-ignore and handle type properly
Using @ts-ignore masks potential type issues that should be handled properly.
Consider adding proper type checking or defining an interface for the runTools array elements:
interface ToolCall {
function: {
arguments: string;
name: string;
};
id: string;
type: string;
index?: number;
}
// Then use type assertion or proper type checking
(runTools[index] as ToolCall).function.arguments += args;
@@ -145,7 +149,7 @@ export class XAIApi implements LLMApi { | |||
}); | |||
} else { | |||
// @ts-ignore | |||
runTools[index]["function"]["arguments"] += args; | |||
runTools[index].function.arguments += args; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Consider adding type safety for tool calls handling
The current implementation uses @ts-ignore
to bypass TypeScript checks. This could lead to runtime errors if the array index or property access is invalid.
Consider adding proper type checking:
- // @ts-ignore
- runTools[index].function.arguments += args;
+ if (index !== undefined && index < runTools.length && runTools[index]?.function) {
+ runTools[index].function.arguments += args;
+ }
📝 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.
runTools[index].function.arguments += args; | |
if (index !== undefined && index < runTools.length && runTools[index]?.function) { | |
runTools[index].function.arguments += args; | |
} |
let responseText = ''; | ||
let remainText = ''; | ||
let finished = false; | ||
let responseRes: Response; | ||
|
||
// animate response to make it looks smooth | ||
function animateResponseText() { | ||
if (finished || controller.signal.aborted) { | ||
responseText += remainText; | ||
console.log("[Response Animation] finished"); | ||
console.log('[Response Animation] finished'); | ||
if (responseText?.length === 0) { | ||
options.onError?.(new Error("empty response from server")); | ||
options.onError?.(new Error('empty response from server')); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Enhance error handling with more specific error messages.
The error handling could be more informative by providing specific error messages.
if (responseText?.length === 0) {
- options.onError?.(new Error('empty response from server'));
+ options.onError?.(new Error('ByteDance API returned an empty response. Please check your API configuration and try again.'));
}
📝 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.
let responseText = ''; | |
let remainText = ''; | |
let finished = false; | |
let responseRes: Response; | |
// animate response to make it looks smooth | |
function animateResponseText() { | |
if (finished || controller.signal.aborted) { | |
responseText += remainText; | |
console.log("[Response Animation] finished"); | |
console.log('[Response Animation] finished'); | |
if (responseText?.length === 0) { | |
options.onError?.(new Error("empty response from server")); | |
options.onError?.(new Error('empty response from server')); | |
let responseText = ''; | |
let remainText = ''; | |
let finished = false; | |
let responseRes: Response; | |
// animate response to make it looks smooth | |
function animateResponseText() { | |
if (finished || controller.signal.aborted) { | |
responseText += remainText; | |
console.log('[Response Animation] finished'); | |
if (responseText?.length === 0) { | |
options.onError?.(new Error('ByteDance API returned an empty response. Please check your API configuration and try again.')); |
? accessStore.azureApiKey | ||
: isAnthropic | ||
? accessStore.anthropicApiKey | ||
: isByteDance | ||
? accessStore.bytedanceApiKey | ||
: isAlibaba | ||
? accessStore.alibabaApiKey | ||
: isMoonshot | ||
? accessStore.moonshotApiKey | ||
: isXAI | ||
? accessStore.xaiApiKey | ||
: isChatGLM | ||
? accessStore.chatglmApiKey | ||
: isIflytek | ||
? accessStore.iflytekApiKey && accessStore.iflytekApiSecret | ||
? `${accessStore.iflytekApiKey}:${accessStore.iflytekApiSecret}` | ||
: '' | ||
: accessStore.openaiApiKey; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Simplify nested ternary operators
The deeply nested ternary operators make the code hard to read and maintain.
Consider using a more readable approach:
function getApiKey() {
const providers = {
[ServiceProvider.Google]: accessStore.googleApiKey,
[ServiceProvider.Azure]: accessStore.azureApiKey,
[ServiceProvider.Anthropic]: accessStore.anthropicApiKey,
// ... other providers
};
if (isIflytek && accessStore.iflytekApiKey && accessStore.iflytekApiSecret) {
return `${accessStore.iflytekApiKey}:${accessStore.iflytekApiSecret}`;
}
return providers[modelConfig.providerName] || accessStore.openaiApiKey;
}
const { url = '' } = image_url || {}; | ||
const colonIndex = url.indexOf(':'); | ||
const semicolonIndex = url.indexOf(';'); | ||
const comma = url.indexOf(','); | ||
|
||
const mimeType = url.slice(colonIndex + 1, semicolonIndex); | ||
const encodeType = url.slice(semicolonIndex + 1, comma); | ||
const data = url.slice(comma + 1); | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Extract image URL parsing logic into a separate function.
The image URL parsing logic is complex and could be more maintainable if extracted.
Apply this diff to improve maintainability:
+ const parseImageUrl = (url: string) => {
+ const colonIndex = url.indexOf(':');
+ const semicolonIndex = url.indexOf(';');
+ const comma = url.indexOf(',');
+ return {
+ mimeType: url.slice(colonIndex + 1, semicolonIndex),
+ encodeType: url.slice(semicolonIndex + 1, comma),
+ data: url.slice(comma + 1),
+ };
+ };
- const colonIndex = url.indexOf(':');
- const semicolonIndex = url.indexOf(';');
- const comma = url.indexOf(',');
-
- const mimeType = url.slice(colonIndex + 1, semicolonIndex);
- const encodeType = url.slice(semicolonIndex + 1, comma);
- const data = url.slice(comma + 1);
+ const { mimeType, encodeType, data } = parseImageUrl(url);
📝 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.
const { url = '' } = image_url || {}; | |
const colonIndex = url.indexOf(':'); | |
const semicolonIndex = url.indexOf(';'); | |
const comma = url.indexOf(','); | |
const mimeType = url.slice(colonIndex + 1, semicolonIndex); | |
const encodeType = url.slice(semicolonIndex + 1, comma); | |
const data = url.slice(comma + 1); | |
const parseImageUrl = (url: string) => { | |
const colonIndex = url.indexOf(':'); | |
const semicolonIndex = url.indexOf(';'); | |
const comma = url.indexOf(','); | |
return { | |
mimeType: url.slice(colonIndex + 1, semicolonIndex), | |
encodeType: url.slice(semicolonIndex + 1, comma), | |
data: url.slice(comma + 1), | |
}; | |
}; | |
const { url = '' } = image_url || {}; | |
const { mimeType, encodeType, data } = parseImageUrl(url); |
<input | ||
type="text" | ||
style={{ minWidth: 200 }} | ||
onInput={(e) => setLoadUrl(e.currentTarget.value)} | ||
></input> | ||
onInput={e => setLoadUrl(e.currentTarget.value)} | ||
> | ||
</input> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fix void element usage.
The input element is a void element and should not have children.
Apply this diff to fix the HTML structure:
- <input
- type="text"
- style={{ minWidth: 200 }}
- onInput={e => setLoadUrl(e.currentTarget.value)}
- >
- </input>
+ <input
+ type="text"
+ style={{ minWidth: 200 }}
+ onInput={e => setLoadUrl(e.currentTarget.value)}
+ />
📝 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.
<input | |
type="text" | |
style={{ minWidth: 200 }} | |
onInput={(e) => setLoadUrl(e.currentTarget.value)} | |
></input> | |
onInput={e => setLoadUrl(e.currentTarget.value)} | |
> | |
</input> | |
<input | |
type="text" | |
style={{ minWidth: 200 }} | |
onInput={e => setLoadUrl(e.currentTarget.value)} | |
/> |
🧰 Tools
🪛 Biome (1.9.4)
[error] 328-333: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
contentEditable | ||
dangerouslySetInnerHTML={{ | ||
__html: editingPlugin.content, | ||
}} | ||
onBlur={onChangePlugin} | ||
></code> | ||
> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Security: Avoid using dangerouslySetInnerHTML with contentEditable.
Using both dangerouslySetInnerHTML and contentEditable can lead to XSS vulnerabilities.
Apply this diff to use controlled components instead:
- <code
- contentEditable
- dangerouslySetInnerHTML={{
- __html: editingPlugin.content,
- }}
- onBlur={onChangePlugin}
- >
- </code>
+ <textarea
+ value={editingPlugin.content}
+ onChange={(e) => onChangePlugin({ target: { innerText: e.target.value } })}
+ style={{ width: '100%', minHeight: '200px' }}
+ />
📝 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.
contentEditable | |
dangerouslySetInnerHTML={{ | |
__html: editingPlugin.content, | |
}} | |
onBlur={onChangePlugin} | |
></code> | |
> | |
<textarea | |
value={editingPlugin.content} | |
onChange={(e) => onChangePlugin({ target: { innerText: e.target.value } })} | |
style={{ width: '100%', minHeight: '200px' }} | |
/> |
🧰 Tools
🪛 Biome (1.9.4)
[error] 354-356: Avoid passing both children and the dangerouslySetInnerHTML prop.
This is the source of the children prop
Setting HTML content will inadvertently override any passed children in React
(lint/security/noDangerouslySetInnerHtmlWithChildren)
[error] 354-354: Avoid passing content using the dangerouslySetInnerHTML prop.
Setting content using code can expose users to cross-site scripting (XSS) attacks
(lint/security/noDangerouslySetInnerHtml)
> | ||
</input> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fix invalid HTML structure for input element
The input
element is a void element and must not have children or closing tags. It should be self-closing.
- >
- </input>
+ />
📝 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.
> | |
</input> | |
/> |
🧰 Tools
🪛 Biome (1.9.4)
[error] 29-39: input is a void element tag and must not have children.
Unsafe fix: Remove the children.
(lint/correctness/noVoidElementsWithChildren)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 4
🧹 Nitpick comments (4)
.github/workflows/release-notes.js (2)
1-5
: Consider adding file-level documentation.Since this is a new script file, it would be beneficial to add a brief comment block explaining its purpose and usage.
#!/usr/bin/env node +/** + * Generates release notes for a specific version by extracting relevant sections + * from CHANGELOG.md. + * + * Usage: release-notes.js <version-tag> + */ + import { readFileSync, writeFileSync } from 'node:fs'; import process from 'node:process';
8-24
: Consider extracting the regex pattern and adding error handling.The main logic could benefit from some improvements in error handling and maintainability.
+// Regex pattern for matching version headers in changelog +const VERSION_HEADER_PATTERN = /^#+ \[/; +const versionRegex = new RegExp(`${VERSION_HEADER_PATTERN.source}${tag}`); let result = ''; let inScope = false; -const regex = new RegExp(`^#+ \\[${tag}`); for (let i = 0; i < log.length; i++) { - if (regex.test(log[i])) { + if (versionRegex.test(log[i])) { inScope = true; result += log[i]; continue; } - if (inScope && /^#+ \[/.test(log[i])) { + if (inScope && VERSION_HEADER_PATTERN.test(log[i])) { inScope = false; break; } if (inScope) { result += `\n${log[i]}`; } } + +if (!result) { + console.error(`Error: No release notes found for version ${tag}`); + process.exit(1); +}.github/workflows/release-tag.yml (2)
1-11
: Add workflow documentation and improve version tag pattern.Consider adding workflow documentation and being more specific with the version tag pattern.
+# Workflow for automatically creating GitHub releases when version tags are pushed +# This workflow is triggered when a version tag (v*) is pushed to the repository + on: push: tags: - - 'v*' # Push events to matching v*, i.e. v1.0, v20.15.10 + - 'v[0-9]+.[0-9]+.[0-9]+' # Semantic version tags only (e.g., v1.0.0) + - 'v[0-9]+.[0-9]+.[0-9]+-*' # Pre-release tags (e.g., v1.0.0-beta.1) name: Create Release
26-37
: Add error handling for release notes generation.Consider adding error handling and verification for the release notes generation step.
- name: Generate Release Notes + id: generate_notes run: | ./.github/workflows/release-notes.js ${{ env.RELEASE_VERSION }} - cat notes-${{ env.RELEASE_VERSION }}.md + if [ ! -f "notes-${{ env.RELEASE_VERSION }}.md" ]; then + echo "Error: Release notes file was not generated" + exit 1 + fi + echo "Generated release notes:" + cat "notes-${{ env.RELEASE_VERSION }}.md" - name: Create Release for Tag id: release_tag uses: softprops/action-gh-release@v2 env: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} with: body_path: notes-${{ env.RELEASE_VERSION }}.md + draft: true # Consider setting as draft for review before publishing🧰 Tools
🪛 actionlint (1.7.4)
33-33: the runner of "softprops/action-gh-release@v1" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
⛔ Files ignored due to path filters (1)
pnpm-lock.yaml
is excluded by!**/pnpm-lock.yaml
📒 Files selected for processing (3)
.github/workflows/release-notes.js
(1 hunks).github/workflows/release-tag.yml
(1 hunks).prettierrc.js
(0 hunks)
💤 Files with no reviewable changes (1)
- .prettierrc.js
🧰 Additional context used
🪛 actionlint (1.7.4)
.github/workflows/release-tag.yml
23-23: shellcheck reported issue in this script: SC2086:info:1:52: Double quote to prevent globbing and word splitting
(shellcheck)
33-33: the runner of "softprops/action-gh-release@v1" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
const tag = process.argv[2].replace('v', ''); | ||
const log = readFileSync('./CHANGELOG.md', { encoding: 'utf-8' }).split('\n'); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Add input validation for the version tag argument.
The script assumes the version tag is always provided and valid. Consider adding validation to handle missing or invalid arguments.
+if (process.argv.length < 3) {
+ console.error('Error: Version tag argument is required');
+ process.exit(1);
+}
+
const tag = process.argv[2].replace('v', '');
+
+if (!tag) {
+ console.error('Error: Invalid version tag');
+ process.exit(1);
+}
+
const log = readFileSync('./CHANGELOG.md', { encoding: 'utf-8' }).split('\n');
📝 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.
const tag = process.argv[2].replace('v', ''); | |
const log = readFileSync('./CHANGELOG.md', { encoding: 'utf-8' }).split('\n'); | |
if (process.argv.length < 3) { | |
console.error('Error: Version tag argument is required'); | |
process.exit(1); | |
} | |
const tag = process.argv[2].replace('v', ''); | |
if (!tag) { | |
console.error('Error: Invalid version tag'); | |
process.exit(1); | |
} | |
const log = readFileSync('./CHANGELOG.md', { encoding: 'utf-8' }).split('\n'); |
result += `\n${log[i]}`; | ||
} | ||
} | ||
writeFileSync(`notes-v${tag}.md`, result); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Add error handling for file operations.
File operations should be wrapped in try-catch blocks to handle potential errors gracefully.
-writeFileSync(`notes-v${tag}.md`, result);
+try {
+ writeFileSync(`notes-v${tag}.md`, result);
+ console.log(`Successfully generated release notes for v${tag}`);
+} catch (error) {
+ console.error('Error writing release notes:', error.message);
+ process.exit(1);
+}
📝 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.
writeFileSync(`notes-v${tag}.md`, result); | |
try { | |
writeFileSync(`notes-v${tag}.md`, result); | |
console.log(`Successfully generated release notes for v${tag}`); | |
} catch (error) { | |
console.error('Error writing release notes:', error.message); | |
process.exit(1); | |
} |
- name: Get the release version from the tag | ||
shell: bash | ||
if: env.RELEASE_VERSION == '' | ||
run: | | ||
echo "RELEASE_VERSION=${GITHUB_REF#refs/tags/}" >> $GITHUB_ENV | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Fix shellcheck warning and add version validation.
Address the shellcheck warning and add basic semantic version validation.
- name: Get the release version from the tag
shell: bash
if: env.RELEASE_VERSION == ''
run: |
- echo "RELEASE_VERSION=${GITHUB_REF#refs/tags/}" >> $GITHUB_ENV
+ VERSION="${GITHUB_REF#refs/tags/}"
+ if [[ ! "$VERSION" =~ ^v[0-9]+\.[0-9]+\.[0-9]+(-[0-9A-Za-z-]+(\.[0-9A-Za-z-]+)*)?$ ]]; then
+ echo "Error: Invalid version tag format: $VERSION"
+ exit 1
+ fi
+ echo "RELEASE_VERSION=\"$VERSION\"" >> "$GITHUB_ENV"
📝 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.
- name: Get the release version from the tag | |
shell: bash | |
if: env.RELEASE_VERSION == '' | |
run: | | |
echo "RELEASE_VERSION=${GITHUB_REF#refs/tags/}" >> $GITHUB_ENV | |
- name: Get the release version from the tag | |
shell: bash | |
if: env.RELEASE_VERSION == '' | |
run: | | |
VERSION="${GITHUB_REF#refs/tags/}" | |
if [[ ! "$VERSION" =~ ^v[0-9]+\.[0-9]+\.[0-9]+(-[0-9A-Za-z-]+(\.[0-9A-Za-z-]+)*)?$ ]]; then | |
echo "Error: Invalid version tag format: $VERSION" | |
exit 1 | |
fi | |
echo "RELEASE_VERSION=\"$VERSION\"" >> "$GITHUB_ENV" | |
🧰 Tools
🪛 actionlint (1.7.4)
23-23: shellcheck reported issue in this script: SC2086:info:1:52: Double quote to prevent globbing and word splitting
(shellcheck)
|
||
- name: Create Release for Tag | ||
id: release_tag | ||
uses: softprops/action-gh-release@v1 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Update action version as recommended by actionlint.
The current version of softprops/action-gh-release is outdated.
- uses: softprops/action-gh-release@v1
+ uses: softprops/action-gh-release@v2
Committable suggestion skipped: line range outside the PR's diff.
🧰 Tools
🪛 actionlint (1.7.4)
33-33: the runner of "softprops/action-gh-release@v1" action is too old to run on GitHub Actions. update the action's version to fix this issue
(action)
💻 变更类型 | Change Type
🔀 变更说明 | Description of Change
📝 补充信息 | Additional Information
Summary by CodeRabbit
.vscode/README.md
outlining essential extensions for project initialization.