| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | package metrics | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	"sync/atomic" | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Meters count events to produce exponentially-weighted moving average rates | 
					
						
							|  |  |  | // at one-, five-, and fifteen-minutes and a mean rate. | 
					
						
							|  |  |  | type Meter interface { | 
					
						
							|  |  |  | 	Count() int64 | 
					
						
							|  |  |  | 	Mark(int64) | 
					
						
							|  |  |  | 	Rate1() float64 | 
					
						
							|  |  |  | 	Rate5() float64 | 
					
						
							|  |  |  | 	Rate15() float64 | 
					
						
							|  |  |  | 	RateMean() float64 | 
					
						
							|  |  |  | 	Snapshot() Meter | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 	Stop() | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetOrRegisterMeter returns an existing Meter or constructs and registers a | 
					
						
							|  |  |  | // new StandardMeter. | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Be sure to unregister the meter from the registry once it is of no use to | 
					
						
							|  |  |  | // allow for garbage collection. | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func GetOrRegisterMeter(name string, r Registry) Meter { | 
					
						
							|  |  |  | 	if nil == r { | 
					
						
							|  |  |  | 		r = DefaultRegistry | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return r.GetOrRegister(name, NewMeter).(Meter) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 20:32:34 +08:00
										 |  |  | // GetOrRegisterMeterForced returns an existing Meter or constructs and registers a | 
					
						
							|  |  |  | // new StandardMeter no matter the global switch is enabled or not. | 
					
						
							|  |  |  | // Be sure to unregister the meter from the registry once it is of no use to | 
					
						
							|  |  |  | // allow for garbage collection. | 
					
						
							|  |  |  | func GetOrRegisterMeterForced(name string, r Registry) Meter { | 
					
						
							|  |  |  | 	if nil == r { | 
					
						
							|  |  |  | 		r = DefaultRegistry | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return r.GetOrRegister(name, NewMeterForced).(Meter) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | // NewMeter constructs a new StandardMeter and launches a goroutine. | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Be sure to call Stop() once the meter is of no use to allow for garbage collection. | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func NewMeter() Meter { | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 	if !Enabled { | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 		return NilMeter{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	m := newStandardMeter() | 
					
						
							|  |  |  | 	arbiter.Lock() | 
					
						
							|  |  |  | 	defer arbiter.Unlock() | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 	arbiter.meters[m] = struct{}{} | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 	if !arbiter.started { | 
					
						
							|  |  |  | 		arbiter.started = true | 
					
						
							|  |  |  | 		go arbiter.tick() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return m | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 20:32:34 +08:00
										 |  |  | // NewMeterForced constructs a new StandardMeter and launches a goroutine no matter | 
					
						
							|  |  |  | // the global switch is enabled or not. | 
					
						
							|  |  |  | // Be sure to call Stop() once the meter is of no use to allow for garbage collection. | 
					
						
							|  |  |  | func NewMeterForced() Meter { | 
					
						
							|  |  |  | 	m := newStandardMeter() | 
					
						
							|  |  |  | 	arbiter.Lock() | 
					
						
							|  |  |  | 	defer arbiter.Unlock() | 
					
						
							|  |  |  | 	arbiter.meters[m] = struct{}{} | 
					
						
							|  |  |  | 	if !arbiter.started { | 
					
						
							|  |  |  | 		arbiter.started = true | 
					
						
							|  |  |  | 		go arbiter.tick() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return m | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewRegisteredMeter constructs and registers a new StandardMeter | 
					
						
							|  |  |  | // and launches a goroutine. | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Be sure to unregister the meter from the registry once it is of no use to | 
					
						
							|  |  |  | // allow for garbage collection. | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func NewRegisteredMeter(name string, r Registry) Meter { | 
					
						
							|  |  |  | 	c := NewMeter() | 
					
						
							|  |  |  | 	if nil == r { | 
					
						
							|  |  |  | 		r = DefaultRegistry | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	r.Register(name, c) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 20:32:34 +08:00
										 |  |  | // NewRegisteredMeterForced constructs and registers a new StandardMeter | 
					
						
							|  |  |  | // and launches a goroutine no matter the global switch is enabled or not. | 
					
						
							|  |  |  | // Be sure to unregister the meter from the registry once it is of no use to | 
					
						
							|  |  |  | // allow for garbage collection. | 
					
						
							|  |  |  | func NewRegisteredMeterForced(name string, r Registry) Meter { | 
					
						
							|  |  |  | 	c := NewMeterForced() | 
					
						
							|  |  |  | 	if nil == r { | 
					
						
							|  |  |  | 		r = DefaultRegistry | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	r.Register(name, c) | 
					
						
							|  |  |  | 	return c | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | // MeterSnapshot is a read-only copy of another Meter. | 
					
						
							|  |  |  | type MeterSnapshot struct { | 
					
						
							| 
									
										
										
										
											2020-08-21 10:04:36 +02:00
										 |  |  | 	// WARNING: The `temp` field is accessed atomically. | 
					
						
							|  |  |  | 	// On 32 bit platforms, only 64-bit aligned fields can be atomic. The struct is | 
					
						
							|  |  |  | 	// guaranteed to be so aligned, so take advantage of that. For more information, | 
					
						
							|  |  |  | 	// see https://golang.org/pkg/sync/atomic/#pkg-note-BUG. | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	temp                           int64 | 
					
						
							| 
									
										
										
										
											2020-08-21 10:04:36 +02:00
										 |  |  | 	count                          int64 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 	rate1, rate5, rate15, rateMean float64 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Count returns the count of events at the time the snapshot was taken. | 
					
						
							|  |  |  | func (m *MeterSnapshot) Count() int64 { return m.count } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Mark panics. | 
					
						
							|  |  |  | func (*MeterSnapshot) Mark(n int64) { | 
					
						
							|  |  |  | 	panic("Mark called on a MeterSnapshot") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate1 returns the one-minute moving average rate of events per second at the | 
					
						
							|  |  |  | // time the snapshot was taken. | 
					
						
							|  |  |  | func (m *MeterSnapshot) Rate1() float64 { return m.rate1 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate5 returns the five-minute moving average rate of events per second at | 
					
						
							|  |  |  | // the time the snapshot was taken. | 
					
						
							|  |  |  | func (m *MeterSnapshot) Rate5() float64 { return m.rate5 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate15 returns the fifteen-minute moving average rate of events per second | 
					
						
							|  |  |  | // at the time the snapshot was taken. | 
					
						
							|  |  |  | func (m *MeterSnapshot) Rate15() float64 { return m.rate15 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RateMean returns the meter's mean rate of events per second at the time the | 
					
						
							|  |  |  | // snapshot was taken. | 
					
						
							|  |  |  | func (m *MeterSnapshot) RateMean() float64 { return m.rateMean } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Snapshot returns the snapshot. | 
					
						
							|  |  |  | func (m *MeterSnapshot) Snapshot() Meter { return m } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Stop is a no-op. | 
					
						
							|  |  |  | func (m *MeterSnapshot) Stop() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | // NilMeter is a no-op Meter. | 
					
						
							|  |  |  | type NilMeter struct{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Count is a no-op. | 
					
						
							|  |  |  | func (NilMeter) Count() int64 { return 0 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Mark is a no-op. | 
					
						
							|  |  |  | func (NilMeter) Mark(n int64) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate1 is a no-op. | 
					
						
							|  |  |  | func (NilMeter) Rate1() float64 { return 0.0 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate5 is a no-op. | 
					
						
							|  |  |  | func (NilMeter) Rate5() float64 { return 0.0 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | // Rate15 is a no-op. | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func (NilMeter) Rate15() float64 { return 0.0 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RateMean is a no-op. | 
					
						
							|  |  |  | func (NilMeter) RateMean() float64 { return 0.0 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Snapshot is a no-op. | 
					
						
							|  |  |  | func (NilMeter) Snapshot() Meter { return NilMeter{} } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Stop is a no-op. | 
					
						
							|  |  |  | func (NilMeter) Stop() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | // StandardMeter is the standard implementation of a Meter. | 
					
						
							|  |  |  | type StandardMeter struct { | 
					
						
							|  |  |  | 	lock        sync.RWMutex | 
					
						
							|  |  |  | 	snapshot    *MeterSnapshot | 
					
						
							|  |  |  | 	a1, a5, a15 EWMA | 
					
						
							|  |  |  | 	startTime   time.Time | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	stopped     uint32 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func newStandardMeter() *StandardMeter { | 
					
						
							|  |  |  | 	return &StandardMeter{ | 
					
						
							|  |  |  | 		snapshot:  &MeterSnapshot{}, | 
					
						
							|  |  |  | 		a1:        NewEWMA1(), | 
					
						
							|  |  |  | 		a5:        NewEWMA5(), | 
					
						
							|  |  |  | 		a15:       NewEWMA15(), | 
					
						
							|  |  |  | 		startTime: time.Now(), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Stop stops the meter, Mark() will be a no-op if you use it after being stopped. | 
					
						
							|  |  |  | func (m *StandardMeter) Stop() { | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	stopped := atomic.SwapUint32(&m.stopped, 1) | 
					
						
							|  |  |  | 	if stopped != 1 { | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 		arbiter.Lock() | 
					
						
							|  |  |  | 		delete(arbiter.meters, m) | 
					
						
							|  |  |  | 		arbiter.Unlock() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | // Count returns the number of events recorded. | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | // It updates the meter to be as accurate as possible | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func (m *StandardMeter) Count() int64 { | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	m.lock.Lock() | 
					
						
							|  |  |  | 	defer m.lock.Unlock() | 
					
						
							|  |  |  | 	m.updateMeter() | 
					
						
							|  |  |  | 	return m.snapshot.count | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // Mark records the occurrence of n events. | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func (m *StandardMeter) Mark(n int64) { | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	atomic.AddInt64(&m.snapshot.temp, n) | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate1 returns the one-minute moving average rate of events per second. | 
					
						
							|  |  |  | func (m *StandardMeter) Rate1() float64 { | 
					
						
							|  |  |  | 	m.lock.RLock() | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	defer m.lock.RUnlock() | 
					
						
							|  |  |  | 	return m.snapshot.rate1 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate5 returns the five-minute moving average rate of events per second. | 
					
						
							|  |  |  | func (m *StandardMeter) Rate5() float64 { | 
					
						
							|  |  |  | 	m.lock.RLock() | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	defer m.lock.RUnlock() | 
					
						
							|  |  |  | 	return m.snapshot.rate5 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Rate15 returns the fifteen-minute moving average rate of events per second. | 
					
						
							|  |  |  | func (m *StandardMeter) Rate15() float64 { | 
					
						
							|  |  |  | 	m.lock.RLock() | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	defer m.lock.RUnlock() | 
					
						
							|  |  |  | 	return m.snapshot.rate15 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // RateMean returns the meter's mean rate of events per second. | 
					
						
							|  |  |  | func (m *StandardMeter) RateMean() float64 { | 
					
						
							|  |  |  | 	m.lock.RLock() | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	defer m.lock.RUnlock() | 
					
						
							|  |  |  | 	return m.snapshot.rateMean | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Snapshot returns a read-only copy of the meter. | 
					
						
							|  |  |  | func (m *StandardMeter) Snapshot() Meter { | 
					
						
							|  |  |  | 	m.lock.RLock() | 
					
						
							|  |  |  | 	snapshot := *m.snapshot | 
					
						
							|  |  |  | 	m.lock.RUnlock() | 
					
						
							|  |  |  | 	return &snapshot | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (m *StandardMeter) updateSnapshot() { | 
					
						
							|  |  |  | 	// should run with write lock held on m.lock | 
					
						
							|  |  |  | 	snapshot := m.snapshot | 
					
						
							|  |  |  | 	snapshot.rate1 = m.a1.Rate() | 
					
						
							|  |  |  | 	snapshot.rate5 = m.a5.Rate() | 
					
						
							|  |  |  | 	snapshot.rate15 = m.a15.Rate() | 
					
						
							|  |  |  | 	snapshot.rateMean = float64(snapshot.count) / time.Since(m.startTime).Seconds() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | func (m *StandardMeter) updateMeter() { | 
					
						
							|  |  |  | 	// should only run with write lock held on m.lock | 
					
						
							| 
									
										
										
										
											2020-08-21 10:04:36 +02:00
										 |  |  | 	n := atomic.SwapInt64(&m.snapshot.temp, 0) | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	m.snapshot.count += n | 
					
						
							|  |  |  | 	m.a1.Update(n) | 
					
						
							|  |  |  | 	m.a5.Update(n) | 
					
						
							|  |  |  | 	m.a15.Update(n) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | func (m *StandardMeter) tick() { | 
					
						
							|  |  |  | 	m.lock.Lock() | 
					
						
							|  |  |  | 	defer m.lock.Unlock() | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | 	m.updateMeter() | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 	m.a1.Tick() | 
					
						
							|  |  |  | 	m.a5.Tick() | 
					
						
							|  |  |  | 	m.a15.Tick() | 
					
						
							|  |  |  | 	m.updateSnapshot() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | // meterArbiter ticks meters every 5s from a single goroutine. | 
					
						
							|  |  |  | // meters are references in a set for future stopping. | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | type meterArbiter struct { | 
					
						
							|  |  |  | 	sync.RWMutex | 
					
						
							|  |  |  | 	started bool | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 	meters  map[*StandardMeter]struct{} | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 	ticker  *time.Ticker | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-18 11:27:04 +02:00
										 |  |  | var arbiter = meterArbiter{ticker: time.NewTicker(5 * time.Second), meters: make(map[*StandardMeter]struct{})} | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Ticks meters on the scheduled interval | 
					
						
							|  |  |  | func (ma *meterArbiter) tick() { | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 	for range ma.ticker.C { | 
					
						
							|  |  |  | 		ma.tickMeters() | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (ma *meterArbiter) tickMeters() { | 
					
						
							|  |  |  | 	ma.RLock() | 
					
						
							|  |  |  | 	defer ma.RUnlock() | 
					
						
							| 
									
										
										
										
											2018-02-23 10:56:08 +01:00
										 |  |  | 	for meter := range ma.meters { | 
					
						
							| 
									
										
										
										
											2015-06-22 12:00:55 +03:00
										 |  |  | 		meter.tick() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |