Skip to content
An alternative side effect model for Redux apps
JavaScript TypeScript
Branch: master
Clone or download

Latest commit

Lao-Ax Update API reference with correct links to Redux docs (#2027)
Fix links for "dispatch" to a proper page:, the previous one leaded to 404.
Latest commit d4585ee Feb 27, 2020


Type Name Latest commit message Commit time
Failed to load latest commit information.
.changeset Use changesets (#1954) Oct 20, 2019
.github Add funding information Nov 6, 2019
build fix for dark theme Apr 20, 2017
docs Update API reference with correct links to Redux docs (#2027) Feb 27, 2020
examples v1.1.3 Nov 5, 2019
logo Add Square Logo (#1957) Nov 1, 2019
packages Fix funding.url in redux-saga package.json (#2010) Feb 4, 2020
.bookignore Added ignore list for gitbook cli Feb 3, 2016
.browserslistrc Migrated to Babel7 and Webpack3 Nov 5, 2017
.eslintignore adjust eslint ignore options Mar 15, 2018
.eslintrc.json Fix __tests__ dir linting Aug 16, 2018
.gitignore Use yarn and its workspaces Feb 28, 2019
.huskyrc Upgrade husky, remove prepush hook (#1626) Oct 7, 2018
.lintstagedrc Tweaked prettier config Aug 16, 2018
.npmrc Add .npmrc Aug 24, 2018
.prettierignore Fix prettierignore Aug 18, 2018
.prettierrc Use trailing comma for all files Aug 18, 2018 Update docs for v1 (#1700) Dec 17, 2018
CNAME Create CNAME Apr 19, 2017 Use yarn and its workspaces Feb 28, 2019
LICENSE Initial commit Nov 29, 2015 Redux-Saga not just Redux Mar 5, 2017 Added license section to the README (#1959) Oct 24, 2019 Translated README file into spanish Dec 7, 2017 updated Feb 13, 2019 Changed build status badges urls Jul 14, 2017 Update (#1337) Feb 22, 2018 Changed build status badges urls Jul 14, 2017 Update (#1690) Nov 27, 2018
book.json Book: specify language Apr 7, 2019
lerna.json v1.1.3 Nov 5, 2019
package.json Use yarn to run prerelease script Oct 23, 2019
yarn.lock Update @changesets/cli Oct 20, 2019

Redux Logo Landscape


npm version CDNJS npm Build Status Join the chat at OpenCollective OpenCollective

redux-saga is a library that aims to make application side effects (i.e. asynchronous things like data fetching and impure things like accessing the browser cache) easier to manage, more efficient to execute, easy to test, and better at handling failures.

The mental model is that a saga is like a separate thread in your application that's solely responsible for side effects. redux-saga is a redux middleware, which means this thread can be started, paused and cancelled from the main application with normal redux actions, it has access to the full redux application state and it can dispatch redux actions as well.

It uses an ES6 feature called Generators to make those asynchronous flows easy to read, write and test. (if you're not familiar with them here are some introductory links) By doing so, these asynchronous flows look like your standard synchronous JavaScript code. (kind of like async/await, but generators have a few more awesome features we need)

You might've used redux-thunk before to handle your data fetching. Contrary to redux thunk, you don't end up in callback hell, you can test your asynchronous flows easily and your actions stay pure.

Getting started


$ npm install --save redux-saga


$ yarn add redux-saga

Alternatively, you may use the provided UMD builds directly in the <script> tag of an HTML page. See this section.

Usage Example

Suppose we have a UI to fetch some user data from a remote server when a button is clicked. (For brevity, we'll just show the action triggering code.)

class UserComponent extends React.Component {
  onSomeButtonClicked() {
    const { userId, dispatch } = this.props
    dispatch({type: 'USER_FETCH_REQUESTED', payload: {userId}})

The Component dispatches a plain Object action to the Store. We'll create a Saga that watches for all USER_FETCH_REQUESTED actions and triggers an API call to fetch the user data.


import { call, put, takeEvery, takeLatest } from 'redux-saga/effects'
import Api from '...'

// worker Saga: will be fired on USER_FETCH_REQUESTED actions
function* fetchUser(action) {
   try {
      const user = yield call(Api.fetchUser, action.payload.userId);
      yield put({type: "USER_FETCH_SUCCEEDED", user: user});
   } catch (e) {
      yield put({type: "USER_FETCH_FAILED", message: e.message});

  Starts fetchUser on each dispatched `USER_FETCH_REQUESTED` action.
  Allows concurrent fetches of user.
function* mySaga() {
  yield takeEvery("USER_FETCH_REQUESTED", fetchUser);

  Alternatively you may use takeLatest.

  Does not allow concurrent fetches of user. If "USER_FETCH_REQUESTED" gets
  dispatched while a fetch is already pending, that pending fetch is cancelled
  and only the latest one will be run.
function* mySaga() {
  yield takeLatest("USER_FETCH_REQUESTED", fetchUser);

export default mySaga;

To run our Saga, we'll have to connect it to the Redux Store using the redux-saga middleware.


import { createStore, applyMiddleware } from 'redux'
import createSagaMiddleware from 'redux-saga'

import reducer from './reducers'
import mySaga from './sagas'

// create the saga middleware
const sagaMiddleware = createSagaMiddleware()
// mount it on the Store
const store = createStore(

// then run the saga

// render the application



Using umd build in the browser

There is also a umd build of redux-saga available in the dist/ folder. When using the umd build redux-saga is available as ReduxSaga in the window object. This enables you to create Saga middleware without using ES6 import syntax like this:

var sagaMiddleware = ReduxSaga.default()

The umd version is useful if you don't use Webpack or Browserify. You can access it directly from unpkg.

The following builds are available:

Important! If the browser you are targeting doesn't support ES2015 generators, you must transpile them (i.e. with babel plugin) and provide a valid runtime, such as the one here. The runtime must be imported before redux-saga:

import 'regenerator-runtime/runtime'
// then
import sagaMiddleware from 'redux-saga'

Building examples from sources

$ git clone
$ cd redux-saga
$ yarn
$ npm test

Below are the examples ported (so far) from the Redux repos.

Counter examples

There are three counter examples.


Demo using vanilla JavaScript and UMD builds. All source is inlined in index.html.

To launch the example, open index.html in your browser.

Important: your browser must support Generators. Latest versions of Chrome/Firefox/Edge are suitable.


Demo using webpack and high-level API takeEvery.

$ npm run counter

# test sample for the generator
$ npm run test-counter


Demo using low-level API to demonstrate task cancellation.

$ npm run cancellable-counter

Shopping Cart example

$ npm run shop

# test sample for the generator
$ npm run test-shop

async example

$ npm run async

# test sample for the generators
$ npm run test-async

real-world example (with webpack hot reloading)

$ npm run real-world

# sorry, no tests yet


Redux-Saga with TypeScript requires DOM.Iterable or ES2015.Iterable. If your target is ES6, you are likely already set, however, for ES5, you will need to add it yourself. Check your tsconfig.json file, and the official compiler options documentation.


You can find the official Redux-Saga logo with different flavors in the logo directory.

Redux Saga chooses generators over async/await

A few issues have been raised asking whether Redux saga plans to use async/await syntax instead of generators.

We will continue to use generators. The primary mechanism of async/await is Promises and it is very difficult to retain the scheduling simplicity and semantics of existing Saga concepts using Promises. async/await simply don't allow for certain things - like i.e. cancellation. With generators we have full power over how & when effects are executed.


Support us with a monthly donation and help us continue our activities. [Become a backer]


Become a sponsor and get your logo on our README on Github with a link to your site. [Become a sponsor]


Copyright (c) 2015 Yassine Elouafi.

Licensed under The MIT License (MIT).

You can’t perform that action at this time.