0.92.25014 • Published 4 months ago

reactronic v0.92.25014

Weekly downloads
2,453
License
Apache-2.0
Repository
github
Last release
4 months ago

Readiness GitHub License NPM Version Package Size Coverage Lines Demo

Reactronic - Transactional Reactive State Management

Reactronic is an experimental JavaScript library that provides transactional reactive state management in a Web application.

Transactional reactivity means that state changes are being made in an isolated data snapshot and then, once atomically applied, are consistently propagated to corresponding visual components for (re)rendering. All that is done in automatic, seamless, and fine-grained way. Reactronic takes full care of tracking dependencies between visual components (observers) and state (observable objects).

Transactional reactivity is based on four fundamental concepts:

  • Observable Objects - a set of objects that store data of an application (state);
  • Atomic Function - a function that makes changes in observable objects in atomic way ("all or nothing");
  • Reactive Function - a function that is (re-)executed in response to changes made by atomic actions;
  • Cached Function - a function which result is remembered and, if becomes obsolete, causes function to re-execute on-demand.

Demo application built with Reactronic: https://nevod.io/#/playground. Source code of the demo: https://gitlab.com/nezaboodka/nevod.web.public/-/blob/master/README.md.

Quick introduction and detailed description is below.

Quick Introduction

Here is an example of transactional reactive code:

class Demo extends ObservableObject {
  name: string = 'Nezaboodka Software'
  email: string = 'contact@nezaboodka.com'

  @atomic
  saveContact(name: string, email: string): void {
    this.name = name
    this.email = email
  }

  @reactive
  printContact(): void {
    // depends on `name` and `email` and reacts to their changes
    if (this.email.indexOf('@') >= 0)
      throw new Error(`wrong email ${this.email}`)
    console.log(this.name + ' <' + this.email + '>')
  }
}

In the example above, Demo is an observable object, meaning that access to its fields are seamlessly tracked to determine dependent reactive and cached functions. Reactive function printContact reads name and email fields, thus depends on them. It is executed automatically in response to changes of these fields made by the atomic function saveContact.

Here is an example of a cached result that is (re-)computed on-demand:

class Demo extends ObservableObject {
  name: string = 'Nezaboodka Software'
  email: string = 'contact@nezaboodka.com'

  @cachedResult
  get contact(): string {
    return this.name + ' <' + this.email + '>'
  }

  @reactive
  printContact(): void {
    if (this.contact !== '')
      Console.log(this.contact)
  }
}

In the example above, the result of contact getter is computed from source fields name and email. Once computed, the result is cached and is reused until source fields name and email are changed. Once source fields changed, contact result becomes obsolete, thus causing execution of depending reactive function printContact. When function of reactive function printContact runs it reads contact and causes its re-computation.

Observable Objects

Observable objects are aimed to store data of an application. All such objects are transparently hooked to track access to their properties, both on reads and writes.

class MyModel extends ObservableObject {
  url: string = "https://github.com/nezaboodka/reactronic"
  content: string = "transactional reactive state management"
  timestamp: Date = Date.now()
}

In the example above, the class MyModel is based on Reactronic's ObservableObject class and all its properties url, content, and timestamp are hooked.

Atomic Function

Atomic function makes changes in observable objects in atomic (transactional) way, thus provoking execution of dependent reactive and cached functions. Atomic function is instrumented with hooks to provide transparent atomicity (by implicit context switching and isolation).

class MyModel extends ObservableObject {
  // ...
  @atomic
  async load(url: string): Promise<void> {
    this.url = url
    this.content = await fetch(url)
    this.timestamp = Date.now()
  }
}

In the example above, the atomic function load makes changes to url, content and timestamp properties. While atomic function is running, the changes are visible only inside the function itself. The new values become atomically visible outside of the function only upon its completion.

Atomicity is achieved by making changes in an isolated data snapshot that is not visible outside of the running function until it is fully finished and applied. Multiple objects and their properties can be changed with full respect to the all-or-nothing principle. To do so, separate data snapshot is automatically maintained for each atomic function. That is a logical snapshot that does not create a full copy of all the data.

Compensating rollback operations are not needed in case of the atomic function failure, because all the changes made by the atomic function in its logical snapshot are simply discarded. In case the atomic function is successfully applied, affected cached results are marked as obsolete and corresponding caching functions are re-executed in a proper order (but only when all the data changes are fully applied).

