support interfaces info

This commit is contained in:
pompurin404 2024-08-19 10:10:22 +08:00
parent 7b4d60e280
commit 4aeac0ebb3
No known key found for this signature in database
7 changed files with 340 additions and 237 deletions

View File

@ -1,3 +1,4 @@
### New Features
- 添加覆写脚本执行日志功能
- 支持查看局域网网络信息

View File

@ -0,0 +1,5 @@
import os from 'os'
export function getInterfaces(): NodeJS.Dict<NetworkInterfaceInfo[]> {
return os.networkInterfaces()
}

View File

@ -51,6 +51,7 @@ import { getFilePath, openUWPTool, readTextFile, setupFirewall } from '../sys/mi
import { getRuntimeConfig, getRuntimeConfigStr } from '../core/factory'
import { isPortable, setPortable } from './dirs'
import { listWebdavBackups, webdavBackup, webdavRestore } from '../resolve/backup'
import { getInterfaces } from '../sys/interface'
function ipcErrorWrapper<T>( // eslint-disable-next-line @typescript-eslint/no-explicit-any
fn: (...args: any[]) => Promise<T> // eslint-disable-next-line @typescript-eslint/no-explicit-any
@ -139,6 +140,7 @@ export function registerIpcMainHandlers(): void {
ipcMain.handle('platform', () => process.platform)
ipcMain.handle('openUWPTool', ipcErrorWrapper(openUWPTool))
ipcMain.handle('setupFirewall', ipcErrorWrapper(setupFirewall))
ipcMain.handle('getInterfaces', getInterfaces)
ipcMain.handle('setPortable', (_e, portable) => ipcErrorWrapper(setPortable)(portable))
ipcMain.handle('isPortable', isPortable)
ipcMain.handle('webdavBackup', ipcErrorWrapper(webdavBackup))

View File

@ -0,0 +1,67 @@
import {
Modal,
ModalContent,
ModalHeader,
ModalBody,
ModalFooter,
Button,
Snippet
} from '@nextui-org/react'
import React, { useEffect, useState } from 'react'
import { getInterfaces } from '@renderer/utils/ipc'
interface Props {
onClose: () => void
}
const InterfaceModal: React.FC<Props> = (props) => {
const { onClose } = props
const [info, setInfo] = useState<Record<string, NetworkInterfaceInfo[]>>({})
const getInfo = async (): Promise<void> => {
setInfo(await getInterfaces())
}
useEffect(() => {
getInfo()
}, [])
return (
<Modal
backdrop="blur"
hideCloseButton
isOpen={true}
onOpenChange={onClose}
scrollBehavior="inside"
>
<ModalContent>
<ModalHeader className="flex"></ModalHeader>
<ModalBody>
{Object.entries(info).map(([key, value]) => {
return (
<div key={key}>
<h4 className="font-bold">{key}</h4>
{value.map((v) => {
return (
<div key={v.address}>
<div className="mt-2 flex justify-between">
{v.family}
<Snippet symbol="" size="sm">
{v.address}
</Snippet>
</div>
</div>
)
})}
</div>
)
})}
</ModalBody>
<ModalFooter>
<Button variant="light" onPress={onClose}>
</Button>
</ModalFooter>
</ModalContent>
</Modal>
)
}
export default InterfaceModal

View File

@ -5,8 +5,10 @@ import SettingItem from '@renderer/components/base/base-setting-item'
import { useAppConfig } from '@renderer/hooks/use-app-config'
import { useControledMihomoConfig } from '@renderer/hooks/use-controled-mihomo-config'
import { platform } from '@renderer/utils/init'
import { FaNetworkWired } from 'react-icons/fa'
import { restartCore } from '@renderer/utils/ipc'
import React, { useState } from 'react'
import InterfaceModal from '@renderer/components/mihomo/interface-modal'
const CoreMap = {
mihomo: '稳定版',
@ -43,128 +45,47 @@ const Mihomo: React.FC = () => {
const [externalControllerInput, setExternalControllerInput] = useState(externalController)
const [secretInput, setSecretInput] = useState(secret)
const [lanOpen, setLanOpen] = useState(false)
const onChangeNeedRestart = async (patch: Partial<IMihomoConfig>): Promise<void> => {
await patchControledMihomoConfig(patch)
await restartCore()
}
return (
<BasePage title="内核设置">
<SettingCard>
<SettingItem title="内核版本" divider>
<Select
className="w-[100px]"
size="sm"
selectedKeys={new Set([core])}
onSelectionChange={async (v) => {
try {
await patchAppConfig({ core: v.currentKey as 'mihomo' | 'mihomo-alpha' })
await restartCore()
} catch (e) {
alert(e)
} finally {
PubSub.publish('mihomo-core-changed')
}
}}
>
<SelectItem key="mihomo">{CoreMap['mihomo']}</SelectItem>
<SelectItem key="mihomo-alpha">{CoreMap['mihomo-alpha']}</SelectItem>
</Select>
</SettingItem>
<SettingItem title="混合端口" divider>
<div className="flex">
{mixedPortInput !== mixedPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'mixed-port': mixedPortInput })
}}
>
</Button>
)}
<Input
size="sm"
type="number"
<>
{lanOpen && <InterfaceModal onClose={() => setLanOpen(false)} />}
<BasePage title="内核设置">
<SettingCard>
<SettingItem title="内核版本" divider>
<Select
className="w-[100px]"
value={mixedPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setMixedPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
<SettingItem title="Socks端口" divider>
<div className="flex">
{socksPortInput !== socksPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'socks-port': socksPortInput })
}}
>
</Button>
)}
<Input
size="sm"
type="number"
className="w-[100px]"
value={socksPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setSocksPortInput(parseInt(v))
selectedKeys={new Set([core])}
onSelectionChange={async (v) => {
try {
await patchAppConfig({ core: v.currentKey as 'mihomo' | 'mihomo-alpha' })
await restartCore()
} catch (e) {
alert(e)
} finally {
PubSub.publish('mihomo-core-changed')
}
}}
/>
</div>
</SettingItem>
<SettingItem title="Http端口" divider>
<div className="flex">
{httpPortInput !== httpPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ port: httpPortInput })
}}
>
</Button>
)}
<Input
size="sm"
type="number"
className="w-[100px]"
value={httpPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setHttpPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
{platform !== 'win32' && (
<SettingItem title="Redir端口" divider>
>
<SelectItem key="mihomo">{CoreMap['mihomo']}</SelectItem>
<SelectItem key="mihomo-alpha">{CoreMap['mihomo-alpha']}</SelectItem>
</Select>
</SettingItem>
<SettingItem title="混合端口" divider>
<div className="flex">
{redirPortInput !== redirPort && (
{mixedPortInput !== mixedPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'redir-port': redirPortInput })
onChangeNeedRestart({ 'mixed-port': mixedPortInput })
}}
>
@ -175,26 +96,24 @@ const Mihomo: React.FC = () => {
size="sm"
type="number"
className="w-[100px]"
value={redirPortInput.toString()}
value={mixedPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setRedirPortInput(parseInt(v))
setMixedPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
)}
{platform === 'linux' && (
<SettingItem title="TProxy端口" divider>
<SettingItem title="Socks端口" divider>
<div className="flex">
{tproxyPortInput !== tproxyPort && (
{socksPortInput !== socksPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'tproxy-port': tproxyPortInput })
onChangeNeedRestart({ 'socks-port': socksPortInput })
}}
>
@ -205,151 +124,255 @@ const Mihomo: React.FC = () => {
size="sm"
type="number"
className="w-[100px]"
value={tproxyPortInput.toString()}
value={socksPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setTproxyPortInput(parseInt(v))
setSocksPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
)}
<SettingItem title="外部控制" divider>
<div className="flex">
{externalControllerInput !== externalController && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'external-controller': externalControllerInput })
}}
>
</Button>
)}
<SettingItem title="Http端口" divider>
<div className="flex">
{httpPortInput !== httpPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ port: httpPortInput })
}}
>
</Button>
)}
<Input
<Input
size="sm"
type="number"
className="w-[100px]"
value={httpPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setHttpPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
{platform !== 'win32' && (
<SettingItem title="Redir端口" divider>
<div className="flex">
{redirPortInput !== redirPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'redir-port': redirPortInput })
}}
>
</Button>
)}
<Input
size="sm"
type="number"
className="w-[100px]"
value={redirPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setRedirPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
)}
{platform === 'linux' && (
<SettingItem title="TProxy端口" divider>
<div className="flex">
{tproxyPortInput !== tproxyPort && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'tproxy-port': tproxyPortInput })
}}
>
</Button>
)}
<Input
size="sm"
type="number"
className="w-[100px]"
value={tproxyPortInput.toString()}
max={65535}
min={0}
onValueChange={(v) => {
setTproxyPortInput(parseInt(v))
}}
/>
</div>
</SettingItem>
)}
<SettingItem title="外部控制" divider>
<div className="flex">
{externalControllerInput !== externalController && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ 'external-controller': externalControllerInput })
}}
>
</Button>
)}
<Input
size="sm"
value={externalControllerInput}
onValueChange={(v) => {
setExternalControllerInput(v)
}}
/>
</div>
</SettingItem>
<SettingItem title="外部控制访问密钥" divider>
<div className="flex">
{secretInput !== secret && (
<Button
size="sm"
color="primary"
className="mr-2"
onPress={() => {
onChangeNeedRestart({ secret: secretInput })
}}
>
</Button>
)}
<Input
size="sm"
type="password"
value={secretInput}
onValueChange={(v) => {
setSecretInput(v)
}}
/>
</div>
</SettingItem>
<SettingItem title="IPv6" divider>
<Switch
size="sm"
value={externalControllerInput}
isSelected={ipv6}
onValueChange={(v) => {
setExternalControllerInput(v)
onChangeNeedRestart({ ipv6: v })
}}
/>
</div>
</SettingItem>
<SettingItem title="外部控制访问密钥" divider>
<div className="flex">
{secretInput !== secret && (
</SettingItem>
<SettingItem
title="允许局域网连接"
actions={
<Button
size="sm"
color="primary"
className="mr-2"
isIconOnly
variant="light"
className="ml-2"
onPress={() => {
onChangeNeedRestart({ secret: secretInput })
setLanOpen(true)
}}
>
<FaNetworkWired className="text-lg" />
</Button>
)}
<Input
}
divider
>
<Switch
size="sm"
type="password"
value={secretInput}
isSelected={allowLan}
onValueChange={(v) => {
setSecretInput(v)
onChangeNeedRestart({ 'allow-lan': v })
}}
/>
</div>
</SettingItem>
<SettingItem title="IPv6" divider>
<Switch
size="sm"
isSelected={ipv6}
onValueChange={(v) => {
onChangeNeedRestart({ ipv6: v })
}}
/>
</SettingItem>
<SettingItem title="允许局域网连接" divider>
<Switch
size="sm"
isSelected={allowLan}
onValueChange={(v) => {
onChangeNeedRestart({ 'allow-lan': v })
}}
/>
</SettingItem>
<SettingItem title="使用RTT延迟测试" divider>
<Switch
size="sm"
isSelected={unifiedDelay}
onValueChange={(v) => {
onChangeNeedRestart({ 'unified-delay': v })
}}
/>
</SettingItem>
<SettingItem title="TCP并发" divider>
<Switch
size="sm"
isSelected={tcpConcurrent}
onValueChange={(v) => {
onChangeNeedRestart({ 'tcp-concurrent': v })
}}
/>
</SettingItem>
<SettingItem title="存储选择节点" divider>
<Switch
size="sm"
isSelected={storeSelected}
onValueChange={(v) => {
onChangeNeedRestart({ profile: { 'store-selected': v } })
}}
/>
</SettingItem>
<SettingItem title="存储FakeIP" divider>
<Switch
size="sm"
isSelected={storeFakeIp}
onValueChange={(v) => {
onChangeNeedRestart({ profile: { 'store-fake-ip': v } })
}}
/>
</SettingItem>
<SettingItem title="日志等级" divider>
<Select
className="w-[100px]"
size="sm"
selectedKeys={new Set([logLevel])}
onSelectionChange={(v) => {
onChangeNeedRestart({ 'log-level': v.currentKey as LogLevel })
}}
>
<SelectItem key="silent"></SelectItem>
<SelectItem key="error"></SelectItem>
<SelectItem key="warning"></SelectItem>
<SelectItem key="info"></SelectItem>
<SelectItem key="debug"></SelectItem>
</Select>
</SettingItem>
<SettingItem title="查找进程">
<Select
className="w-[100px]"
size="sm"
selectedKeys={new Set([findProcessMode])}
onSelectionChange={(v) => {
onChangeNeedRestart({ 'find-process-mode': v.currentKey as FindProcessMode })
}}
>
<SelectItem key="strict"></SelectItem>
<SelectItem key="off"></SelectItem>
<SelectItem key="always"></SelectItem>
</Select>
</SettingItem>
</SettingCard>
</BasePage>
</SettingItem>
<SettingItem title="使用RTT延迟测试" divider>
<Switch
size="sm"
isSelected={unifiedDelay}
onValueChange={(v) => {
onChangeNeedRestart({ 'unified-delay': v })
}}
/>
</SettingItem>
<SettingItem title="TCP并发" divider>
<Switch
size="sm"
isSelected={tcpConcurrent}
onValueChange={(v) => {
onChangeNeedRestart({ 'tcp-concurrent': v })
}}
/>
</SettingItem>
<SettingItem title="存储选择节点" divider>
<Switch
size="sm"
isSelected={storeSelected}
onValueChange={(v) => {
onChangeNeedRestart({ profile: { 'store-selected': v } })
}}
/>
</SettingItem>
<SettingItem title="存储FakeIP" divider>
<Switch
size="sm"
isSelected={storeFakeIp}
onValueChange={(v) => {
onChangeNeedRestart({ profile: { 'store-fake-ip': v } })
}}
/>
</SettingItem>
<SettingItem title="日志等级" divider>
<Select
className="w-[100px]"
size="sm"
selectedKeys={new Set([logLevel])}
onSelectionChange={(v) => {
onChangeNeedRestart({ 'log-level': v.currentKey as LogLevel })
}}
>
<SelectItem key="silent"></SelectItem>
<SelectItem key="error"></SelectItem>
<SelectItem key="warning"></SelectItem>
<SelectItem key="info"></SelectItem>
<SelectItem key="debug"></SelectItem>
</Select>
</SettingItem>
<SettingItem title="查找进程">
<Select
className="w-[100px]"
size="sm"
selectedKeys={new Set([findProcessMode])}
onSelectionChange={(v) => {
onChangeNeedRestart({ 'find-process-mode': v.currentKey as FindProcessMode })
}}
>
<SelectItem key="strict"></SelectItem>
<SelectItem key="off"></SelectItem>
<SelectItem key="always"></SelectItem>
</Select>
</SettingItem>
</SettingCard>
</BasePage>
</>
)
}

View File

@ -251,6 +251,10 @@ export async function setupFirewall(): Promise<void> {
return ipcErrorWrapper(await window.electron.ipcRenderer.invoke('setupFirewall'))
}
export async function getInterfaces(): Promise<Record<string, NetworkInterfaceInfo[]>> {
return ipcErrorWrapper(await window.electron.ipcRenderer.invoke('getInterfaces'))
}
export async function setPortable(portable: boolean): Promise<void> {
return ipcErrorWrapper(await window.electron.ipcRenderer.invoke('setPortable', portable))
}

View File

@ -36,6 +36,7 @@ type MihomoProxyType =
type TunStack = 'gvisor' | 'mixed' | 'system'
type FindProcessMode = 'off' | 'strict' | 'always'
type DnsMode = 'normal' | 'fake-ip' | 'redir-host'
type NetworkInterfaceInfo = os.NetworkInterfaceInfo
interface IAppVersion {
version: string