OLD | NEW |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 library barback.pool; | 5 library barback.pool; |
6 | 6 |
7 import 'dart:async'; | 7 import 'dart:async'; |
8 import 'dart:collection'; | 8 import 'dart:collection'; |
9 | 9 |
10 import 'package:stack_trace/stack_trace.dart'; | 10 import 'package:stack_trace/stack_trace.dart'; |
11 | 11 |
| 12 import 'utils.dart'; |
| 13 |
12 // TODO(nweiz): put this somewhere that it can be shared between packages. | 14 // TODO(nweiz): put this somewhere that it can be shared between packages. |
13 /// Manages an abstract pool of resources with a limit on how many may be in use | 15 /// Manages an abstract pool of resources with a limit on how many may be in use |
14 /// at once. | 16 /// at once. |
15 /// | 17 /// |
16 /// When a resource is needed, the user should call [request]. When the returned | 18 /// When a resource is needed, the user should call [request]. When the returned |
17 /// future completes with a [PoolResource], the resource may be allocated. Once | 19 /// future completes with a [PoolResource], the resource may be allocated. Once |
18 /// the resource has been released, the user should call [PoolResource.release]. | 20 /// the resource has been released, the user should call [PoolResource.release]. |
19 /// The pool will ensure that only a certain number of [PoolResource]s may be | 21 /// The pool will ensure that only a certain number of [PoolResource]s may be |
20 /// allocated at once. | 22 /// allocated at once. |
21 class Pool { | 23 class Pool { |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
67 return completer.future; | 69 return completer.future; |
68 } | 70 } |
69 } | 71 } |
70 | 72 |
71 /// Requests a resource for the duration of [callback], which may return a | 73 /// Requests a resource for the duration of [callback], which may return a |
72 /// Future. | 74 /// Future. |
73 /// | 75 /// |
74 /// The return value of [callback] is piped to the returned Future. | 76 /// The return value of [callback] is piped to the returned Future. |
75 Future withResource(callback()) { | 77 Future withResource(callback()) { |
76 return request().then((resource) => | 78 return request().then((resource) => |
77 new Future.sync(callback).whenComplete(resource.release)); | 79 syncFuture(callback).whenComplete(resource.release)); |
78 } | 80 } |
79 | 81 |
80 /// If there are any pending requests, this will fire the oldest one. | 82 /// If there are any pending requests, this will fire the oldest one. |
81 void _onResourceReleased() { | 83 void _onResourceReleased() { |
82 if (_requestedResources.isEmpty) { | 84 if (_requestedResources.isEmpty) { |
83 _allocatedResources--; | 85 _allocatedResources--; |
84 if (_timer != null) { | 86 if (_timer != null) { |
85 _timer.cancel(); | 87 _timer.cancel(); |
86 _timer = null; | 88 _timer = null; |
87 } | 89 } |
(...skipping 13 matching lines...) Expand all Loading... |
101 } else { | 103 } else { |
102 _timer = new Timer(_timeout, _onTimeout); | 104 _timer = new Timer(_timeout, _onTimeout); |
103 } | 105 } |
104 } | 106 } |
105 | 107 |
106 /// Handles [_timer] timing out by causing all pending resource completers to | 108 /// Handles [_timer] timing out by causing all pending resource completers to |
107 /// emit exceptions. | 109 /// emit exceptions. |
108 void _onTimeout() { | 110 void _onTimeout() { |
109 for (var completer in _requestedResources) { | 111 for (var completer in _requestedResources) { |
110 completer.completeError("Pool deadlock: all resources have been " | 112 completer.completeError("Pool deadlock: all resources have been " |
111 "allocated for too long.", new Trace.current().vmTrace); | 113 "allocated for too long.", new Chain.current()); |
112 } | 114 } |
113 _requestedResources.clear(); | 115 _requestedResources.clear(); |
114 _timer = null; | 116 _timer = null; |
115 } | 117 } |
116 } | 118 } |
117 | 119 |
118 /// A member of a [Pool]. | 120 /// A member of a [Pool]. |
119 /// | 121 /// |
120 /// A [PoolResource] is a token that indicates that a resource is allocated. | 122 /// A [PoolResource] is a token that indicates that a resource is allocated. |
121 /// When the associated resource is released, the user should call [release]. | 123 /// When the associated resource is released, the user should call [release]. |
122 class PoolResource { | 124 class PoolResource { |
123 final Pool _pool; | 125 final Pool _pool; |
124 | 126 |
125 /// Whether [this] has been released yet. | 127 /// Whether [this] has been released yet. |
126 bool _released = false; | 128 bool _released = false; |
127 | 129 |
128 PoolResource._(this._pool); | 130 PoolResource._(this._pool); |
129 | 131 |
130 /// Tells the parent [Pool] that the resource associated with this resource is | 132 /// Tells the parent [Pool] that the resource associated with this resource is |
131 /// no longer allocated, and that a new [PoolResource] may be allocated. | 133 /// no longer allocated, and that a new [PoolResource] may be allocated. |
132 void release() { | 134 void release() { |
133 if (_released) { | 135 if (_released) { |
134 throw new StateError("A PoolResource may only be released once."); | 136 throw new StateError("A PoolResource may only be released once."); |
135 } | 137 } |
136 _released = true; | 138 _released = true; |
137 _pool._onResourceReleased(); | 139 _pool._onResourceReleased(); |
138 } | 140 } |
139 } | 141 } |
OLD | NEW |