最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Cancel File Upload Post request using Axios - Stack Overflow

programmeradmin2浏览0评论

I am trying to implement a cancel function to cancel a file upload of a video while it's uploading. This is what my front-end looks like while the video is uploading

Here is my code for the post request using axios:

export function processPost (body, callback, errorUpdate) {
// config for post
var config = {
  timeout: 300000,
  headers: { 'content-type': 'multipart/form-data' }

}

// url for server endpoint
var url = 'http://localhost:5000/uploader'

// actual post
axios
  .post(url, body, config)
  .then(callback)
  .catch(function (error) {
    // Non 200 Response
    if (error.response) {
      errorUpdate('non 200 response from server')
      // Request made but no response received
    } else if (error.request) {
      errorUpdate('no response from server')
      // something else happened
    } else {
      errorUpdate('Something Else Happened')
    }
    throw error
  })
}

And here's the code for the front-end:

export default class Wele extends React.Component {
  constructor (props) {
    super(props)
    this.state = {
      state: 'PENDING',
      selectedFile: null
    }
    this.handleClick = this.handleClick.bind(this)
    this.post = this.post.bind(this)
  }

  handleClick () {
    if (this.state.state === 'PENDING') {
      this.refs.fileUploader.click()
    } else {
      cancelPost()
      this.setState({ state: 'PENDING' })
    }
  }

  handleChange (selectorFiles) {
    if (selectorFiles.length === 0) return
    console.log(selectorFiles)
    this.post(selectorFiles[0])
  }

  post (file) {
    // set screen to loading
    this.setState({
      state: 'PROCESSING',
      selectedFile: file
    })

    // set up body
    const data = new FormData()
    data.append('file', file)

    // pass in header, body, then callback
    processPost(data,
      resp => {
        console.log(resp.data)
        this.props.successfulPost(file, URL.createObjectURL(file), resp.data.boardTranscription, resp.data.audioTranscription)
      },
      error => {
        // if there is an error, log it and reset state
        console.log(error)
        this.setState({ state: 'PENDING' })
      })
  }

  render () {
    var jumboClass = classNames({
      'jumbotron col-8 offset-2 light-pink text-center': true,
      'jumboProcessing': this.state.state === 'PROCESSING'
    })

    var input
    var loading
    if (this.state.state === 'PROCESSING') {
      input = null
      loading = (
        <div>
          <div className="spinner-border" role="status">
            <span className="sr-only">Loading...</span>
          </div>
          <p>processing {this.state.selectedFile.name} ...</p>
          <button className={'btn btn-outline-light'} onClick={this.handleChange('')}>Cancel</button>
        </div>
      )
    } else {
      input = (
        <React.Fragment>
          <br></br>
          <button
            className={'btn-light btn-lg'}
            onClick={this.handleClick}
          >Upload Video</button>
          <input id="file" ref="fileUploader" type="file" accept='video/*'
            style={{ display: 'none' }}
            onChange={ (e) => this.handleChange(e.target.files) } />
        </React.Fragment>
      )
      loading = null
    }
    return (
      <div className={'row vertical-center'}>
        <div className={jumboClass}>
          <h2 className={'h2 font-size:10vw'}>Wele to AutoNote!</h2>
          <h6 className={'h6 font-size:5vw'}>Upload a video to generate a linked transcription.</h6>
          {input}
          <br></br>
          {loading}
        </div>
      </div>)
  }
}

Wele.propsType = {
  successfulPost: PropTypes.function
}

How would i go about implementing the cancellation functionality within Axios so that when the user clicks on the "cancel" button, the post request will be cancelled? Any help is much appreciated. Thank you!

I am trying to implement a cancel function to cancel a file upload of a video while it's uploading. This is what my front-end looks like while the video is uploading

Here is my code for the post request using axios:

export function processPost (body, callback, errorUpdate) {
// config for post
var config = {
  timeout: 300000,
  headers: { 'content-type': 'multipart/form-data' }

}

// url for server endpoint
var url = 'http://localhost:5000/uploader'

// actual post
axios
  .post(url, body, config)
  .then(callback)
  .catch(function (error) {
    // Non 200 Response
    if (error.response) {
      errorUpdate('non 200 response from server')
      // Request made but no response received
    } else if (error.request) {
      errorUpdate('no response from server')
      // something else happened
    } else {
      errorUpdate('Something Else Happened')
    }
    throw error
  })
}

And here's the code for the front-end:

