You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
292 lines
10 KiB
TypeScript
292 lines
10 KiB
TypeScript
import React from 'react'
|
|
import Button from 'react-bootstrap/Button'
|
|
import FormCheck from 'react-bootstrap/FormCheck'
|
|
import fetchToCurl from 'fetch-to-curl'
|
|
import copy from 'copy-to-clipboard'
|
|
import JSONPretty from 'react-json-pretty'
|
|
import { Flow, IResponse } from '../message'
|
|
import { isTextBody } from '../utils'
|
|
|
|
import EditFlow from './EditFlow'
|
|
|
|
interface Iprops {
|
|
flow: Flow | null
|
|
onClose: () => void
|
|
onReRenderFlows: () => void
|
|
onMessage: (msg: ArrayBufferLike) => void
|
|
}
|
|
|
|
interface IState {
|
|
flowTab: 'Headers' | 'Preview' | 'Response' | 'Hexview'
|
|
copied: boolean
|
|
requestBodyViewTab: 'Raw' | 'Preview'
|
|
responseBodyLineBreak: boolean
|
|
}
|
|
|
|
class ViewFlow extends React.Component<Iprops, IState> {
|
|
constructor(props: Iprops) {
|
|
super(props)
|
|
|
|
this.state = {
|
|
flowTab: 'Headers',
|
|
copied: false,
|
|
requestBodyViewTab: 'Raw',
|
|
responseBodyLineBreak: false,
|
|
}
|
|
}
|
|
|
|
preview() {
|
|
const { flow } = this.props
|
|
if (!flow) return null
|
|
const response = flow.response
|
|
if (!response) return null
|
|
|
|
if (!(response.body && response.body.byteLength)) {
|
|
return <div style={{ color: 'gray' }}>No response</div>
|
|
}
|
|
|
|
const pv = flow.previewResponseBody()
|
|
if (!pv) return <div style={{ color: 'gray' }}>Not support preview</div>
|
|
|
|
if (pv.type === 'image') {
|
|
return <img src={`data:image/png;base64,${pv.data}`} />
|
|
}
|
|
else if (pv.type === 'json') {
|
|
return <div><JSONPretty data={pv.data} keyStyle={'color: rgb(130,40,144);'} stringStyle={'color: rgb(153,68,60);'} valueStyle={'color: rgb(25,1,199);'} booleanStyle={'color: rgb(94,105,192);'} /></div>
|
|
}
|
|
|
|
return <div style={{ color: 'gray' }}>Not support preview</div>
|
|
}
|
|
|
|
requestBodyPreview() {
|
|
const { flow } = this.props
|
|
if (!flow) return null
|
|
|
|
const pv = flow.previewRequestBody()
|
|
if (!pv) return <div style={{ color: 'gray' }}>Not support preview</div>
|
|
|
|
if (pv.type === 'json') {
|
|
return <div><JSONPretty data={pv.data} keyStyle={'color: rgb(130,40,144);'} stringStyle={'color: rgb(153,68,60);'} valueStyle={'color: rgb(25,1,199);'} booleanStyle={'color: rgb(94,105,192);'} /></div>
|
|
}
|
|
else if (pv.type === 'binary') {
|
|
return <div><pre>{pv.data}</pre></div>
|
|
}
|
|
|
|
return <div style={{ color: 'gray' }}>Not support preview</div>
|
|
}
|
|
|
|
hexview() {
|
|
const { flow } = this.props
|
|
if (!flow) return null
|
|
const response = flow.response
|
|
if (!response) return null
|
|
|
|
if (!(response.body && response.body.byteLength)) {
|
|
return <div style={{ color: 'gray' }}>No response</div>
|
|
}
|
|
|
|
return <pre>{flow.hexviewResponseBody()}</pre>
|
|
}
|
|
|
|
render() {
|
|
if (!this.props.flow) return null
|
|
|
|
const flow = this.props.flow
|
|
const flowTab = this.state.flowTab
|
|
|
|
const request = flow.request
|
|
const response: IResponse = (flow.response || {}) as any
|
|
|
|
// Query String Parameters
|
|
const searchItems: Array<{ key: string; value: string }> = []
|
|
if (flow.url && flow.url.search) {
|
|
flow.url.searchParams.forEach((value, key) => {
|
|
searchItems.push({ key, value })
|
|
})
|
|
}
|
|
|
|
return (
|
|
<div className="flow-detail">
|
|
<div className="header-tabs">
|
|
<span onClick={() => { this.props.onClose() }}>x</span>
|
|
<span className={flowTab === 'Headers' ? 'selected' : undefined} onClick={() => { this.setState({ flowTab: 'Headers' }) }}>Headers</span>
|
|
<span className={flowTab === 'Preview' ? 'selected' : undefined} onClick={() => { this.setState({ flowTab: 'Preview' }) }}>Preview</span>
|
|
<span className={flowTab === 'Response' ? 'selected' : undefined} onClick={() => { this.setState({ flowTab: 'Response' }) }}>Response</span>
|
|
<span className={flowTab === 'Hexview' ? 'selected' : undefined} onClick={() => { this.setState({ flowTab: 'Hexview' }) }}>Hexview</span>
|
|
|
|
<EditFlow
|
|
flow={flow}
|
|
onChangeRequest={request => {
|
|
flow.request.method = request.method
|
|
flow.request.url = request.url
|
|
flow.request.header = request.header
|
|
if (isTextBody(flow.request)) flow.request.body = request.body
|
|
this.props.onReRenderFlows()
|
|
}}
|
|
onChangeResponse={response => {
|
|
if (!flow.response) flow.response = {} as IResponse
|
|
|
|
flow.response.statusCode = response.statusCode
|
|
flow.response.header = response.header
|
|
if (isTextBody(flow.response)) flow.response.body = response.body
|
|
this.props.onReRenderFlows()
|
|
}}
|
|
onMessage={msg => {
|
|
this.props.onMessage(msg)
|
|
flow.waitIntercept = false
|
|
this.props.onReRenderFlows()
|
|
}}
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div style={{ padding: '20px' }}>
|
|
{
|
|
!(flowTab === 'Headers') ? null :
|
|
<div>
|
|
<p><Button size="sm" variant={this.state.copied ? 'success' : 'primary'} disabled={this.state.copied} onClick={() => {
|
|
const curl = fetchToCurl({
|
|
url: flow.request.url,
|
|
method: flow.request.method,
|
|
headers: Object.keys(flow.request.header).reduce((obj: any, key: string) => {
|
|
obj[key] = flow.request.header[key][0]
|
|
return obj
|
|
}, {}),
|
|
body: flow.requestBody(),
|
|
})
|
|
copy(curl)
|
|
|
|
this.setState({ copied: true }, () => {
|
|
setTimeout(() => {
|
|
this.setState({ copied: false })
|
|
}, 1000)
|
|
})
|
|
|
|
}}>{this.state.copied ? 'Copied' : 'Copy as cURL'}</Button></p>
|
|
|
|
<div className="header-block">
|
|
<p>General</p>
|
|
<div className="header-block-content">
|
|
<p>Request URL: {request.url}</p>
|
|
<p>Request Method: {request.method}</p>
|
|
<p>Status Code: {`${response.statusCode || '(pending)'}`}</p>
|
|
</div>
|
|
</div>
|
|
|
|
{
|
|
!(response.header) ? null :
|
|
<div className="header-block">
|
|
<p>Response Headers</p>
|
|
<div className="header-block-content">
|
|
{
|
|
Object.keys(response.header).map(key => {
|
|
return (
|
|
<p key={key}>{key}: {response.header[key].join(' ')}</p>
|
|
)
|
|
})
|
|
}
|
|
</div>
|
|
</div>
|
|
}
|
|
|
|
<div className="header-block">
|
|
<p>Request Headers</p>
|
|
<div className="header-block-content">
|
|
{
|
|
!(request.header) ? null :
|
|
Object.keys(request.header).map(key => {
|
|
return (
|
|
<p key={key}>{key}: {request.header[key].join(' ')}</p>
|
|
)
|
|
})
|
|
}
|
|
</div>
|
|
</div>
|
|
|
|
{
|
|
!(searchItems.length) ? null :
|
|
<div className="header-block">
|
|
<p>Query String Parameters</p>
|
|
<div className="header-block-content">
|
|
{
|
|
searchItems.map(({ key, value }) => {
|
|
return (
|
|
<p key={key}>{key}: {value}</p>
|
|
)
|
|
})
|
|
}
|
|
</div>
|
|
</div>
|
|
}
|
|
|
|
{
|
|
!(request.body && request.body.byteLength) ? null :
|
|
<div className="header-block">
|
|
<p>Request Body</p>
|
|
<div className="header-block-content">
|
|
<div>
|
|
<div className="request-body-detail" style={{ marginBottom: '15px' }}>
|
|
<span className={this.state.requestBodyViewTab === 'Raw' ? 'selected' : undefined} onClick={() => { this.setState({ requestBodyViewTab: 'Raw' }) }}>Raw</span>
|
|
<span className={this.state.requestBodyViewTab === 'Preview' ? 'selected' : undefined} onClick={() => { this.setState({ requestBodyViewTab: 'Preview' }) }}>Preview</span>
|
|
</div>
|
|
|
|
{
|
|
!(this.state.requestBodyViewTab === 'Raw') ? null :
|
|
<div>
|
|
{
|
|
!(flow.isTextRequest()) ? <span style={{ color: 'gray' }}>Not text Request</span> : flow.requestBody()
|
|
}
|
|
</div>
|
|
}
|
|
|
|
{
|
|
!(this.state.requestBodyViewTab === 'Preview') ? null :
|
|
<div>{this.requestBodyPreview()}</div>
|
|
}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
}
|
|
|
|
</div>
|
|
}
|
|
|
|
{
|
|
!(flowTab === 'Response') ? null :
|
|
!(response.body && response.body.byteLength) ? <div style={{ color: 'gray' }}>No response</div> :
|
|
!(flow.isTextResponse()) ? <div style={{ color: 'gray' }}>Not text response</div> :
|
|
<div>
|
|
<div style={{ marginBottom: '20px' }}>
|
|
<FormCheck
|
|
inline
|
|
type="checkbox"
|
|
checked={this.state.responseBodyLineBreak}
|
|
onChange={e => {
|
|
this.setState({ responseBodyLineBreak: e.target.checked })
|
|
}}
|
|
label="自动换行"></FormCheck>
|
|
</div>
|
|
<div style={{ whiteSpace: this.state.responseBodyLineBreak ? 'pre-wrap' : 'pre' }}>
|
|
{flow.responseBody()}
|
|
</div>
|
|
</div>
|
|
}
|
|
|
|
{
|
|
!(flowTab === 'Preview') ? null :
|
|
<div>{this.preview()}</div>
|
|
}
|
|
|
|
{
|
|
!(flowTab === 'Hexview') ? null :
|
|
<div>{this.hexview()}</div>
|
|
}
|
|
</div>
|
|
|
|
</div>
|
|
)
|
|
}
|
|
}
|
|
|
|
export default ViewFlow
|