Skip to content

Commit 408b307

Browse files
authored
Merge pull request #2 from vimeo/README_license
Add a README, a License and expand documentation with examples
2 parents 17102a6 + 4ef26f8 commit 408b307

File tree

6 files changed

+442
-6
lines changed

6 files changed

+442
-6
lines changed

LICENSE

Lines changed: 201 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,201 @@
1+
Apache License
2+
Version 2.0, January 2004
3+
http://www.apache.org/licenses/
4+
5+
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6+
7+
1. Definitions.
8+
9+
"License" shall mean the terms and conditions for use, reproduction,
10+
and distribution as defined by Sections 1 through 9 of this document.
11+
12+
"Licensor" shall mean the copyright owner or entity authorized by
13+
the copyright owner that is granting the License.
14+
15+
"Legal Entity" shall mean the union of the acting entity and all
16+
other entities that control, are controlled by, or are under common
17+
control with that entity. For the purposes of this definition,
18+
"control" means (i) the power, direct or indirect, to cause the
19+
direction or management of such entity, whether by contract or
20+
otherwise, or (ii) ownership of fifty percent (50%) or more of the
21+
outstanding shares, or (iii) beneficial ownership of such entity.
22+
23+
"You" (or "Your") shall mean an individual or Legal Entity
24+
exercising permissions granted by this License.
25+
26+
"Source" form shall mean the preferred form for making modifications,
27+
including but not limited to software source code, documentation
28+
source, and configuration files.
29+
30+
"Object" form shall mean any form resulting from mechanical
31+
transformation or translation of a Source form, including but
32+
not limited to compiled object code, generated documentation,
33+
and conversions to other media types.
34+
35+
"Work" shall mean the work of authorship, whether in Source or
36+
Object form, made available under the License, as indicated by a
37+
copyright notice that is included in or attached to the work
38+
(an example is provided in the Appendix below).
39+
40+
"Derivative Works" shall mean any work, whether in Source or Object
41+
form, that is based on (or derived from) the Work and for which the
42+
editorial revisions, annotations, elaborations, or other modifications
43+
represent, as a whole, an original work of authorship. For the purposes
44+
of this License, Derivative Works shall not include works that remain
45+
separable from, or merely link (or bind by name) to the interfaces of,
46+
the Work and Derivative Works thereof.
47+
48+
"Contribution" shall mean any work of authorship, including
49+
the original version of the Work and any modifications or additions
50+
to that Work or Derivative Works thereof, that is intentionally
51+
submitted to Licensor for inclusion in the Work by the copyright owner
52+
or by an individual or Legal Entity authorized to submit on behalf of
53+
the copyright owner. For the purposes of this definition, "submitted"
54+
means any form of electronic, verbal, or written communication sent
55+
to the Licensor or its representatives, including but not limited to
56+
communication on electronic mailing lists, source code control systems,
57+
and issue tracking systems that are managed by, or on behalf of, the
58+
Licensor for the purpose of discussing and improving the Work, but
59+
excluding communication that is conspicuously marked or otherwise
60+
designated in writing by the copyright owner as "Not a Contribution."
61+
62+
"Contributor" shall mean Licensor and any individual or Legal Entity
63+
on behalf of whom a Contribution has been received by Licensor and
64+
subsequently incorporated within the Work.
65+
66+
2. Grant of Copyright License. Subject to the terms and conditions of
67+
this License, each Contributor hereby grants to You a perpetual,
68+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69+
copyright license to reproduce, prepare Derivative Works of,
70+
publicly display, publicly perform, sublicense, and distribute the
71+
Work and such Derivative Works in Source or Object form.
72+
73+
3. Grant of Patent License. Subject to the terms and conditions of
74+
this License, each Contributor hereby grants to You a perpetual,
75+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76+
(except as stated in this section) patent license to make, have made,
77+
use, offer to sell, sell, import, and otherwise transfer the Work,
78+
where such license applies only to those patent claims licensable
79+
by such Contributor that are necessarily infringed by their
80+
Contribution(s) alone or by combination of their Contribution(s)
81+
with the Work to which such Contribution(s) was submitted. If You
82+
institute patent litigation against any entity (including a
83+
cross-claim or counterclaim in a lawsuit) alleging that the Work
84+
or a Contribution incorporated within the Work constitutes direct
85+
or contributory patent infringement, then any patent licenses
86+
granted to You under this License for that Work shall terminate
87+
as of the date such litigation is filed.
88+
89+
4. Redistribution. You may reproduce and distribute copies of the
90+
Work or Derivative Works thereof in any medium, with or without
91+
modifications, and in Source or Object form, provided that You
92+
meet the following conditions:
93+
94+
(a) You must give any other recipients of the Work or
95+
Derivative Works a copy of this License; and
96+
97+
(b) You must cause any modified files to carry prominent notices
98+
stating that You changed the files; and
99+
100+
(c) You must retain, in the Source form of any Derivative Works
101+
that You distribute, all copyright, patent, trademark, and
102+
attribution notices from the Source form of the Work,
103+
excluding those notices that do not pertain to any part of
104+
the Derivative Works; and
105+
106+
(d) If the Work includes a "NOTICE" text file as part of its
107+
distribution, then any Derivative Works that You distribute must
108+
include a readable copy of the attribution notices contained
109+
within such NOTICE file, excluding those notices that do not
110+
pertain to any part of the Derivative Works, in at least one
111+
of the following places: within a NOTICE text file distributed
112+
as part of the Derivative Works; within the Source form or
113+
documentation, if provided along with the Derivative Works; or,
114+
within a display generated by the Derivative Works, if and
115+
wherever such third-party notices normally appear. The contents
116+
of the NOTICE file are for informational purposes only and
117+
do not modify the License. You may add Your own attribution
118+
notices within Derivative Works that You distribute, alongside
119+
or as an addendum to the NOTICE text from the Work, provided
120+
that such additional attribution notices cannot be construed
121+
as modifying the License.
122+
123+
You may add Your own copyright statement to Your modifications and
124+
may provide additional or different license terms and conditions
125+
for use, reproduction, or distribution of Your modifications, or
126+
for any such Derivative Works as a whole, provided Your use,
127+
reproduction, and distribution of the Work otherwise complies with
128+
the conditions stated in this License.
129+
130+
5. Submission of Contributions. Unless You explicitly state otherwise,
131+
any Contribution intentionally submitted for inclusion in the Work
132+
by You to the Licensor shall be under the terms and conditions of
133+
this License, without any additional terms or conditions.
134+
Notwithstanding the above, nothing herein shall supersede or modify
135+
the terms of any separate license agreement you may have executed
136+
with Licensor regarding such Contributions.
137+
138+
6. Trademarks. This License does not grant permission to use the trade
139+
names, trademarks, service marks, or product names of the Licensor,
140+
except as required for reasonable and customary use in describing the
141+
origin of the Work and reproducing the content of the NOTICE file.
142+
143+
7. Disclaimer of Warranty. Unless required by applicable law or
144+
agreed to in writing, Licensor provides the Work (and each
145+
Contributor provides its Contributions) on an "AS IS" BASIS,
146+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147+
implied, including, without limitation, any warranties or conditions
148+
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149+
PARTICULAR PURPOSE. You are solely responsible for determining the
150+
appropriateness of using or redistributing the Work and assume any
151+
risks associated with Your exercise of permissions under this License.
152+
153+
8. Limitation of Liability. In no event and under no legal theory,
154+
whether in tort (including negligence), contract, or otherwise,
155+
unless required by applicable law (such as deliberate and grossly
156+
negligent acts) or agreed to in writing, shall any Contributor be
157+
liable to You for damages, including any direct, indirect, special,
158+
incidental, or consequential damages of any character arising as a
159+
result of this License or out of the use or inability to use the
160+
Work (including but not limited to damages for loss of goodwill,
161+
work stoppage, computer failure or malfunction, or any and all
162+
other commercial damages or losses), even if such Contributor
163+
has been advised of the possibility of such damages.
164+
165+
9. Accepting Warranty or Additional Liability. While redistributing
166+
the Work or Derivative Works thereof, You may choose to offer,
167+
and charge a fee for, acceptance of support, warranty, indemnity,
168+
or other liability obligations and/or rights consistent with this
169+
License. However, in accepting such obligations, You may act only
170+
on Your own behalf and on Your sole responsibility, not on behalf
171+
of any other Contributor, and only if You agree to indemnify,
172+
defend, and hold each Contributor harmless for any liability
173+
incurred by, or claims asserted against, such Contributor by reason
174+
of your accepting any such warranty or additional liability.
175+
176+
END OF TERMS AND CONDITIONS
177+
178+
APPENDIX: How to apply the Apache License to your work.
179+
180+
To apply the Apache License to your work, attach the following
181+
boilerplate notice, with the fields enclosed by brackets "[]"
182+
replaced with your own identifying information. (Don't include
183+
the brackets!) The text should be enclosed in the appropriate
184+
comment syntax for the file format. We also recommend that a
185+
file or class name and description of purpose be included on the
186+
same "printed page" as the copyright notice for easier
187+
identification within third-party archives.
188+
189+
Copyright 2020 Vimeo Inc.
190+
191+
Licensed under the Apache License, Version 2.0 (the "License");
192+
you may not use this file except in compliance with the License.
193+
You may obtain a copy of the License at
194+
195+
http://www.apache.org/licenses/LICENSE-2.0
196+
197+
Unless required by applicable law or agreed to in writing, software
198+
distributed under the License is distributed on an "AS IS" BASIS,
199+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
200+
See the License for the specific language governing permissions and
201+
limitations under the License.

