V8 Bindings for Go
The v8 bindings allow a user to execute javascript from within a go executable.
The bindings are tested to work with several recent v8 builds matching the
Chrome builds 54 - 60 (see the .travis.yml file for specific versions). For
example, Chrome 59 (dev branch) uses v8 5.9.211.4 when this was written.
Note that v8 releases match the Chrome release timeline:
Chrome 48 corresponds to v8 4.8.*, Chrome 49 matches v8 4.9.*. You can see
the table of current chrome and the associated v8 releases at:
http://omahaproxy.appspot.com/
Using a pre-compiled v8
v8 is very slow to compile, it's a large project. If you want to go that route, there are building instructions below.
Fortunately, there's a project that pre-builds v8 for various platforms. It's packaged as a ruby gem called libv8.
curl https://rubygems.org/downloads/libv8-6.3.292.48.1-x86_64-darwin-16.gem > libv8.gem
tar -xf libv8.gem
cd libv8-6.3.292.48.1-x86_64-darwin-16
tar -xzf data.tar.gz
ln -s $(pwd)/data/vendor/v8/include $GOPATH/src/github.com/augustoroman/v8/include
ln -s $(pwd)/data/vendor/v8/out/x64.release $GOPATH/src/github.com/augustoroman/v8/libv8
cd $GOPATH/src/github.com/augustoroman/v8
go test
Using docker (linux only)
For linux builds, you can use pre-built libraries or build your own.
Pre-built versions
To use a pre-built library, select the desired v8 version from https://hub.docker.com/r/augustoroman/v8-lib/tags/ and then run:
export V8_VERSION=6.7.77
docker pull augustoroman/v8-lib:$V8_VERSION
docker rm v8 ||:
docker run --name v8 augustoroman/v8-lib:$V8_VERSION
docker cp v8:/v8/include include/
docker cp v8:/v8/lib libv8/
Build your own via docker
This takes a lot longer, but is still easy:
export V8_VERSION=6.7.77
docker build --build-arg V8_VERSION=$V8_VERSION --tag augustoroman/v8-lib:$V8_VERSION docker-v8-lib/
and then extract the files as above:
docker rm v8 ||:
docker run --name v8 augustoroman/v8-lib:$V8_VERSION
docker cp v8:/v8/include include/
docker cp v8:/v8/lib libv8/
Building v8
Prep
You need to build v8 statically and place it in a location cgo knows about. This requires special tooling and a build directory. Using the official instructions as a guide, the general steps of this process are:
go get
the binding library (this library)- Create a v8 build directory
- Install depot tools
- Configure environment
- Download v8
- Build v8
- Copy or symlink files to the go library path
- Build the bindings
go get github.com/augustoroman/v8
export V8_GO=$GOPATH/src/github.com/augustoroman/v8
export V8_BUILD=$V8_GO/v8/build #or wherever you like
mkdir -p $V8_BUILD
cd $V8_BUILD
git clone https://chromium.googlesource.com/chromium/tools/depot_tools.git
export PATH=$PATH:$V8_BUILD/depot_tools
fetch v8 #pull down v8 (this will take some time)
cd v8
git checkout 6.7.77
gclient sync
Linux
./build/install-build-deps.sh #only needed once
gn gen out.gn/golib --args="strip_debug_info=true v8_use_external_startup_data=false v8_enable_i18n_support=false v8_enable_gdbjit=false v8_static_library=true symbol_level=0 v8_experimental_extra_library_files=[] v8_extra_library_files=[]"
ninja -C out.gn/golib
# go get some coffee
OSX
gn gen out.gn/golib --args="is_official_build=true strip_debug_info=true v8_use_external_startup_data=false v8_enable_i18n_support=false v8_enable_gdbjit=false v8_static_library=true symbol_level=0 v8_experimental_extra_library_files=[] v8_extra_library_files=[]"
ninja -C out.gn/golib
# go get some coffee
Symlinking
Now you can create symlinks so that cgo can associate the v8 binaries with the go library.
cd $V8_GO
./symlink.sh $V8_BUILD/v8
Verifying
You should be done! Try running go test
Reference
Also relevant is the v8 API release changes doc:
https://docs.google.com/document/d/1g8JFi8T_oAE_7uAri7Njtig7fKaPDfotU6huOa1alds/edit
Credits
This work is based off of several existing libraries: