Full Code of bmizerany/noeqd for AI

master c3ea4225e1fe cached
8 files
17.0 KB
5.2k tokens
32 symbols
1 requests
Download .txt
Repository: bmizerany/noeqd
Branch: master
Commit: c3ea4225e1fe
Files: 8
Total size: 17.0 KB

Directory structure:
gitextract_ldnxyfje/

├── .gitignore
├── Makefile
├── README.md
├── bench.sh
├── bench_test.go
├── coord-exec.sh
├── main.go
└── main_test.go

================================================
FILE CONTENTS
================================================

================================================
FILE: .gitignore
================================================
*.[568]
[568].out
_obj
_test*
a.out
*.gz


================================================
FILE: Makefile
================================================
# figure out what GOROOT is supposed to be
GOROOT ?= $(shell printf 't:;@echo $$(GOROOT)\n' | gomake -f -)
include $(GOROOT)/src/Make.inc

TARG=noeqd
GOFILES=\
	main.go\

include $(GOROOT)/src/Make.cmd

VERSION=$(shell git describe --tags --always)

tar: clean $(TARG)
	tar -czf $(TARG)-$(VERSION)-$(GOOS)-$(GOARCH).tar.gz $(TARG) README.md


================================================
FILE: README.md
================================================
# noeqd - A fault-tolerant network service for meaningful GUID generation

Based on [snowflake][].

## Motivation

GUIDs (Globally Unique IDs) are useful for a number a obvious reasons:
database keys, logging, etc.

Generating GUIDs with pure randomness is not always ideal because it doesn't
cluster well, produces terrible locality, and no insight as to when it was
generated.

This network service should also have these properties (Differences from [snowflake][]):

* easy distribution with *no dependencies* and little to *no setup*
* dirt simple wire-protocol (trivial to implement clients without added dependencies and complexity)
* low memory footprint (starts and stays around ~1MB)
* zero configuration
* reduced network IO when multiple keys are needed at once

## Glossary of terms to follow

* `GUID`: Globally Unique Identifier
* `datacenter`: A facility used to house computer systems.
* `worker`: A single `noeqd` process with a worker and datacenter ID combination unique to their cohort.
* `datacenter-id`: An integer representing a particular datacenter.
* `worker-id`: An integer representing a particular worker.
* `machine-id`: The comination of `datacenter-id` and `worker-id`
* `twepoch`: custom epoch (same as [snowflake][])

## Important note:

Reliability, and guarantees depend on:

**System clock depedency and skew protection:** - (From [snowflake][] README and slightly modified)

You should use NTP to keep your system clock accurate. Noeq protects from
non-monotonic clocks, i.e. clocks that run backwards. If your clock is running
fast and NTP tells it to repeat a few milliseconds, Noeq will refuse to
generate ids until a time that is after the last time we generated an id. Even
better, run in a mode where ntp won't move the clock backwards. See
<http://wiki.dovecot.org/TimeMovedBackwards#Time_synchronization> for tips on how
to do this.

**Avoiding the reuse of a worker-id + datacenter-id too quickly**

It's important to know that a newly born process has no way of tracking its
previous life and where it left of. This means time could have moved
backwards while it was dead.

It's important to **not** use the same worker-id + datacenter-id without
telling the new process when to start generating new IDs to avoid duplicates.

It is only safe to reuse the same worker-id + datacenter-id when you can
guarantee the current time is greater than the time of death. You can use the
`-t` option to specifiy this.

You may have up to 1024 machine ids. It's generally safe to not reuse them
until you've reached this limit.

## Install