README.md

Lines changed: 89 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
![Go](https://github.com/vimeo/leaderelection/workflows/Go/badge.svg)
2+
[![GoDoc](https://godoc.org/github.com/vimeo/leaderelection?status.svg)](https://godoc.org/github.com/vimeo/leaderelection)
3+
4+
# LeaderElection
5+
6+
LeaderElection is a library for electing a leader with a pluggable backend (a
7+
[`RaceDecider`]).
8+
9+
## [`RaceDecider`]
10+
The [`RaceDecider`] implementation needs to provide a conditional-write or
11+
transactional-write mechanism to ensure that only one write succeeds per
12+
election-cycle, but otherwise has very minimal requirements.
13+
14+
Currently, this implementation only has two implementations:
15+
- [Google Cloud Storage][`gcs`]: intended for easy production use while running in GCP, using a GCS object for locking.
16+
- [Memory][`memory`]: intended for tests.
17+
18+
## Electing a leader
19+
20+
The [`Config`] struct contains callbacks and tunables for the leader election
21+
"campaign".
22+
23+
Each process that would like to acquire leadership must register callbacks for
24+
all three of `OnElected`, `OnOusting` and `LeaderChanged`, as well as specify
25+
unique `LeaderID` and `HostPort`s (the latter two are used for communication, so
26+
some use-cases may be able to ignore them)
27+
28+
The `TermLength` is the length of time that a leader acquires leadership for,
29+
and the length of any extension. This duration should be long enough to get
30+
useful work done, but short enough that you won't have a problem if the leader
31+
dies and no one takes over the remainder of the lease term. The `Config.Acquire`
32+
method takes care of extending the lease twice per term to reduce the likelihood
33+
of spuriously losing the lock.
34+
35+
`MaxClockSkew` specifies the corrections added to and subtracted from sleeps
36+
and leases to account for a lack of perfect synchronization among the clocks of
37+
all candidates.
38+
39+
`ConnectionParams` is a generic byte-payload side-channel. The `legrpc` package
40+
uses it for the GRPC [`ServiceConfig`], but other use-cases may stash other
41+
serialized data there (may be `nil`).
42+
43+
`Clock` is an instance of a `clocks.Clock`, which should be `nil` outside of
44+
tests, in which case it uses `clocks.DefaultClock()`.
45+
46+
### There's more to Leadership than getting elected
47+
48+
The `OnElected` callback takes two arguments which indicate the state of the
49+
leadership lock with different degrees of certainty.
50+
51+
The `ctx` argument is a context derived from the one passed to `Acquire` that
52+
will be canceled upon losing the leadership role. This is an explicit
53+
cancellation by the goroutine handling lease renewals and acquisition, and as
54+
such is subject to normal thread-scheduling delay caveats (particularly relevant
55+
when operating with heavy CPU-contention).
56+
57+
To address the thread-scheduling-delay issues plaguing use of `ctx`, the
58+
second argument to `OnElected` is a `*TimeView` containing the current
59+
expiration time. This pointer tracks an atomic value which is updated every time
60+
the lease is extended. Before taking any action that requires holding
61+
leadership, one should always check that the time returned by `t.Get()` is in
62+
the future. `TimeView` has a `ValueInFuture()` convenience-method to facilitate
63+
such checks for quick operations against the correct clock.
64+
65+
## Picking a `RaceDecider`
66+
67+
The [`gcs`] [`RaceDecider`] is the only currently usable implementation. It
68+
requires a Google Cloud Storage client, a bucket and an object.
69+
70+
In tests, one can use the [`memory`] [`RaceDecider`], as that implementation
71+
is trivially fast and lacks any external dependencies, but doesn't work outside
72+
a single process.
73+
74+
## Watching an Election
75+
76+
Leader election is useful on its own only for a subset of use cases. Many times
77+
it is necessary for other processes to observe a leader election to send
78+
requests to the correct process. As such, one can define a [`WatchConfig`] and
79+
call [`Watch()`] on it. The callback will be called sequentially for every
80+
lease-extension and acquisition, thus allowing an observer to track the
81+
expiration of the current leader's leadership term.
82+
83+
[`RaceDecider`]: https://pkg.go.dev/github.com/vimeo/leaderelection?tab=doc#RaceDecider
84+
[`WatchConfig`]: https://pkg.go.dev/github.com/vimeo/leaderelection?tab=doc#WatchConfig
85+
[`Watch()`]: https://pkg.go.dev/github.com/vimeo/leaderelection?tab=doc#WatchConfig.Watch
86+
[`Config`]: https://pkg.go.dev/github.com/vimeo/leaderelection?tab=doc#Config
87+
[`ServiceConfig`]: https://github.com/grpc/grpc/blob/master/doc/service_config.md
88+
[`gcs`]: https://pkg.go.dev/github.com/vimeo/leaderelection/gcs
89+
[`memory`]: https://pkg.go.dev/github.com/vimeo/leaderelection/memory

campaign.go

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,7 @@ const jitterTermMaxFraction = 0.1
126126

127127
func (c *campaign) manageWin(ctx context.Context, winningEntry *entry.RaceEntry, wg *sync.WaitGroup) (*entry.RaceEntry, error) {
128128
// we won!
129-
tv := TimeView{}
129+
tv := TimeView{clock: c.clock}
130130
tv.Set(winningEntry.TermExpiry)
131131
// run the elected callback in a background goroutine and cancel the
132132
// context on ousting

campaign_test.go

Lines changed: 67 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ package leaderelection
22

33
import (
44
"context"
5+
"fmt"
56
"math/rand"
67
"os"
78
"strconv"
@@ -723,3 +724,69 @@ func TestMultipleContendersWithFakeClockAndSkew(t *testing.T) {
723724

724725
cancel()
725726
}
727+
728+
func ExampleConfig_Acquire() {
729+
ctx, cancel := context.WithCancel(context.Background())
730+
defer cancel()
731+
d := memory.NewDecider()
732+
733+
now := time.Date(2020, 5, 6, 0, 0, 0, 0, time.UTC)
734+
fc := clocks.NewFakeClock(now)
735+
736+
tvLeaderIndicator := (*TimeView)(nil)
737+
electedCh := make(chan struct{})
738+
c := Config{
739+
Decider: d,
740+
HostPort: "127.0.0.1:8080",
741+
LeaderID: "yabadabadoo",
742+
OnElected: func(ctx context.Context, tv *TimeView) {
743+
fmt.Printf("I won! Leader term expiration: %s; held: %t\n",
744+
tv.Get(), tv.ValueInFuture())
745+
tvLeaderIndicator = tv
746+
close(electedCh)
747+
},
748+
OnOusting: func(ctx context.Context) {
749+
// make sure we've already set `tvLeaderIndictor` before touching it
750+
<-electedCh
751+
fmt.Printf("I lost! Holding Lock: %t; expires: %s\n",
752+
// Note that ValueInFuture will return true
753+
// here because the clock is still within the
754+
// term.
755+
tvLeaderIndicator.ValueInFuture(),
756+
tvLeaderIndicator.Get())
757+
},
758+
LeaderChanged: func(ctx context.Context, entry entry.RaceEntry) {
759+
fmt.Printf("%q won\n", entry.LeaderID)
760+
},
761+
TermLength: time.Minute * 30,
762+
MaxClockSkew: time.Second * 10,
763+
Clock: fc,
764+
}
765+
acquireCh := make(chan error, 1)
766+
go func() {
767+
acquireCh <- c.Acquire(ctx)
768+
}()
769+
770+
fc.AwaitSleepers(1)
771+
<-electedCh
772+
fc.Advance(c.TermLength / 2)
773+
774+
// Wait for the lease renewal to happen before cancelling (so the
775+
// output is predictable)
776+
fc.AwaitSleepers(1)
777+
778+
cancel()
779+
// Acquire blocks until all callbacks return (there's an internal WaitGroup)
780+
<-acquireCh
781+
// advance past the end of the current term (after an extension)
782+
fc.Advance(c.TermLength + time.Minute)
783+
fmt.Printf("Still Leading: %t; expiry: %s; current time: %s\n",
784+
tvLeaderIndicator.ValueInFuture(),
785+
tvLeaderIndicator.Get(), fc.Now())
786+
787+
// Unordered output:
788+
// I won! Leader term expiration: 2020-05-06 00:30:00 +0000 UTC; held: true
789+
// I lost! Holding Lock: true; expires: 2020-05-06 00:45:00 +0000 UTC
790+
// "" won
791+
// Still Leading: false; expiry: 2020-05-06 00:45:00 +0000 UTC; current time: 2020-05-06 00:46:00 +0000 UTC
792+
}

0 commit comments

Comments
 (0)