Asynchronous operations (promises) are supported out of the box during atomic function execution. Atomic function may consist of a set of asynchronous calls prolonging the function until completion of all of them. An asynchronous call may spawn other asynchronous calls, which prolong atomic atomic execution until the whole chain of asynchronous operations is fully completed.

Reactive & Cached Functions

Reactive function is automatically and immediately called in response to changes in observable objects made by atomic functions. Cached function is called on-demand to renew the result if it was marked as obsolete due to changes made by an atomic functions. Reactive and cached functions are instrumented with hooks to seamlessly subscribe to those observable objects and other cached functions (dependencies), which are used during their execution.

class MyView extends Component<{model: MyModel}> {
  @cachedResult
  render(): React.JSX.Element {
    return (
      <div>
        <h1>{this.props.model.url}</h1>
        <div>{this.props.model.content}</div>
      </div>
    )
  } // render is subscribed to "url" and "content"
}
class Component<P> extends React.Component<P> {
  @cachedResult
  render(): React.JSX.Element {
    throw new Error('render method is undefined')
  }

  @reactive // called in response to changes
  ensureUpToDate(): void {
    if (this.shouldComponentUpdate()) {
      // Ask React to re-render
      Transaction.outside(() => this.setState({}))
    }
  } // EnsureUpToDate is subscribed to render

  shouldComponentUpdate(): boolean {
    const r = ReactiveSystem.getController(this.render)
    return !r.isUpToDate
  }

  componentDidMount(): void {
    // Run to subscribe for the first time
    this.ensureUpToDate()
  }

  componentWillUnmount(): void {
    atomicAction(ReactiveSystem.dispose, this)
  }
}

In the example above, reactive function refresh is transparently subscribed to the cached function render. In turn, cached function render is subscribed to the properties url and content of a corresponding MyModel object. Once url or content values are changed, the cached function render becomes obsolete and causes the reactive function refresh to become obsolete and re-executed. While being executed, the reactive function refresh enqueues re-rendering request to React, which calls cached function render causing it to renew its cached value.

In general case, all reactive and cached functions are automatically and immediately marked as obsolete when changes are made in those observable objects and other cached results that were used during their execution. And once marked, the functions are automatically executed again, either immediately (for reactive functions) or on-demand (for cached functions).

Reactronic takes full care of tracking dependencies between all the observable objects and reactive/cached functions. With Reactronic, you no longer need to create data change events in one set of objects, subscribe to these events in other objects, and manually maintain switching from the previous object version to a new one.

Behavior Options

There are multiple options to configure behavior of transactional reactivity.

Order options defines order of execution for reactive functions:

  • (TBD)

Throttling option defines how often reactive function is executed in case of recurring changes:

  • (ms) - minimal delay in milliseconds between executions;
  • -1 - execute immediately once atomic function is applied (synchronously);
  • 0 - execute immediately via event loop (asynchronously with zero timeout);
  • >= Number.MAX_SAFE_INTEGER - never execute (suspended reaction).

Reentrance option defines how to handle reentrant calls of atomic and reactive functions:

  • preventWithError - fail with error if there is an existing call in progress;
  • waitAndRestart - wait for previous call to finish and then restart current one;
  • cancelPrevious - cancel previous call in favor of recent one;
  • cancelAndWaitPrevious - cancel previous call in favor of recent one (but wait until canceling is completed)
  • runSideBySide - multiple simultaneous calls are allowed.

Indicator is an object that maintains status of running functions, which it is attached to. A single indicator object can be shared between multiple atomic, reactive, and cached functions, thus maintaining consolidated status for all of them (busy, workers, etc).

Notes

Inspired by: MobX, Nezaboodka, Excel.

Key Reactronic principles and differentiators:

  • No compromises on consistency, clarity, and simplicity;
  • Minimalism and zero boilerplating (it's not a framework bloating your code);
  • Asynchrony, patches, undo/redo, conflict resolving are provided out of the box;
  • Seamless integration with transactional reactive object-oriented databases like Nezaboodka;
  • Compact dependency-free implementation consisting of less than 2K lines of code.

Roadmap:

  • Patches and conflict resolution API (partially done)
  • History/undo/redo API and implementation (partially done)
  • Sync API and implementation (not implemented yet)

Installation

NPM: npm install reactronic

API (TypeScript)

// Classes

class TransactionalObject { }
class ObservableObject { }

// Decorators & Operators

function observable(proto, prop) // field only
function unobservable(proto, prop) // field only
function atomic(proto, prop, pd) // method only
function reactive(proto, prop, pd) // method only
function cached(proto, prop, pd) // method only
function options(value: Partial<MemberOptions>): F<any>

function nonReactiveRun<T>(func: F<T>, ...args: any[]): T
function sensitiveRun<T>(sensitivity: Sensitivity, func: F<T>, ...args: any[]): T

// SnapshotOptions, MemberOptions, Kind, Reentrance, Indicator, LoggingOptions, ProfilingOptions

export type SnapshotOptions = {
  readonly hint?: string
  readonly isolation?: Isolation
  readonly journal?: Journal
  readonly logging?: Partial<LoggingOptions>
  readonly token?: any
}

type MemberOptions = {
  readonly kind: Kind
  readonly isolation: Isolation
  readonly order: number
  readonly noSideEffects: boolean
  readonly triggeringArgs: boolean
  readonly throttling: number // milliseconds, -1 is immediately, Number.MAX_SAFE_INTEGER is never
  readonly reentrance: Reentrance
  readonly journal: Journal | undefined
  readonly indicator: Indicator | null
  readonly logging?: Partial<LoggingOptions>
}

enum Kind {
  plain = 0,
  atomic = 1,
  reactive = 2,
  cached = 3
}

enum Reentrance {
  preventWithError = 1, // fail with error if there is an existing call in progress (default)
  waitAndRestart = 0, // wait for existing call to finish and then restart current one
  cancelPrevious = -1, // cancel previous call in favor of recent one
  cancelAndWaitPrevious = -2, // cancel previous call in favor of recent one (but wait until canceling is completed)
  overwritePrevious = -2, // allow previous to complete, but overwrite it with ignoring any conflicts
  runSideBySide = -3 // multiple simultaneous calls are allowed
}

class Indicator {
  readonly isBusy: boolean
  readonly counter: number
  readonly workers: ReadonlySet<Worker>
  readonly busyDuration: number
  abstract whenBusy(): Promise<void>
  abstract whenIdle(): Promise<void>
  static create(hint: string, activationDelay: number, deactivationDelay: number): Indicator
}

type Worker = {
  readonly id: number
  readonly hint: string
  isCanceled: boolean
  isFinished: boolean
  cancel(error?: Error, retryAfter?: Transaction): this
  whenFinished(): Promise<void>
}

type LoggingOptions = {
  readonly off: boolean
  readonly transaction: boolean
  readonly operation: boolean
  readonly step: boolean
  readonly indicator: boolean
  readonly read: boolean
  readonly write: boolean
  readonly change: boolean
  readonly obsolete: boolean
  readonly error: boolean
  readonly warning: boolean
  readonly gc: boolean
}

type ProfilingOptions = {
  repetitiveUsageWarningThreshold: number // default: 10 times
  mainThreadBlockingWarningThreshold: number // default: 16.6 ms
  asyncActionDurationWarningThreshold: number // default: 150 ms
}

// Transaction

type F<T> = (...args: any[]) => T

class Transaction implements Worker {
  static readonly current: Transaction

  readonly id: number
  readonly hint: string

  run<T>(func: F<T>, ...args: any[]): T
  wrap<T>(func: F<T>): F<T>
  apply(): void
  seal(): this // a1.seal().whenFinished().then(fulfill, reject)
  cancel(error?: Error, retryAfter?: Transaction): this
  isCanceled: boolean
  isFinished: boolean
  whenFinished(): Promise<void>
  join<T>(p: Promise<T>): Promise<T>

  static create(options: SnapshotOptions | null): Transaction
  static run<T>(options: SnapshotOptions | null, func: F<T>, ...args: any[]): T
  static off<T>(func: F<T>, ...args: any[]): T

  static isFrameOver(everyN: number, timeLimit: number): boolean
  static requestNextFrame(sleepTime: number): Promise<void>
  static isCanceled: boolean
}

// Operation

abstract class Operation<T> {
  readonly options: Options
  readonly args: ReadonlyArray<any>
  readonly value: T
  readonly error: any
  readonly stamp: number
  readonly isReusable: boolean

  configure(options: Partial<Options>): Options
  markObsolete(): boolean
  pullLastResult(args?: any[]): T | undefined
}

// ReactiveSystem

class ReactiveSystem {
  static why(short: boolean = false): string
  static getMethodCache<T>(method: F<T>): Cache<T>
  static configureCurrentOperation(options: Partial<Options>): Options
  static getRevisionOf(obj: any): number
  static takeSnapshot<T>(obj: T): T
  static dispose(obj: any): void
  static reactivityAutoStartDisabled: boolean
  static readonly isLogging: boolean
  static readonly loggingOptions: LoggingOptions
  static setLoggingMode(isOn: boolean, options?: LoggingOptions)
  static setLoggingHint<T extends object>(obj: T, name: string | undefined): void
  static getLoggingHint<T extends object>(obj: T): string | undefined
  static setProfilingMode(isOn: boolean, options?: Partial<ProfilingOptions>): void
}

Contribution

By contributing, you agree that your contributions will be automatically licensed under the Apache 2.0 license (see LICENSE file).

0.92.25014

4 months ago

0.92.25013

4 months ago

0.92.25012

4 months ago

0.92.25011

4 months ago

0.92.25010

4 months ago

0.91.25002

4 months ago

0.24.500

7 months ago

0.24.501

7 months ago

0.24.502

6 months ago

0.24.400

7 months ago

0.24.401

7 months ago

0.24.316

7 months ago

0.92.25005

4 months ago

0.92.25004

4 months ago

0.92.25003

4 months ago

0.92.25009

4 months ago

0.92.25008

4 months ago

0.92.25007

4 months ago

0.92.25006

4 months ago

0.24.307

8 months ago

0.24.308

8 months ago

0.24.309

8 months ago

0.24.301

10 months ago

0.24.302

9 months ago

0.24.303

9 months ago

0.24.304

9 months ago

0.24.305

9 months ago

0.24.306

9 months ago

0.24.274

1 year ago

0.24.275

12 months ago

0.24.273

1 year ago

0.24.272

1 year ago

0.24.271

1 year ago

0.24.270

1 year ago

0.24.266

1 year ago

0.24.267

1 year ago

0.24.268

1 year ago

0.24.261

1 year ago

0.24.262

1 year ago

0.24.263

1 year ago

0.24.264

1 year ago

0.24.265

1 year ago

0.24.260

1 year ago

0.24.250

1 year ago

0.24.200

1 year ago

0.24.201

1 year ago

0.24.202

1 year ago

0.24.128

1 year ago

0.24.127

1 year ago

0.24.126

1 year ago

0.24.123

1 year ago

0.24.124

1 year ago

0.24.125

1 year ago

0.24.120

1 year ago

0.24.121

1 year ago

0.24.122

1 year ago

0.24.113

1 year ago

0.24.114

1 year ago

0.24.115

1 year ago

0.24.116

1 year ago

0.24.117

1 year ago

0.24.118

1 year ago

0.24.119

1 year ago

0.24.112

2 years ago

0.24.110

2 years ago

0.24.111

2 years ago

0.24.107

2 years ago

0.24.101

2 years ago

0.24.102

2 years ago

0.24.103

2 years ago

0.24.104

2 years ago

0.24.106

2 years ago

0.23.115

2 years ago

0.23.113

2 years ago

0.23.114

2 years ago

0.23.111

2 years ago

0.23.112

2 years ago

0.23.110

2 years ago

0.23.108

2 years ago

0.23.109

2 years ago

0.23.104

2 years ago

0.23.105

2 years ago

0.23.106

2 years ago

0.23.107

2 years ago

0.23.103

2 years ago

0.22.410

3 years ago

0.22.411

3 years ago

0.23.100

2 years ago

0.23.101

2 years ago

0.22.502

3 years ago

0.22.503

3 years ago

0.22.504

3 years ago

0.22.505

3 years ago

0.22.506

3 years ago

0.22.507

3 years ago

0.22.508

3 years ago

0.22.509

3 years ago

0.22.510

3 years ago

0.22.511

2 years ago

0.22.317

3 years ago

0.22.318

3 years ago

0.22.400

3 years ago

0.22.320

3 years ago

0.22.303

3 years ago

0.22.304

3 years ago

0.22.306

3 years ago

0.22.307

3 years ago

0.22.308

3 years ago

0.22.309

3 years ago

0.22.310

3 years ago

0.22.311

3 years ago

0.22.312

3 years ago

0.22.313

3 years ago

0.22.314

3 years ago

0.22.315

3 years ago

0.22.316

3 years ago

0.22.300

3 years ago

0.22.301

3 years ago

0.22.302

3 years ago

0.22.205

3 years ago

0.22.206

3 years ago

0.22.210

3 years ago

0.21.603

3 years ago

0.21.604

3 years ago

0.21.601

3 years ago

0.21.602

3 years ago

0.21.600

3 years ago

0.22.101

3 years ago

0.22.102

3 years ago

0.22.103

3 years ago

0.22.104

3 years ago

0.22.105

3 years ago

0.22.106

3 years ago

0.22.107

3 years ago

0.22.108

3 years ago

0.22.109

3 years ago

0.22.110

3 years ago

0.22.200

3 years ago

0.22.201

3 years ago

0.22.202

3 years ago

0.21.528

4 years ago

0.22.203

3 years ago

0.21.529

3 years ago

0.22.204

3 years ago

0.21.526

4 years ago

0.21.527

4 years ago

0.21.524

4 years ago

0.21.525

4 years ago

0.21.522

4 years ago

0.21.523

4 years ago

0.21.521

4 years ago

0.21.519

4 years ago

0.21.518

4 years ago

0.21.520

4 years ago

0.21.517

4 years ago

0.21.515

4 years ago

0.21.516

4 years ago

0.21.513

4 years ago

0.21.514

4 years ago

0.21.502

4 years ago

0.21.503

4 years ago

0.21.500

4 years ago

0.21.501

4 years ago

0.21.511

4 years ago

0.21.512

4 years ago

0.21.510

4 years ago

0.21.302

4 years ago

0.21.300

4 years ago

0.21.301

4 years ago

0.21.400

4 years ago

0.21.200

4 years ago

0.21.120

4 years ago

0.21.110

4 years ago

0.21.102

4 years ago

0.21.101

4 years ago

0.20.707

4 years ago

0.20.706

5 years ago

0.20.705

5 years ago

0.20.704

5 years ago

0.20.703

5 years ago

0.20.702

5 years ago

0.20.701

5 years ago

0.20.700

5 years ago

0.20.600

5 years ago

0.20.601

5 years ago

0.20.500

5 years ago

0.20.400

5 years ago

0.20.310

5 years ago

0.20.305

5 years ago

0.20.304

5 years ago

0.20.303

5 years ago

0.20.302

5 years ago

0.20.301

5 years ago

0.20.300

5 years ago

0.20.212

5 years ago

0.20.211

5 years ago

0.20.210

5 years ago

0.20.209

5 years ago

0.20.208

5 years ago

0.20.207

5 years ago

0.20.206

5 years ago

0.20.204

5 years ago

0.20.205

5 years ago

0.20.200

5 years ago

0.20.201

5 years ago

0.20.202

5 years ago

0.20.203

5 years ago

0.20.107

5 years ago

0.20.108

5 years ago

0.20.106

5 years ago

0.20.105

5 years ago

0.20.103

5 years ago

0.20.104

5 years ago

0.20.102

5 years ago

0.20.101

5 years ago

0.20.100

5 years ago

0.20.99

5 years ago

0.20.98

5 years ago

0.20.97

5 years ago

0.20.95

5 years ago

0.20.96

5 years ago

0.20.94

5 years ago

0.20.93

5 years ago

0.20.92

5 years ago

0.20.91

5 years ago

0.20.88

5 years ago

0.20.89

5 years ago

0.20.90

5 years ago

0.20.87

5 years ago

0.20.86

5 years ago

0.20.85

5 years ago

0.20.84

5 years ago

0.20.83

5 years ago

0.20.82

5 years ago

0.20.81

5 years ago

0.20.80

5 years ago

0.20.79

5 years ago

0.20.78

5 years ago

0.20.77

5 years ago

0.20.76

5 years ago

0.20.75

5 years ago

0.20.73

5 years ago

0.20.74

5 years ago

0.20.71

5 years ago

0.20.72

5 years ago

0.20.70

5 years ago

0.20.69

5 years ago

0.20.68

5 years ago

0.20.67

5 years ago

0.20.66

5 years ago

0.20.64

5 years ago

0.20.63

5 years ago

0.20.62

5 years ago

0.20.61

5 years ago

0.20.60

5 years ago

0.20.59

5 years ago

0.20.58

5 years ago

0.20.57

5 years ago

0.20.55

5 years ago

0.20.56

5 years ago

0.20.53

5 years ago

0.20.54

5 years ago

0.20.52

5 years ago

0.20.51

5 years ago

0.20.50

5 years ago

0.20.49

5 years ago

0.20.48

5 years ago

0.20.47

5 years ago

0.20.46

5 years ago

0.20.44

5 years ago

0.20.45

5 years ago

0.20.42

5 years ago

0.20.43

5 years ago

0.20.40

5 years ago

0.20.41

5 years ago

0.20.39

5 years ago

0.20.38

5 years ago

0.20.37

5 years ago

0.20.36

5 years ago

0.20.35

5 years ago

0.20.34

5 years ago

0.20.33

5 years ago

0.20.32

5 years ago

0.20.31

5 years ago

0.20.30

5 years ago

0.20.29

5 years ago

0.20.28

5 years ago

0.20.27

5 years ago

0.20.26

5 years ago

0.20.25

5 years ago

0.20.24

5 years ago

0.20.23

5 years ago

0.20.22

5 years ago

0.20.21

5 years ago

0.20.20

5 years ago

0.20.11

5 years ago

0.20.12

5 years ago

0.20.10

5 years ago

0.20.9

5 years ago

0.20.8

5 years ago

0.20.7

5 years ago

0.20.6

5 years ago

0.20.5

5 years ago

0.20.4

5 years ago

0.20.3

5 years ago

0.20.2

5 years ago

0.20.1

5 years ago

0.19.450

5 years ago

0.19.449

6 years ago

0.19.448

6 years ago

0.19.446

6 years ago

0.19.447

6 years ago

0.19.445

6 years ago

0.19.444

6 years ago

0.19.443

6 years ago

0.19.442

6 years ago

0.19.441

6 years ago

0.19.440

6 years ago

0.19.439

6 years ago

0.19.438

6 years ago

0.19.437

6 years ago

0.19.436

6 years ago

0.19.435

6 years ago

0.19.434

6 years ago

0.19.433

6 years ago

0.19.432

6 years ago

0.19.431

6 years ago

0.19.430

6 years ago

0.19.429

6 years ago

0.19.428

6 years ago

0.19.427

6 years ago

0.19.426

6 years ago

0.19.425

6 years ago

0.19.424

6 years ago

0.19.423

6 years ago

0.19.422

6 years ago

0.19.420

6 years ago

0.19.419

6 years ago

0.19.418

6 years ago

0.19.417

6 years ago

0.19.416

6 years ago

0.19.415

6 years ago

0.19.414

6 years ago

0.19.413

6 years ago

0.19.412

6 years ago

0.19.411

6 years ago

0.19.410

6 years ago

0.19.409

6 years ago

0.19.407

6 years ago

0.19.405

6 years ago

0.19.404

6 years ago

0.19.403

6 years ago

0.19.402

6 years ago

0.19.401

6 years ago

0.19.400

6 years ago

0.19.399

6 years ago

0.19.398

6 years ago

0.19.397

6 years ago

0.19.396

6 years ago

0.19.395

6 years ago

0.19.394

6 years ago

0.19.393

6 years ago

0.19.392

6 years ago

0.19.390

6 years ago

0.19.389

6 years ago

0.19.388

6 years ago

0.19.387

6 years ago

0.19.386

6 years ago

0.19.385

6 years ago

0.19.384

6 years ago

0.19.383

6 years ago

0.19.382

6 years ago

0.19.381

6 years ago

0.19.380

6 years ago

0.19.379

6 years ago

0.19.378

6 years ago

0.19.377

6 years ago

0.19.376

6 years ago

0.19.375

6 years ago

0.19.374

6 years ago

0.19.373

6 years ago

0.19.372

6 years ago

0.19.371

6 years ago

0.19.370

6 years ago

0.19.369

6 years ago

0.19.368

6 years ago

0.19.367

6 years ago

0.19.366

6 years ago

0.19.365

6 years ago

0.19.364

6 years ago

0.19.363

6 years ago

0.19.362

6 years ago

0.19.361

6 years ago

0.19.360

6 years ago

0.19.359

6 years ago

0.19.358

6 years ago

0.19.357

6 years ago

0.19.356

6 years ago

0.19.354

6 years ago

0.19.353

6 years ago

0.19.351

6 years ago

0.19.350

6 years ago

0.19.312

6 years ago

0.19.310

6 years ago

0.19.309

6 years ago

0.19.308

6 years ago

0.19.307

6 years ago

0.19.306

6 years ago

0.19.305

6 years ago

0.19.304

6 years ago

0.19.303

6 years ago

0.19.302

6 years ago

0.19.301

6 years ago

0.19.300

6 years ago

0.19.208

6 years ago

0.19.207

6 years ago

0.19.206

6 years ago

0.19.205

6 years ago

0.19.204

6 years ago

0.19.203

6 years ago

0.19.202

6 years ago

0.19.201

6 years ago

0.19.200

6 years ago

0.19.123

6 years ago

0.19.122

6 years ago

0.19.121

6 years ago

0.19.120

6 years ago

0.19.119

6 years ago

0.19.118

6 years ago

0.19.117

6 years ago

0.19.115

6 years ago

0.19.114

6 years ago

0.19.113

6 years ago

0.19.112

6 years ago

0.19.111

6 years ago

0.19.110

6 years ago

0.19.109

6 years ago

0.19.108

6 years ago

0.19.107

6 years ago

0.19.106

6 years ago

0.19.105

6 years ago

0.19.104

6 years ago

0.19.103

6 years ago

0.19.102

6 years ago

0.19.101

6 years ago

0.19.90

6 years ago

0.19.89

6 years ago

0.19.88

6 years ago

0.19.87-a

6 years ago

0.19.87

6 years ago

0.19.86

6 years ago

0.19.85

6 years ago

0.19.84

6 years ago

0.19.83

6 years ago

0.19.82

6 years ago

0.19.81

6 years ago

0.19.80

6 years ago

0.19.79

6 years ago

0.19.78

6 years ago

0.19.77

6 years ago

0.19.76

6 years ago

0.19.75

6 years ago

0.19.74

6 years ago

0.19.73

6 years ago

0.19.72

6 years ago

0.19.71

6 years ago

0.19.70

6 years ago

0.19.69

6 years ago

0.19.68

6 years ago

0.19.67

6 years ago

0.19.66

6 years ago

0.19.65

6 years ago

0.19.64

6 years ago

0.19.63

6 years ago

0.19.62

6 years ago

0.19.61

6 years ago

0.19.60

6 years ago

0.19.59

6 years ago

0.19.58

6 years ago

0.19.57

6 years ago

0.19.56

6 years ago

0.19.55

6 years ago

0.19.54

6 years ago

0.19.53

6 years ago

0.19.52

6 years ago

0.19.51

6 years ago

0.19.50

6 years ago

0.19.49

6 years ago

0.19.48

6 years ago

0.19.47

6 years ago

0.19.46

6 years ago

0.19.45

6 years ago

0.19.44

6 years ago

0.19.43

6 years ago

0.19.42

6 years ago

0.19.41

6 years ago

0.19.40

6 years ago

0.19.39

6 years ago

0.19.38

6 years ago

0.19.37

6 years ago

0.19.36

6 years ago

0.19.35

6 years ago

0.19.34

6 years ago

0.19.33

6 years ago

0.19.32

6 years ago

0.19.31

6 years ago

0.19.30

6 years ago

0.19.29

6 years ago

0.19.28

6 years ago

0.19.27

6 years ago

0.19.26

6 years ago

0.19.25

6 years ago

0.19.24

6 years ago

0.19.23

6 years ago

0.19.22

6 years ago

0.19.21

6 years ago

0.19.20

6 years ago

0.19.19

6 years ago

0.19.18

6 years ago

0.19.17

6 years ago

0.19.16

6 years ago

0.19.15

6 years ago

0.19.14

6 years ago

0.19.13

6 years ago

0.19.12

6 years ago

0.19.11

6 years ago

0.19.10

6 years ago

0.19.8

6 years ago

0.19.7

6 years ago

0.19.6

6 years ago

0.19.5

6 years ago

0.19.4

6 years ago

0.19.3

6 years ago

0.19.2

6 years ago

0.19.1

6 years ago

0.9.95

6 years ago

0.9.94

6 years ago

0.9.93

6 years ago

0.9.92

6 years ago

0.9.91

6 years ago

0.9.90

6 years ago

0.9.48

6 years ago

0.9.47

6 years ago

0.9.46

6 years ago

0.9.45

6 years ago

0.9.44

6 years ago

0.9.43

6 years ago

0.9.42

6 years ago

0.9.41

6 years ago

0.9.40

6 years ago

0.9.39

6 years ago

0.9.36

6 years ago

0.9.35

6 years ago

0.9.34

6 years ago

0.9.33

6 years ago

0.9.32

6 years ago

0.9.31

6 years ago

0.9.30

6 years ago

0.9.20

6 years ago

0.9.11

6 years ago

0.9.10

6 years ago

0.9.9

6 years ago

0.9.8

6 years ago

0.9.7

6 years ago

0.9.6

6 years ago

0.9.5

6 years ago

0.9.4

6 years ago

0.9.3

6 years ago

0.9.2

6 years ago

0.9.1

6 years ago

0.9.0

6 years ago

0.8.94

6 years ago

0.8.94-es6

6 years ago

0.8.92

6 years ago

0.8.91

6 years ago

0.8.90

6 years ago

0.8.89

6 years ago

0.8.88

6 years ago

0.8.87

6 years ago

0.8.86

6 years ago

0.8.85

6 years ago

0.8.84

6 years ago

0.8.83

6 years ago

0.8.82

6 years ago

0.8.81

6 years ago

0.8.80

6 years ago

0.8.75

6 years ago

0.8.74

6 years ago

0.8.73

6 years ago

0.8.72

6 years ago

0.8.71

6 years ago

0.8.70

6 years ago

0.8.69

6 years ago

0.8.68

6 years ago

0.8.67

6 years ago

0.8.66

6 years ago

0.8.65

6 years ago

0.8.63

6 years ago

0.8.62

6 years ago

0.8.61

6 years ago

0.8.60

6 years ago

0.8.59

6 years ago

0.8.58

6 years ago

0.8.57

6 years ago

0.8.56

6 years ago

0.8.55

6 years ago

0.8.54

6 years ago

0.8.53

6 years ago

0.8.51

6 years ago

0.8.50

6 years ago

0.8.33

6 years ago

0.8.32

6 years ago

0.8.31

6 years ago

0.8.30

6 years ago

0.8.29

6 years ago

0.8.28

6 years ago

0.8.27

6 years ago

0.8.26

6 years ago

0.8.25

6 years ago

0.8.24

6 years ago

0.8.23

6 years ago

0.8.22

6 years ago

0.8.21

6 years ago

0.8.20

6 years ago

0.8.12

6 years ago

0.8.11

6 years ago

0.8.10

6 years ago

0.8.9

6 years ago

0.8.8

6 years ago

0.8.7

6 years ago

0.8.6

6 years ago

0.8.5

6 years ago

0.8.4

6 years ago

0.8.3

6 years ago

0.8.2

6 years ago

0.8.1

6 years ago

0.8.0

6 years ago

0.7.15

6 years ago

0.7.12

6 years ago

0.7.11

6 years ago

0.7.10

6 years ago

0.7.9

6 years ago

0.7.8

6 years ago

0.7.7

6 years ago

0.7.6

6 years ago

0.7.5

6 years ago

0.7.4

6 years ago

0.7.3

6 years ago

0.7.2

6 years ago

0.7.1

6 years ago

0.7.0

6 years ago

0.6.9

6 years ago

0.6.8

6 years ago

0.6.7

6 years ago

0.6.5

6 years ago

0.6.4

6 years ago

0.6.3

6 years ago

0.6.2

6 years ago

0.6.1

6 years ago

0.6.0

6 years ago

0.5.9

6 years ago

0.5.8

6 years ago

0.5.7

6 years ago

0.5.6

6 years ago

0.5.5

6 years ago

0.5.4

6 years ago

0.5.3

6 years ago

0.5.2

6 years ago

0.5.1

6 years ago

0.5.0

6 years ago

0.4.6

6 years ago

0.4.5

6 years ago

0.4.4

6 years ago

0.4.3

6 years ago

0.4.2

6 years ago

0.4.1

6 years ago

0.4.0

6 years ago

0.3.2

6 years ago

0.3.1

6 years ago

0.3.0

6 years ago

0.2.3

6 years ago

0.2.2

6 years ago

0.2.1

6 years ago

0.2.0

6 years ago

0.1.38

6 years ago

0.1.37

6 years ago

0.1.36

6 years ago

0.1.35

6 years ago

0.1.34

6 years ago

0.1.33

6 years ago

0.1.32

6 years ago

0.1.31

6 years ago

0.1.30

6 years ago

0.1.29

6 years ago

0.1.28

6 years ago

0.1.27

6 years ago

0.1.26

6 years ago

0.1.25

6 years ago

0.1.24

6 years ago

0.1.23

6 years ago

0.1.22

6 years ago

0.1.21

6 years ago

0.1.20

6 years ago

0.1.19

6 years ago

0.1.17

6 years ago

0.1.16

6 years ago

0.1.15

6 years ago

0.1.14

6 years ago

0.1.12

6 years ago

0.1.11

6 years ago

0.1.10

6 years ago

0.1.0

6 years ago