You can install noeqd by downloading the binary
[here](http://github.com/bmizerany/noeqd/downloads) and putting it in your
`PATH`.

*or*

Clone the repo and build with [Go](http://golang.org/doc/install.html) (Requires Go `0beb796b4ef8 weekly/weekly.2011-12-02` or later)

		$ git clone http://github.com/bmizerany/noeqd
		$ cd noeqd
		$ make install

## Run

		$ noeqd -h
		Usage of noeqd:
		  -d=0: datacenter id
		  -l="0.0.0.0:4444": the address to listen on
		  -w=0: worker id

**Coordinating machine-ids**

Noeq does not assume you're using any automated coordination because it isn't
always correct to assume this. Its easy to do without baking it in. Here is an
example script in the repo for doing so if you need it (using [Doozer][]):

		#!/bin/sh
		# usage: ./coord-exec.sh <datacenter-id>

		did=$1
		wid=0

		[ -z "$did" ] && did=0

		_set() {
		  printf 1 | doozer set /goflake/$did/$wid 0
		}

		while ! _set
		do wid=`expr $wid + 1`
		done

		exec noeqd -w $wid -d $did

## The Why

**Uniqness**

We must know that a GUID, once generated, has never and will never be generated
again (i.e. Globally Unique) in our system.

**Performance**

Heroku serves many 10's of thousands of requests a second. Each request can
require multiple actions that need their own id. To be on the safe side, we
will require a minimum of 100k ids/sec (without network latency); possibly more
in the very near future. See benchmarks near the end of this README.

**Uncoordinated**

We need all `noeqd`s to be able to generate GUIDs without coordinating with
other `noeqd` processes. Coordination requires more time complexity than if
we didn't require it and reduces the amount of GUIDs we can generate during
that time. It also affects the yield (the probability the service will complete
a request).

**Direcly sortable by time (roughly)**

Noeq (like [snowflake][]) will guarantee the GUIDs will be k-sorted within
a reasonable bound (10's of ms to no more than 1s). More on this in "How it works."

References:

<http://ci.nii.ac.jp/naid/110002673556>

<http://ci.nii.ac.jp/naid/110002673489>

# The "Why not snowflake?"

At Heroku, we value services that are simple, as self-contained as possible,
and use nothing more than they can reasonably get away with. The setup and
distribution of an application should be as quick and painless as possible.
This means ruthlessly eliminating as much baggage, waste, and other overhead as
possible.

# How it works

## GUID generation and guarantees

GUIDs are represented as 64bit integers and are composed of (as described by the [snowflake][] README):

* time - 41 bits (millisecond precision with a custom epoch gives us 69 years)
* configured machine id - 10 bits - gives us up to 1024 machines
* sequence number - 12 bits - rolls over every 4096 per machine (with protection to avoid rollover in the same ms)

## Sorting - Time Ordered

*Strictly sorted*:

* GUIDs generated in a single request by a worker will strictly sort.
* GUIDs generated one second or longer apart, by more than one worker, will strictly sort.
* GUIDs generated over multiple requests by the same worker, will strictly sort.

*Roughly sorted*:

* GUIDs generated by multiple workers within a second could roughly sort.

An example of roughly sorted:

If client A requests three GUIDs from worker A in one request, and client B
requests three GUIDs from worker B in another request, and both requests are
processed within the same second, together they may sort like:

		GUID-A1
		GUID-A2
		GUID-B1
		GUID-B2
		GUID-A3
		GUID-B3

NOTE: The A GUIDs will strictly sort, as will B's.

## Clients

Clients implement a simple wire-protocol that is specified below. Implementing
a client in your favorite language is trivial and should require no
dependencies.

**Failure Recovery**

Each client should keep a list of addresses of all known worker process (or
use DNS) so that if one fails, it can move to another. To recover
from a lost connection, a client should randomly select another address from
its list, or in the case of DNS: reconnect using the same address allowing DNS
to choose the next IP.

See [noeq.go](http://github.com/bmizerany/noeq.go) for a working
example.

## Protocol

*Auth Request*

If the server has its `NOEQ_TOKEN` environment variable set to an non-empty string, the server will require authentication.

		------------------------
		|0|<len byte>|token ...|
		------------------------

An auth request starts with byte(0) and then a byte that is the length of the
follwing token, followed by the token string. (NOTE: If a client and server
hang during authentication, it's probably because the token is the client sent
is too short.)

*Id Request*:

		-------
		|uint8|
		-------

A request must contain only one byte. The value of the byte tells the
server how many ids to respond with. A client can request up to 255 (or max uint8)
ids per request.

*Response*:

		-------------------------------------------------- ...
		|uint8|uint8|uint8|uint8|uint8|uint8|uint8|uint8|  ...
		-------------------------------------------------- ...

Each id comes as a 64bit integer in network byte order. The
number of 64bit integers returned is the `request-byte * 8`

*Errors*:

Errors are logged by the server to stdout. Clients will have their
connection closed to signal the need to try elsewhere until the server can
recover. This generally happens if the servers clock is running backwards.

## Benchmarks (fwiw)

**MacAir 3, OS X 10.7.2, 2.13 GHz Intel Core 2 Duo, 4 GB 1067 MHz DDR3)**


**Id Generation *without encoding* or network latency**

This is the benchmark done by [snowflake] and reported in their README.

		BenchmarkIdGeneration	675 ns/op	# 1.481 million ids/s
		
**Id Generation *with* encoding and *without* network latency**

I find these benchmarks more realistic. The ids must be encoded so we want to
know how fast an id can be generated and encoded in order to hit the wire.
Benchmarks including a network are left as an exercise for the reader because
all networks vary.

These show that when a client can safely ask for more one than one id at a time,
they can reduce time to wire and the expensive read/write operations.

		BenchmarkServe01	 1677 ns/op	# 596303 ids/sec
		BenchmarkServe02	 2352 ns/op	# 850340 ids/sec
		BenchmarkServe03	 3067 ns/op	# 978155 ids/sec
		BenchmarkServe05	 4436 ns/op	# 1.127 million ids/sec
		BenchmarkServe08	 6436 ns/op	# 1.243 million ids/sec
		BenchmarkServe13	10169 ns/op	# 1.278 million ids/sec
		BenchmarkServe21	16257 ns/op	# 1.292 million ids/sec
		BenchmarkServe34	25603 ns/op	# 1.328 million ids/sec
		BenchmarkServe55	39693 ns/op	# 1.386 million ids/sec

## Contributing

This is Github. You know the drill. Please make sure you keep your changes in a
branch other than `master` and in nice, clean, atomic commits. If you modify a
`.go` file, please use `gofmt` with no parameters to format it; then hit the
pull-request button.

## Issues

These are tracked in this repos Github [issues tracker](http://github.com/bmizerany/noeqd/issues).

## See Also

Noeq command line util:
<http://github.com/bmizerany/noeq>

Noeq.go for Go:
<http://github.com/bmizerany/noeq.go>

## Thank you

I want to make sure I give the Snowflake team at Twitter as much credit as
possible. The heart of this program is their doing.

## LICENSE

Copyright (C) 2011 by Blake Mizerany ([@bmizerany](http://twitter.com/bmizerany))

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE. 

[Doozer]: http://github.com/ha/doozerd
[snowflake]: http://github.com/twitter/snowflake


================================================
FILE: bench.sh
================================================
#!/bin/sh
gotest -run="no tests" -bench=".*"


================================================
FILE: bench_test.go
================================================
package main

import (
	"bytes"
	"testing"
)

type nopWriter struct{}

func (nw *nopWriter) Write(b []byte) (n int, err error) {
	return
}

func BenchmarkIdGeneration(b *testing.B) {
	for n := 0; n < b.N; n++ {
		nextId()
	}
}

func BenchmarkServe01(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{1}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe02(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{2}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe03(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{3}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe05(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{5}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe08(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{8}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe13(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{13}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe21(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{21}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe34(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{34}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}

func BenchmarkServe55(b *testing.B) {
	for n := 0; n < b.N; n++ {
		b.StopTimer()
		i, o := bytes.NewBuffer([]byte{55}), new(nopWriter)
		b.StartTimer()
		serve(i, o)
	}
}


================================================
FILE: coord-exec.sh
================================================
#!/bin/sh

#/ usage: exec.sh

did=$1
wid=0

[ -z "$did" ] && did=0

_set() {
  # TODO: use an id rather than 1
  printf 1 | doozer set /noeq/$did/$wid 0
}

while ! _set
do wid=`expr $wid + 1`
done

exec noeqd -w $wid -d $did


================================================
FILE: main.go
================================================
package main

import (
	"errors"
	"flag"
	"fmt"
	"io"
	"log"
	"net"
	"os"
	"sync"
	"time"
)

var (
	ErrInvalidRequest = errors.New("invalid request")
	ErrInvalidAuth    = errors.New("invalid auth")
)

var (
	token = os.Getenv("NOEQ_TOKEN")
)

const (
	workerIdBits       = uint64(5)
	datacenterIdBits   = uint64(5)
	maxWorkerId        = int64(-1) ^ (int64(-1) << workerIdBits)
	maxDatacenterId    = int64(-1) ^ (int64(-1) << datacenterIdBits)
	sequenceBits       = uint64(12)
	workerIdShift      = sequenceBits
	datacenterIdShift  = sequenceBits + workerIdBits
	timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits
	sequenceMask       = int64(-1) ^ (int64(-1) << sequenceBits)

	// Tue, 21 Mar 2006 20:50:14.000 GMT
	twepoch = int64(1288834974657)
)

// Flags
var (
	wid   = flag.Int64("w", 0, "worker id")
	did   = flag.Int64("d", 0, "datacenter id")
	laddr = flag.String("l", "0.0.0.0:4444", "the address to listen on")
	lts   = flag.Int64("t", -1, "the last timestamp in milliseconds")
)

var (
	mu  sync.Mutex
	seq int64
)

func main() {
	parseFlags()
	acceptAndServe(mustListen())
}

func parseFlags() {
	flag.Parse()
	if *wid < 0 || *wid > maxWorkerId {
		log.Fatalf("worker id must be between 0 and %d", maxWorkerId)
	}

	if *did < 0 || *did > maxDatacenterId {
		log.Fatalf("datacenter id must be between 0 and %d", maxDatacenterId)
	}
}

func mustListen() net.Listener {
	l, err := net.Listen("tcp", *laddr)
	if err != nil {
		log.Fatal(err)
	}
	return l
}

func acceptAndServe(l net.Listener) {
	for {
		cn, err := l.Accept()
		if err != nil {
			log.Println(err)
		}

		go func() {
			err := serve(cn, cn)
			if err != io.EOF {
				log.Println(err)
			}
			cn.Close()
		}()
	}
}

func serve(r io.Reader, w io.Writer) error {
	if token != "" {
		err := auth(r)
		if err != nil {
			return err
		}
	}

	c := make([]byte, 1)
	for {
		// Wait for 1 byte request
		_, err := io.ReadFull(r, c)
		if err != nil {
			return err
		}

		n := uint(c[0])
		if n == 0 {
			// No authing at this point
			return ErrInvalidRequest
		}

		b := make([]byte, n*8)
		for i := uint(0); i < n; i++ {
			id, err := nextId()
			if err != nil {
				return err
			}

			off := i * 8
			b[off+0] = byte(id >> 56)
			b[off+1] = byte(id >> 48)
			b[off+2] = byte(id >> 40)
			b[off+3] = byte(id >> 32)
			b[off+4] = byte(id >> 24)
			b[off+5] = byte(id >> 16)
			b[off+6] = byte(id >> 8)
			b[off+7] = byte(id)
		}

		_, err = w.Write(b)
		if err != nil {
			return err
		}
	}

	panic("not reached")
}

func milliseconds() int64 {
	return time.Now().UnixNano() / 1e6
}

func nextId() (int64, error) {
	mu.Lock()
	defer mu.Unlock()

	ts := milliseconds()

	if ts < *lts {
		return 0, fmt.Errorf("time is moving backwards, waiting until %d\n", *lts)
	}

	if *lts == ts {
		seq = (seq + 1) & sequenceMask
		if seq == 0 {
			for ts <= *lts {
				ts = milliseconds()
			}
		}
	} else {
		seq = 0
	}

	*lts = ts

	id := ((ts - twepoch) << timestampLeftShift) |
		(*did << datacenterIdShift) |
		(*wid << workerIdShift) |
		seq

	return id, nil
}

func auth(r io.Reader) error {
	b := make([]byte, 2)
	_, err := io.ReadFull(r, b)
	if err != nil {
		return err
	}

	if b[0] != 0 {
		return ErrInvalidRequest
	}

	b = make([]byte, b[1])
	_, err = io.ReadFull(r, b)
	if err != nil {
		return err
	}

	if string(b) != token {
		return ErrInvalidAuth
	}

	return nil
}


================================================
FILE: main_test.go
================================================
package main

import (
	"bytes"
	"github.com/bmizerany/assert"
	"testing"
)

func TestServeZero(t *testing.T) {
	i, o := bytes.NewBuffer([]byte{0}), new(bytes.Buffer)
	serve(i, o)
	assert.Equal(t, 0, o.Len())
}

func TestServeMoreThanZero(t *testing.T) {
	i, o := bytes.NewBuffer([]byte{1}), new(bytes.Buffer)
	serve(i, o)
	assert.Equal(t, 8, o.Len())

	i, o = bytes.NewBuffer([]byte{2}), new(bytes.Buffer)
	serve(i, o)
	assert.Equal(t, 16, o.Len())

	i, o = bytes.NewBuffer([]byte{255}), new(bytes.Buffer)
	serve(i, o)
	assert.Equal(t, 255*8, o.Len())
}
Download .txt
gitextract_ldnxyfje/

├── .gitignore
├── Makefile
├── README.md
├── bench.sh
├── bench_test.go
├── coord-exec.sh
├── main.go
└── main_test.go
Download .txt
SYMBOL INDEX (32 symbols across 3 files)

FILE: bench_test.go
  type nopWriter (line 8) | type nopWriter struct
    method Write (line 10) | func (nw *nopWriter) Write(b []byte) (n int, err error) {
  function BenchmarkIdGeneration (line 14) | func BenchmarkIdGeneration(b *testing.B) {
  function BenchmarkServe01 (line 20) | func BenchmarkServe01(b *testing.B) {
  function BenchmarkServe02 (line 29) | func BenchmarkServe02(b *testing.B) {
  function BenchmarkServe03 (line 38) | func BenchmarkServe03(b *testing.B) {
  function BenchmarkServe05 (line 47) | func BenchmarkServe05(b *testing.B) {
  function BenchmarkServe08 (line 56) | func BenchmarkServe08(b *testing.B) {
  function BenchmarkServe13 (line 65) | func BenchmarkServe13(b *testing.B) {
  function BenchmarkServe21 (line 74) | func BenchmarkServe21(b *testing.B) {
  function BenchmarkServe34 (line 83) | func BenchmarkServe34(b *testing.B) {
  function BenchmarkServe55 (line 92) | func BenchmarkServe55(b *testing.B) {

FILE: main.go
  constant workerIdBits (line 25) | workerIdBits       = uint64(5)
  constant datacenterIdBits (line 26) | datacenterIdBits   = uint64(5)
  constant maxWorkerId (line 27) | maxWorkerId        = int64(-1) ^ (int64(-1) << workerIdBits)
  constant maxDatacenterId (line 28) | maxDatacenterId    = int64(-1) ^ (int64(-1) << datacenterIdBits)
  constant sequenceBits (line 29) | sequenceBits       = uint64(12)
  constant workerIdShift (line 30) | workerIdShift      = sequenceBits
  constant datacenterIdShift (line 31) | datacenterIdShift  = sequenceBits + workerIdBits
  constant timestampLeftShift (line 32) | timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits
  constant sequenceMask (line 33) | sequenceMask       = int64(-1) ^ (int64(-1) << sequenceBits)
  constant twepoch (line 36) | twepoch = int64(1288834974657)
  function main (line 52) | func main() {
  function parseFlags (line 57) | func parseFlags() {
  function mustListen (line 68) | func mustListen() net.Listener {
  function acceptAndServe (line 76) | func acceptAndServe(l net.Listener) {
  function serve (line 93) | func serve(r io.Reader, w io.Writer) error {
  function milliseconds (line 142) | func milliseconds() int64 {
  function nextId (line 146) | func nextId() (int64, error) {
  function auth (line 177) | func auth(r io.Reader) error {

FILE: main_test.go
  function TestServeZero (line 9) | func TestServeZero(t *testing.T) {
  function TestServeMoreThanZero (line 15) | func TestServeMoreThanZero(t *testing.T) {
Condensed preview — 8 files, each showing path, character count, and a content snippet. Download the .json file or copy for the full structured content (19K chars).
[
  {
    "path": ".gitignore",
    "chars": 41,
    "preview": "*.[568]\n[568].out\n_obj\n_test*\na.out\n*.gz\n"
  },
  {
    "path": "Makefile",
    "chars": 341,
    "preview": "# figure out what GOROOT is supposed to be\nGOROOT ?= $(shell printf 't:;@echo $$(GOROOT)\\n' | gomake -f -)\ninclude $(GOR"
  },
  {
    "path": "README.md",
    "chars": 11102,
    "preview": "# noeqd - A fault-tolerant network service for meaningful GUID generation\n\nBased on [snowflake][].\n\n## Motivation\n\nGUIDs"
  },
  {
    "path": "bench.sh",
    "chars": 45,
    "preview": "#!/bin/sh\ngotest -run=\"no tests\" -bench=\".*\"\n"
  },
  {
    "path": "bench_test.go",
    "chars": 1779,
    "preview": "package main\n\nimport (\n\t\"bytes\"\n\t\"testing\"\n)\n\ntype nopWriter struct{}\n\nfunc (nw *nopWriter) Write(b []byte) (n int, err "
  },
  {
    "path": "coord-exec.sh",
    "chars": 225,
    "preview": "#!/bin/sh\n\n#/ usage: exec.sh\n\ndid=$1\nwid=0\n\n[ -z \"$did\" ] && did=0\n\n_set() {\n  # TODO: use an id rather than 1\n  printf "
  },
  {
    "path": "main.go",
    "chars": 3355,
    "preview": "package main\n\nimport (\n\t\"errors\"\n\t\"flag\"\n\t\"fmt\"\n\t\"io\"\n\t\"log\"\n\t\"net\"\n\t\"os\"\n\t\"sync\"\n\t\"time\"\n)\n\nvar (\n\tErrInvalidRequest = "
  },
  {
    "path": "main_test.go",
    "chars": 555,
    "preview": "package main\n\nimport (\n\t\"bytes\"\n\t\"github.com/bmizerany/assert\"\n\t\"testing\"\n)\n\nfunc TestServeZero(t *testing.T) {\n\ti, o :="
  }
]

About this extraction

This page contains the full source code of the bmizerany/noeqd GitHub repository, extracted and formatted as plain text for AI agents and large language models (LLMs). The extraction includes 8 files (17.0 KB), approximately 5.2k tokens, and a symbol index with 32 extracted functions, classes, methods, constants, and types. Use this with OpenClaw, Claude, ChatGPT, Cursor, Windsurf, or any other AI tool that accepts text input. You can copy the full output to your clipboard or download it as a .txt file.

Extracted by GitExtract — free GitHub repo to text converter for AI. Built by Nikandr Surkov.

Copied to clipboard!