export default class Wele extends React.Component {
  constructor (props) {
    super(props)
    this.state = {
      state: 'PENDING',
      selectedFile: null
    }
    this.handleClick = this.handleClick.bind(this)
    this.post = this.post.bind(this)
  }

  handleClick () {
    if (this.state.state === 'PENDING') {
      this.refs.fileUploader.click()
    } else {
      cancelPost()
      this.setState({ state: 'PENDING' })
    }
  }

  handleChange (selectorFiles) {
    if (selectorFiles.length === 0) return
    console.log(selectorFiles)
    this.post(selectorFiles[0])
  }

  post (file) {
    // set screen to loading
    this.setState({
      state: 'PROCESSING',
      selectedFile: file
    })

    // set up body
    const data = new FormData()
    data.append('file', file)

    // pass in header, body, then callback
    processPost(data,
      resp => {
        console.log(resp.data)
        this.props.successfulPost(file, URL.createObjectURL(file), resp.data.boardTranscription, resp.data.audioTranscription)
      },
      error => {
        // if there is an error, log it and reset state
        console.log(error)
        this.setState({ state: 'PENDING' })
      })
  }

  render () {
    var jumboClass = classNames({
      'jumbotron col-8 offset-2 light-pink text-center': true,
      'jumboProcessing': this.state.state === 'PROCESSING'
    })

    var input
    var loading
    if (this.state.state === 'PROCESSING') {
      input = null
      loading = (
        <div>
          <div className="spinner-border" role="status">
            <span className="sr-only">Loading...</span>
          </div>
          <p>processing {this.state.selectedFile.name} ...</p>
          <button className={'btn btn-outline-light'} onClick={this.handleChange('')}>Cancel</button>
        </div>
      )
    } else {
      input = (
        <React.Fragment>
          <br></br>
          <button
            className={'btn-light btn-lg'}
            onClick={this.handleClick}
          >Upload Video</button>
          <input id="file" ref="fileUploader" type="file" accept='video/*'
            style={{ display: 'none' }}
            onChange={ (e) => this.handleChange(e.target.files) } />
        </React.Fragment>
      )
      loading = null
    }
    return (
      <div className={'row vertical-center'}>
        <div className={jumboClass}>
          <h2 className={'h2 font-size:10vw'}>Wele to AutoNote!</h2>
          <h6 className={'h6 font-size:5vw'}>Upload a video to generate a linked transcription.</h6>
          {input}
          <br></br>
          {loading}
        </div>
      </div>)
  }
}

Wele.propsType = {
  successfulPost: PropTypes.function
}

How would i go about implementing the cancellation functionality within Axios so that when the user clicks on the "cancel" button, the post request will be cancelled? Any help is much appreciated. Thank you!

Share edited Mar 7, 2019 at 20:20 davidism 127k31 gold badges415 silver badges347 bronze badges asked Mar 7, 2019 at 19:59 KingKing 1071 gold badge2 silver badges9 bronze badges
Add a ment  | 

3 Answers 3

Reset to default 7

Starting with axios version 0.22.0 you should be using so called AbortController to achieve the required behavior, since CancelToken was deprecated.

const controller = new AbortController();

axios.get('/foo/bar', {
   signal: controller.signal
});

// cancel the request
controller.abort();

You need to make use of CancelToken.

Basically, you need to pass a canceltoken in the config object while using axios.post and when you want to cancel, you can call cancel function.

Example:

let cancelTokenSource = axios.CancelToken.source();

then, in the request config object, pass the token as cancelToken say :

config = { ...otherConfigs, cancelToken : cancelTokenSource.token }

So now, we can use

 cancelTokenSource.cancel('Upload cancelled');

to cancel wherever you want say, upon clicking some cancel button.

Update: CancelToken was deprecated. According to axios cancellation docs starting from v0.22.0 Axios supports AbortController to cancel requests in API way:

const controller = new AbortController();

// Example: axios.post(url, data, config)
axios.post('/foo/bar', {}, {
   signal: controller.signal
});

// cancel the request
controller.abort()

Notice: The following codes are deprecated and work for versions of Axios older than v0.22.0

You will first have to create a cancel token as shown below and add it before your axios post request

const CancelToken = axios.CancelToken;
const source = CancelToken.source();

then add cancelToken: source.token to your axios config

var config = {
    /* ... */
    cancelToken: source.token,
}

now you have to store source in a state, so put this after your axios post

this.setState({source});

finally create a handleCancel() method and call it when user clicked on the cancel button

handleCancel() {
    this.state.source.cancel()
}
发布评论

评论列表(0)

  1. 暂无评论