Compare commits
65 Commits
cherry-pic
...
main
Author | SHA1 | Date |
---|---|---|
Jeff Carr | c9a7e84e5b | |
Jeff Carr | a4ee4aafbc | |
Jeff Carr | cf84056045 | |
Jeff Carr | 585a8dbbe5 | |
Jeff Carr | d5a0e0056f | |
Jeff Carr | 530f3618a8 | |
Roland Shoemaker | e65b1c188c | |
lhchavez | 4b14d29c20 | |
Sanskar Jaiswal | c1ec21d89c | |
Calin | 9db5de109c | |
lhchavez | 7bff4ca7ad | |
William Bain | eae00773cc | |
Patrick Steinhardt | e7d1b2b69f | |
Dylan Richardson | c598ea5718 | |
Aurélien | 5eca48cda9 | |
James Fargher | 1fcc9d8743 | |
Kirill | 5e35338d58 | |
Ignacio Taranto | 533c82f270 | |
Sunny | 6cea7a7a59 | |
Yashodhan Ghadge | 0e8009f00a | |
lhchavez | 6eae74c128 | |
lhchavez | 9b155184fe | |
lhchavez | c6da3b97a8 | |
lhchavez | c8ce59d4eb | |
lhchavez | 922f2f7487 | |
lhchavez | dcc9331226 | |
lhchavez | 018647fd48 | |
lhchavez | b78bde3d74 | |
lhchavez | 5def02a589 | |
lhchavez | 70e5e419cf | |
lhchavez | b983e1daeb | |
lhchavez | f1fa96c7b7 | |
lhchavez | dbe032c347 | |
lhchavez | 549706bb57 | |
lhchavez | 2077003fa5 | |
Gustav Westling | d4524761d9 | |
lhchavez | 15434610fe | |
lhchavez | be5a99a807 | |
lhchavez | fbaf9d1d1a | |
lhchavez | df7084d36a | |
Vladimir Buzuev | 1e2cb92b48 | |
Vladimir Buzuev | 0d7c8dadb4 | |
Vladimir Buzuev | a4d202ed7b | |
michael boulton | aeb22bcf7d | |
Suhaib Mujahid | f6c5753df8 | |
Byoungchan Lee | 2fd0495c43 | |
Byoungchan Lee | 73d97b9bbe | |
Byoungchan Lee | 07147a8ea8 | |
lhchavez | 4b2ac7c998 | |
lhchavez | 698ddfb4ac | |
lhchavez | 10c67474a8 | |
lhchavez | e28cce87c7 | |
lhchavez | abf02bc7d7 | |
lhchavez | 54afccfa0f | |
lhchavez | 5d8eaf7e65 | |
lhchavez | 137c05e802 | |
nmeum | 1fabe95fb7 | |
Hans Duedal | 7497529f70 | |
Segev Finer | 2bd574b6bd | |
Devendra | 2d639d8e49 | |
Suhaib Mujahid | ccbe4719bd | |
Suhaib Mujahid | b46ebfab8c | |
lhchavez | 77460dd7f0 | |
Suhaib Mujahid | c3664193f3 | |
Suhaib Mujahid | ad3ec3664d |
|
@ -1,53 +0,0 @@
|
|||
name: Backport to older releases
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
|
||||
jobs:
|
||||
|
||||
backport:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
branch: [ 'release-0.28', 'release-0.27' ]
|
||||
name: Backport change to branch ${{ matrix.branch }}
|
||||
|
||||
runs-on: ubuntu-20.04
|
||||
|
||||
steps:
|
||||
- name: Check out code
|
||||
uses: actions/checkout@v1
|
||||
with:
|
||||
fetch-depth: 0
|
||||
- name: Create a cherry-pick PR
|
||||
run: |
|
||||
if ! git diff --quiet HEAD^ HEAD -- vendor/libgit2; then
|
||||
echo '::warning::Skipping cherry-pick since it is a vendored libgit2 bump'
|
||||
exit 0
|
||||
fi
|
||||
|
||||
BRANCH_NAME="cherry-pick-${{ github.run_id }}-${{ matrix.branch }}"
|
||||
|
||||
# Setup usernames and authentication
|
||||
git config --global user.name "${{ github.actor }}"
|
||||
git config --global user.email "${{ github.actor }}@users.noreply.github.com"
|
||||
cat <<- EOF > $HOME/.netrc
|
||||
machine github.com
|
||||
login ${{ github.actor }}
|
||||
password ${{ secrets.GITHUB_TOKEN }}
|
||||
machine api.github.com
|
||||
login ${{ github.actor }}
|
||||
password ${{ secrets.GITHUB_TOKEN }}
|
||||
EOF
|
||||
chmod 600 $HOME/.netrc
|
||||
|
||||
# Create the cherry-pick commit and create the PR for it.
|
||||
git checkout "${{ matrix.branch }}"
|
||||
git switch -c "${BRANCH_NAME}"
|
||||
git cherry-pick -x "${{ github.sha }}"
|
||||
git push --set-upstream origin "${BRANCH_NAME}"
|
||||
GITHUB_TOKEN="${{ secrets.GITHUB_TOKEN }}" gh pr create \
|
||||
--base "${{ matrix.branch }}" \
|
||||
--title "$(git --no-pager show --format="%s" --no-patch HEAD)" \
|
||||
--body "$(git --no-pager show --format="%b" --no-patch HEAD)"
|
|
@ -1,133 +0,0 @@
|
|||
name: git2go CI
|
||||
on:
|
||||
pull_request:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
- release-*
|
||||
- v*
|
||||
|
||||
jobs:
|
||||
|
||||
build-legacy:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
go: [ '1.9', '1.10' ]
|
||||
name: Go ${{ matrix.go }}
|
||||
|
||||
runs-on: ubuntu-18.04
|
||||
|
||||
steps:
|
||||
- name: Set up Go
|
||||
uses: actions/setup-go@v1
|
||||
with:
|
||||
go-version: ${{ matrix.go }}
|
||||
id: go
|
||||
- name: Check out code into the GOPATH
|
||||
uses: actions/checkout@v1
|
||||
with:
|
||||
fetch-depth: 1
|
||||
path: src/github.com/${{ github.repository }}
|
||||
- name: Build
|
||||
env:
|
||||
GOPATH: /home/runner/work/git2go
|
||||
run: |
|
||||
git submodule update --init
|
||||
make build-libgit2-static
|
||||
go get -tags static -t github.com/${{ github.repository }}/...
|
||||
go build -tags static github.com/${{ github.repository }}/...
|
||||
- name: Test
|
||||
env:
|
||||
GOPATH: /home/runner/work/git2go
|
||||
run: make test-static
|
||||
|
||||
build-static:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
go: [ '1.11', '1.12', '1.13', '1.14', '1.15' ]
|
||||
name: Go ${{ matrix.go }}
|
||||
|
||||
runs-on: ubuntu-20.04
|
||||
|
||||
steps:
|
||||
- name: Set up Go
|
||||
uses: actions/setup-go@v1
|
||||
with:
|
||||
go-version: ${{ matrix.go }}
|
||||
id: go
|
||||
- name: Check out code into the Go module directory
|
||||
uses: actions/checkout@v1
|
||||
- name: Build
|
||||
run: |
|
||||
git submodule update --init
|
||||
make build-libgit2-static
|
||||
- name: Test
|
||||
run: make test-static
|
||||
|
||||
build-dynamic:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
name: Go (dynamic)
|
||||
|
||||
runs-on: ubuntu-20.04
|
||||
|
||||
steps:
|
||||
- name: Set up Go
|
||||
uses: actions/setup-go@v1
|
||||
with:
|
||||
go-version: '1.15'
|
||||
id: go
|
||||
- name: Check out code into the Go module directory
|
||||
uses: actions/checkout@v1
|
||||
- name: Build
|
||||
run: |
|
||||
git submodule update --init
|
||||
make build-libgit2-dynamic
|
||||
- name: Test
|
||||
run: make test-dynamic
|
||||
|
||||
build-system-dynamic:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
name: Go (system-wide, dynamic)
|
||||
|
||||
runs-on: ubuntu-20.04
|
||||
|
||||
steps:
|
||||
- name: Set up Go
|
||||
uses: actions/setup-go@v1
|
||||
with:
|
||||
go-version: '1.15'
|
||||
id: go
|
||||
- name: Check out code into the Go module directory
|
||||
uses: actions/checkout@v1
|
||||
- name: Build libgit2
|
||||
run: |
|
||||
git submodule update --init
|
||||
sudo ./script/build-libgit2.sh --dynamic --system
|
||||
- name: Test
|
||||
run: make test
|
||||
|
||||
build-system-static:
|
||||
strategy:
|
||||
fail-fast: false
|
||||
name: Go (system-wide, static)
|
||||
|
||||
runs-on: ubuntu-20.04
|
||||
|
||||
steps:
|
||||
- name: Set up Go
|
||||
uses: actions/setup-go@v1
|
||||
with:
|
||||
go-version: '1.15'
|
||||
id: go
|
||||
- name: Check out code into the Go module directory
|
||||
uses: actions/checkout@v1
|
||||
- name: Build libgit2
|
||||
run: |
|
||||
git submodule update --init
|
||||
sudo ./script/build-libgit2.sh --static --system
|
||||
- name: Test
|
||||
run: go test --count=1 --tags "static,system_libgit2" ./...
|
|
@ -1,28 +0,0 @@
|
|||
name: Tag new releases
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
- release-*
|
||||
|
||||
jobs:
|
||||
|
||||
tag-release:
|
||||
name: Bump tag in ${{ github.ref }}
|
||||
|
||||
runs-on: ubuntu-20.04
|
||||
|
||||
steps:
|
||||
- name: Check out code
|
||||
uses: actions/checkout@v1
|
||||
with:
|
||||
fetch-depth: 0
|
||||
- name: Bump version and push tag
|
||||
id: bump-version
|
||||
uses: anothrNick/github-tag-action@9aaabdb5e989894e95288328d8b17a6347217ae3
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
WITH_V: true
|
||||
DEFAULT_BUMP: patch
|
||||
TAG_CONTEXT: branch
|
||||
RELEASE_BRANCHES: .*
|
|
@ -1,2 +1,4 @@
|
|||
/static-build/
|
||||
/dynamic-build/
|
||||
|
||||
go.*
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
[submodule "vendor/libgit2"]
|
||||
path = vendor/libgit2
|
||||
url = https://github.com/libgit2/libgit2
|
20
.travis.yml
20
.travis.yml
|
@ -1,20 +0,0 @@
|
|||
language: go
|
||||
|
||||
go:
|
||||
- tip
|
||||
|
||||
install:
|
||||
- make build-libgit2-static
|
||||
- go get --tags "static" ./...
|
||||
|
||||
script:
|
||||
- make test-static
|
||||
|
||||
git:
|
||||
submodules: true
|
||||
|
||||
branches:
|
||||
only:
|
||||
- master
|
||||
- /v\d+/
|
||||
- /release-.*/
|
|
@ -1,16 +1,17 @@
|
|||
//go:build static && !system_libgit2
|
||||
// +build static,!system_libgit2
|
||||
|
||||
package git
|
||||
|
||||
/*
|
||||
#cgo windows CFLAGS: -I${SRCDIR}/static-build/install/include/
|
||||
#cgo windows LDFLAGS: -L${SRCDIR}/static-build/install/lib/ -lgit2 -lwinhttp
|
||||
#cgo windows LDFLAGS: -L${SRCDIR}/static-build/install/lib/ -lgit2 -lwinhttp -lws2_32 -lole32 -lrpcrt4 -lcrypt32
|
||||
#cgo !windows pkg-config: --static ${SRCDIR}/static-build/install/lib/pkgconfig/libgit2.pc
|
||||
#cgo CFLAGS: -DLIBGIT2_STATIC
|
||||
#include <git2.h>
|
||||
|
||||
#if LIBGIT2_VER_MAJOR != 1 || LIBGIT2_VER_MINOR != 0
|
||||
# error "Invalid libgit2 version; this git2go supports libgit2 v1.0"
|
||||
#if LIBGIT2_VER_MAJOR != 1 || LIBGIT2_VER_MINOR < 5 || LIBGIT2_VER_MINOR > 5
|
||||
# error "Invalid libgit2 version; this libgit2 supports libgit2 between v1.5.0 and v1.5.0"
|
||||
#endif
|
||||
*/
|
||||
import "C"
|
|
@ -0,0 +1,14 @@
|
|||
//go:build !static
|
||||
// +build !static
|
||||
|
||||
package git
|
||||
|
||||
/*
|
||||
#cgo pkg-config: libgit2
|
||||
#cgo CFLAGS: -DLIBGIT2_DYNAMIC -I/opt/libgit2/include
|
||||
#cgo LDFLAGS: -L/opt/libgit2 -lgit2
|
||||
#include <git2.h>
|
||||
|
||||
|
||||
*/
|
||||
import "C"
|
|
@ -0,0 +1,15 @@
|
|||
//go:build static && system_libgit2
|
||||
// +build static,system_libgit2
|
||||
|
||||
package git
|
||||
|
||||
/*
|
||||
#cgo pkg-config: libgit2 --static
|
||||
#cgo CFLAGS: -DLIBGIT2_STATIC
|
||||
#include <git2.h>
|
||||
|
||||
#if LIBGIT2_VER_MAJOR != 1 || LIBGIT2_VER_MINOR < 5 || LIBGIT2_VER_MINOR > 5
|
||||
# error "Invalid libgit2 version; this libgit2 supports libgit2 between v1.5.0 and v1.5.0"
|
||||
#endif
|
||||
*/
|
||||
import "C"
|
2
LICENSE
2
LICENSE
|
@ -1,6 +1,6 @@
|
|||
The MIT License
|
||||
|
||||
Copyright (c) 2013 The git2go contributors
|
||||
Copyright (c) 2013 The libgit2 contributors
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
|
|
25
Makefile
25
Makefile
|
@ -1,15 +1,32 @@
|
|||
default: test
|
||||
TEST_ARGS ?= --count=1
|
||||
|
||||
PKG_CONFIG_PATH=/opt/libgit2/
|
||||
|
||||
default: goimports test
|
||||
|
||||
goimports:
|
||||
goimports -w *.go
|
||||
|
||||
generate: static-build/install/lib/libgit2.a
|
||||
go generate --tags "static" ./...
|
||||
|
||||
# System library
|
||||
# ==============
|
||||
# This uses whatever version of libgit2 can be found in the system.
|
||||
test:
|
||||
-go-mod-clean # go install go.wit.com/apps/go-mod-clean@latest
|
||||
go run script/check-MakeGitError-thread-lock.go
|
||||
go test --count=1 ./...
|
||||
PKG_CONFIG_PATH=/opt/libgit2/ go test -v -x $(TEST_ARGS) ./...
|
||||
|
||||
add-remote:
|
||||
git remote add git2go https://github.com/libgit2/git2go.git
|
||||
|
||||
install:
|
||||
go install ./...
|
||||
|
||||
clean:
|
||||
rm go.*
|
||||
|
||||
# Bundled dynamic library
|
||||
# =======================
|
||||
# In order to avoid having to manipulate `git_dynamic.go`, which would prevent
|
||||
|
@ -28,7 +45,7 @@ test-dynamic: dynamic-build/install/lib/libgit2.so
|
|||
go run script/check-MakeGitError-thread-lock.go
|
||||
PKG_CONFIG_PATH=dynamic-build/install/lib/pkgconfig \
|
||||
LD_LIBRARY_PATH=dynamic-build/install/lib \
|
||||
go test --count=1 ./...
|
||||
go test $(TEST_ARGS) ./...
|
||||
|
||||
install-dynamic: dynamic-build/install/lib/libgit2.so
|
||||
PKG_CONFIG_PATH=dynamic-build/install/lib/pkgconfig \
|
||||
|
@ -47,7 +64,7 @@ static-build/install/lib/libgit2.a:
|
|||
|
||||
test-static: static-build/install/lib/libgit2.a
|
||||
go run script/check-MakeGitError-thread-lock.go
|
||||
go test --count=1 --tags "static" ./...
|
||||
go test --tags "static" $(TEST_ARGS) ./...
|
||||
|
||||
install-static: static-build/install/lib/libgit2.a
|
||||
go install --tags "static" ./...
|
||||
|
|
75
README.md
75
README.md
|
@ -1,92 +1,53 @@
|
|||
git2go
|
||||
GO libgit2
|
||||
======
|
||||
[![GoDoc](https://godoc.org/github.com/libgit2/git2go?status.svg)](http://godoc.org/github.com/libgit2/git2go) [![Build Status](https://travis-ci.org/libgit2/git2go.svg?branch=master)](https://travis-ci.org/libgit2/git2go)
|
||||
[![GoDoc](https://godoc.org/go.wit.com/lib/libgit2?status.svg)](http://godoc.org/go.wit.com/lib/libgit2) [![Build Status](https://travis-ci.org/libgit2/libgit2.svg?branch=main)](https://travis-ci.org/libgit2/libgit2)
|
||||
|
||||
Go bindings for [libgit2](http://libgit2.github.com/).
|
||||
|
||||
### Updated 2024/12/16
|
||||
|
||||
### Which Go version to use
|
||||
|
||||
Due to the fact that Go 1.11 module versions have semantic meaning and don't necessarily align with libgit2's release schedule, please consult the following table for a mapping between libgit2 and git2go module versions:
|
||||
|
||||
| libgit2 | git2go |
|
||||
|---------|---------------|
|
||||
| master | (will be v31) |
|
||||
| 1.0 | v30 |
|
||||
| 0.99 | v29 |
|
||||
| 0.28 | v28 |
|
||||
| 0.27 | v27 |
|
||||
|
||||
You can import them in your project with the version's major number as a suffix. For example, if you have libgit2 v1.0 installed, you'd import git2go v30 with
|
||||
* This package is updated to work against libgit2 version 1.8 on Debian sid
|
||||
* There is one line commented out which needs to be fixed in remote.go
|
||||
* some of the tests seem to run
|
||||
|
||||
```sh
|
||||
go get github.com/libgit2/git2go/v30
|
||||
```
|
||||
```go
|
||||
import "github.com/libgit2/git2go/v30"
|
||||
go install go.wit.com/apps/go-clone@latest
|
||||
go install go.wit.com/apps/go-mod-clean@latest
|
||||
go-clone --recusive go.wit.com/lib/libgit2
|
||||
```
|
||||
|
||||
which will ensure there are no sudden changes to the API.
|
||||
|
||||
The `master` branch follows the tip of libgit2 itself (with some lag) and as such has no guarantees on the stability of libgit2's API. Thus this only supports statically linking against libgit2.
|
||||
|
||||
### Which branch to send Pull requests to
|
||||
|
||||
If there's something version-specific that you'd want to contribute to, you can send them to the `release-${MAJOR}.${MINOR}` branches, which follow libgit2's releases.
|
||||
TODO: not sure yet
|
||||
|
||||
Installing
|
||||
----------
|
||||
|
||||
This project wraps the functionality provided by libgit2. It thus needs it in order to perform the work.
|
||||
|
||||
This project wraps the functionality provided by libgit2. If you're using a versioned branch, install it to your system via your system's package manager and then install git2go.
|
||||
This project wraps the functionality provided by libgit2. If you're using a versioned branch, install it to your system via your system's package manager and then install libgit2.
|
||||
|
||||
|
||||
### Versioned branch, dynamic linking
|
||||
|
||||
When linking dynamically against a released version of libgit2, install it via your system's package manager. CGo will take care of finding its pkg-config file and set up the linking. Import via Go modules, e.g. to work against libgit2 v1.0
|
||||
When linking dynamically against a released version of libgit2, install it via your system's package manager. CGo will take care of finding its pkg-config file and set up the linking. Import via Go modules, e.g. to work against libgit2 v1.2
|
||||
|
||||
```go
|
||||
import "github.com/libgit2/git2go/v30"
|
||||
goimports -w *.go
|
||||
```
|
||||
|
||||
### Versioned branch, static linking
|
||||
|
||||
Follow the instructions for [Versioned branch, dynamic linking](#versioned-branch-dynamic-linking), but pass the `-tags static,system_libgit2` flag to all `go` commands that build any binaries. For instance:
|
||||
|
||||
go build -tags static,system_libgit2 github.com/my/project/...
|
||||
go test -tags static,system_libgit2 github.com/my/project/...
|
||||
go install -tags static,system_libgit2 github.com/my/project/...
|
||||
|
||||
### Master branch, or vendored static linking
|
||||
|
||||
If using `master` or building a branch with the vendored libgit2 statically, we need to build libgit2 first. In order to build it, you need `cmake`, `pkg-config` and a C compiler. You will also need the development packages for OpenSSL (outside of Windows or macOS) and LibSSH2 installed if you want libgit2 to support HTTPS and SSH respectively. Note that even if libgit2 is included in the resulting binary, its dependencies will not be.
|
||||
|
||||
Run `go get -d github.com/libgit2/git2go` to download the code and go to your `$GOPATH/src/github.com/libgit2/git2go` directory. From there, we need to build the C code and put it into the resulting go binary.
|
||||
|
||||
git submodule update --init # get libgit2
|
||||
make install-static
|
||||
|
||||
will compile libgit2, link it into git2go and install it. The `master` branch is set up to follow the specific libgit2 version that is vendored, so trying dynamic linking may or may not work depending on the exact versions involved.
|
||||
|
||||
In order to let Go pass the correct flags to `pkg-config`, `-tags static` needs to be passed to all `go` commands that build any binaries. For instance:
|
||||
|
||||
go build -tags static github.com/my/project/...
|
||||
go test -tags static github.com/my/project/...
|
||||
go install -tags static github.com/my/project/...
|
||||
|
||||
One thing to take into account is that since Go expects the `pkg-config` file to be within the same directory where `make install-static` was called, so the `go.mod` file may need to have a [`replace` directive](https://github.com/golang/go/wiki/Modules#when-should-i-use-the-replace-directive) so that the correct setup is achieved. So if `git2go` is checked out at `$GOPATH/src/github.com/libgit2/git2go` and your project at `$GOPATH/src/github.com/my/project`, the `go.mod` file of `github.com/my/project` might need to have a line like
|
||||
|
||||
replace github.com/libgit2/git2go/v30 ../../libgit2/git2go
|
||||
|
||||
Parallelism and network operations
|
||||
----------------------------------
|
||||
|
||||
libgit2 may use OpenSSL and LibSSH2 for performing encrypted network connections. For now, git2go asks libgit2 to set locking for OpenSSL. This makes HTTPS connections thread-safe, but it is fragile and will likely stop doing it soon. This may also make SSH connections thread-safe if your copy of libssh2 is linked against OpenSSL. Check libgit2's `THREADSAFE.md` for more information.
|
||||
libgit2 may use OpenSSL and LibSSH2 for performing encrypted network connections. For now, libgit2 asks libgit2 to set locking for OpenSSL. This makes HTTPS connections thread-safe, but it is fragile and will likely stop doing it soon. This may also make SSH connections thread-safe if your copy of libssh2 is linked against OpenSSL. Check libgit2's `THREADSAFE.md` for more information.
|
||||
|
||||
Running the tests
|
||||
-----------------
|
||||
|
||||
For the stable version, `go test` will work as usual. For the `master` branch, similarly to installing, running the tests requires building a local libgit2 library, so the Makefile provides a wrapper that makes sure it's built
|
||||
For the stable version, `go test` will work as usual. For the `main` branch, similarly to installing, running the tests requires building a local libgit2 library, so the Makefile provides a wrapper that makes sure it's built
|
||||
|
||||
make test-static
|
||||
|
||||
|
@ -103,6 +64,6 @@ M to the I to the T. See the LICENSE file if you've never seen an MIT license be
|
|||
Authors
|
||||
-------
|
||||
|
||||
- Carlos Martín (@carlosmn)
|
||||
- Vicent Martí (@vmg)
|
||||
- Carlos Martín (github@carlosmn)
|
||||
- Vicent Martí (github@vmg)
|
||||
|
||||
|
|
3
blame.go
3
blame.go
|
@ -47,6 +47,8 @@ const (
|
|||
BlameTrackCopiesSameCommitCopies BlameOptionsFlag = C.GIT_BLAME_TRACK_COPIES_SAME_COMMIT_COPIES
|
||||
BlameTrackCopiesAnyCommitCopies BlameOptionsFlag = C.GIT_BLAME_TRACK_COPIES_ANY_COMMIT_COPIES
|
||||
BlameFirstParent BlameOptionsFlag = C.GIT_BLAME_FIRST_PARENT
|
||||
BlameUseMailmap BlameOptionsFlag = C.GIT_BLAME_USE_MAILMAP
|
||||
BlameIgnoreWhitespace BlameOptionsFlag = C.GIT_BLAME_IGNORE_WHITESPACE
|
||||
)
|
||||
|
||||
func (v *Repository) BlameFile(path string, opts *BlameOptions) (*Blame, error) {
|
||||
|
@ -85,6 +87,7 @@ func (v *Repository) BlameFile(path string, opts *BlameOptions) (*Blame, error)
|
|||
}
|
||||
|
||||
type Blame struct {
|
||||
doNotCompare
|
||||
ptr *C.git_blame
|
||||
}
|
||||
|
||||
|
|
35
blob.go
35
blob.go
|
@ -9,13 +9,13 @@ void _go_git_writestream_free(git_writestream *stream);
|
|||
*/
|
||||
import "C"
|
||||
import (
|
||||
"io"
|
||||
"reflect"
|
||||
"runtime"
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
type Blob struct {
|
||||
doNotCompare
|
||||
Object
|
||||
cast_ptr *C.git_blob
|
||||
}
|
||||
|
@ -68,7 +68,7 @@ func (repo *Repository) CreateBlobFromBuffer(data []byte) (*Oid, error) {
|
|||
size = C.size_t(0)
|
||||
}
|
||||
|
||||
ecode := C.git_blob_create_frombuffer(&id, repo.ptr, unsafe.Pointer(&data[0]), size)
|
||||
ecode := C.git_blob_create_from_buffer(&id, repo.ptr, unsafe.Pointer(&data[0]), size)
|
||||
runtime.KeepAlive(repo)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
|
@ -76,32 +76,6 @@ func (repo *Repository) CreateBlobFromBuffer(data []byte) (*Oid, error) {
|
|||
return newOidFromC(&id), nil
|
||||
}
|
||||
|
||||
type BlobChunkCallback func(maxLen int) ([]byte, error)
|
||||
|
||||
type BlobCallbackData struct {
|
||||
Callback BlobChunkCallback
|
||||
Error error
|
||||
}
|
||||
|
||||
//export blobChunkCb
|
||||
func blobChunkCb(buffer *C.char, maxLen C.size_t, handle unsafe.Pointer) int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*BlobCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve blob callback data")
|
||||
}
|
||||
|
||||
goBuf, err := data.Callback(int(maxLen))
|
||||
if err == io.EOF {
|
||||
return 0
|
||||
} else if err != nil {
|
||||
data.Error = err
|
||||
return -1
|
||||
}
|
||||
C.memcpy(unsafe.Pointer(buffer), unsafe.Pointer(&goBuf[0]), C.size_t(len(goBuf)))
|
||||
return len(goBuf)
|
||||
}
|
||||
|
||||
func (repo *Repository) CreateFromStream(hintPath string) (*BlobWriteStream, error) {
|
||||
var chintPath *C.char = nil
|
||||
var stream *C.git_writestream
|
||||
|
@ -114,7 +88,7 @@ func (repo *Repository) CreateFromStream(hintPath string) (*BlobWriteStream, err
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_blob_create_fromstream(&stream, repo.ptr, chintPath)
|
||||
ecode := C.git_blob_create_from_stream(&stream, repo.ptr, chintPath)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
|
@ -123,6 +97,7 @@ func (repo *Repository) CreateFromStream(hintPath string) (*BlobWriteStream, err
|
|||
}
|
||||
|
||||
type BlobWriteStream struct {
|
||||
doNotCompare
|
||||
ptr *C.git_writestream
|
||||
repo *Repository
|
||||
}
|
||||
|
@ -166,7 +141,7 @@ func (stream *BlobWriteStream) Commit() (*Oid, error) {
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_blob_create_fromstream_commit(&oid, stream.ptr)
|
||||
ecode := C.git_blob_create_from_stream_commit(&oid, stream.ptr)
|
||||
runtime.KeepAlive(stream)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
|
|
|
@ -19,6 +19,7 @@ const (
|
|||
)
|
||||
|
||||
type Branch struct {
|
||||
doNotCompare
|
||||
*Reference
|
||||
}
|
||||
|
||||
|
@ -27,6 +28,7 @@ func (r *Reference) Branch() *Branch {
|
|||
}
|
||||
|
||||
type BranchIterator struct {
|
||||
doNotCompare
|
||||
ptr *C.git_branch_iterator
|
||||
repo *Repository
|
||||
}
|
||||
|
@ -73,7 +75,7 @@ func (i *BranchIterator) ForEach(f BranchIteratorFunc) error {
|
|||
}
|
||||
}
|
||||
|
||||
if err != nil && IsErrorCode(err, ErrIterOver) {
|
||||
if err != nil && IsErrorCode(err, ErrorCodeIterOver) {
|
||||
return nil
|
||||
}
|
||||
|
||||
|
|
|
@ -22,7 +22,7 @@ func TestBranchIterator(t *testing.T) {
|
|||
t.Fatalf("expected BranchLocal, not %v", t)
|
||||
}
|
||||
b, bt, err = i.Next()
|
||||
if !IsErrorCode(err, ErrIterOver) {
|
||||
if !IsErrorCode(err, ErrorCodeIterOver) {
|
||||
t.Fatal("expected iterover")
|
||||
}
|
||||
}
|
||||
|
@ -49,7 +49,7 @@ func TestBranchIteratorEach(t *testing.T) {
|
|||
}
|
||||
|
||||
err = i.ForEach(f)
|
||||
if err != nil && !IsErrorCode(err, ErrIterOver) {
|
||||
if err != nil && !IsErrorCode(err, ErrorCodeIterOver) {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
|
|
174
checkout.go
174
checkout.go
|
@ -3,7 +3,7 @@ package git
|
|||
/*
|
||||
#include <git2.h>
|
||||
|
||||
extern void _go_git_populate_checkout_cb(git_checkout_options *opts);
|
||||
extern void _go_git_populate_checkout_callbacks(git_checkout_options *opts);
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
|
@ -48,10 +48,10 @@ const (
|
|||
CheckoutUpdateSubmodulesIfChanged CheckoutStrategy = C.GIT_CHECKOUT_UPDATE_SUBMODULES_IF_CHANGED // Recursively checkout submodules if HEAD moved in super repo (NOT IMPLEMENTED)
|
||||
)
|
||||
|
||||
type CheckoutNotifyCallback func(why CheckoutNotifyType, path string, baseline, target, workdir DiffFile) ErrorCode
|
||||
type CheckoutProgressCallback func(path string, completed, total uint) ErrorCode
|
||||
type CheckoutNotifyCallback func(why CheckoutNotifyType, path string, baseline, target, workdir DiffFile) error
|
||||
type CheckoutProgressCallback func(path string, completed, total uint)
|
||||
|
||||
type CheckoutOpts struct {
|
||||
type CheckoutOptions struct {
|
||||
Strategy CheckoutStrategy // Default will be a dry run
|
||||
DisableFilters bool // Don't apply filters like CRLF conversion
|
||||
DirMode os.FileMode // Default is 0755
|
||||
|
@ -65,19 +65,20 @@ type CheckoutOpts struct {
|
|||
Baseline *Tree
|
||||
}
|
||||
|
||||
func checkoutOptionsFromC(c *C.git_checkout_options) CheckoutOpts {
|
||||
opts := CheckoutOpts{}
|
||||
opts.Strategy = CheckoutStrategy(c.checkout_strategy)
|
||||
opts.DisableFilters = c.disable_filters != 0
|
||||
opts.DirMode = os.FileMode(c.dir_mode)
|
||||
opts.FileMode = os.FileMode(c.file_mode)
|
||||
opts.FileOpenFlags = int(c.file_open_flags)
|
||||
opts.NotifyFlags = CheckoutNotifyType(c.notify_flags)
|
||||
func checkoutOptionsFromC(c *C.git_checkout_options) CheckoutOptions {
|
||||
opts := CheckoutOptions{
|
||||
Strategy: CheckoutStrategy(c.checkout_strategy),
|
||||
DisableFilters: c.disable_filters != 0,
|
||||
DirMode: os.FileMode(c.dir_mode),
|
||||
FileMode: os.FileMode(c.file_mode),
|
||||
FileOpenFlags: int(c.file_open_flags),
|
||||
NotifyFlags: CheckoutNotifyType(c.notify_flags),
|
||||
}
|
||||
if c.notify_payload != nil {
|
||||
opts.NotifyCallback = pointerHandles.Get(c.notify_payload).(*CheckoutOpts).NotifyCallback
|
||||
opts.NotifyCallback = pointerHandles.Get(c.notify_payload).(*checkoutCallbackData).options.NotifyCallback
|
||||
}
|
||||
if c.progress_payload != nil {
|
||||
opts.ProgressCallback = pointerHandles.Get(c.progress_payload).(*CheckoutOpts).ProgressCallback
|
||||
opts.ProgressCallback = pointerHandles.Get(c.progress_payload).(*checkoutCallbackData).options.ProgressCallback
|
||||
}
|
||||
if c.target_directory != nil {
|
||||
opts.TargetDirectory = C.GoString(c.target_directory)
|
||||
|
@ -85,19 +86,19 @@ func checkoutOptionsFromC(c *C.git_checkout_options) CheckoutOpts {
|
|||
return opts
|
||||
}
|
||||
|
||||
func (opts *CheckoutOpts) toC() *C.git_checkout_options {
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
c := C.git_checkout_options{}
|
||||
populateCheckoutOpts(&c, opts)
|
||||
return &c
|
||||
type checkoutCallbackData struct {
|
||||
options *CheckoutOptions
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export checkoutNotifyCallback
|
||||
func checkoutNotifyCallback(why C.git_checkout_notify_t, cpath *C.char, cbaseline, ctarget, cworkdir, data unsafe.Pointer) int {
|
||||
if data == nil {
|
||||
return 0
|
||||
func checkoutNotifyCallback(
|
||||
why C.git_checkout_notify_t,
|
||||
cpath *C.char,
|
||||
cbaseline, ctarget, cworkdir, handle unsafe.Pointer,
|
||||
) C.int {
|
||||
if handle == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
path := C.GoString(cpath)
|
||||
var baseline, target, workdir DiffFile
|
||||
|
@ -110,85 +111,105 @@ func checkoutNotifyCallback(why C.git_checkout_notify_t, cpath *C.char, cbaselin
|
|||
if cworkdir != nil {
|
||||
workdir = diffFileFromC((*C.git_diff_file)(cworkdir))
|
||||
}
|
||||
opts := pointerHandles.Get(data).(*CheckoutOpts)
|
||||
if opts.NotifyCallback == nil {
|
||||
return 0
|
||||
data := pointerHandles.Get(handle).(*checkoutCallbackData)
|
||||
if data.options.NotifyCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
return int(opts.NotifyCallback(CheckoutNotifyType(why), path, baseline, target, workdir))
|
||||
err := data.options.NotifyCallback(CheckoutNotifyType(why), path, baseline, target, workdir)
|
||||
if err != nil {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export checkoutProgressCallback
|
||||
func checkoutProgressCallback(path *C.char, completed_steps, total_steps C.size_t, data unsafe.Pointer) int {
|
||||
opts := pointerHandles.Get(data).(*CheckoutOpts)
|
||||
if opts.ProgressCallback == nil {
|
||||
return 0
|
||||
func checkoutProgressCallback(
|
||||
path *C.char,
|
||||
completed_steps, total_steps C.size_t,
|
||||
handle unsafe.Pointer,
|
||||
) {
|
||||
data := pointerHandles.Get(handle).(*checkoutCallbackData)
|
||||
if data.options.ProgressCallback == nil {
|
||||
return
|
||||
}
|
||||
return int(opts.ProgressCallback(C.GoString(path), uint(completed_steps), uint(total_steps)))
|
||||
data.options.ProgressCallback(C.GoString(path), uint(completed_steps), uint(total_steps))
|
||||
}
|
||||
|
||||
// Convert the CheckoutOpts struct to the corresponding
|
||||
// C-struct. Returns a pointer to ptr, or nil if opts is nil, in order
|
||||
// to help with what to pass.
|
||||
func populateCheckoutOpts(ptr *C.git_checkout_options, opts *CheckoutOpts) *C.git_checkout_options {
|
||||
// populateCheckoutOptions populates the provided C-struct with the contents of
|
||||
// the provided CheckoutOptions struct. Returns copts, or nil if opts is nil,
|
||||
// in order to help with what to pass.
|
||||
func populateCheckoutOptions(copts *C.git_checkout_options, opts *CheckoutOptions, errorTarget *error) *C.git_checkout_options {
|
||||
C.git_checkout_options_init(copts, C.GIT_CHECKOUT_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
C.git_checkout_options_init(ptr, 1)
|
||||
ptr.checkout_strategy = C.uint(opts.Strategy)
|
||||
ptr.disable_filters = cbool(opts.DisableFilters)
|
||||
ptr.dir_mode = C.uint(opts.DirMode.Perm())
|
||||
ptr.file_mode = C.uint(opts.FileMode.Perm())
|
||||
ptr.notify_flags = C.uint(opts.NotifyFlags)
|
||||
copts.checkout_strategy = C.uint(opts.Strategy)
|
||||
copts.disable_filters = cbool(opts.DisableFilters)
|
||||
copts.dir_mode = C.uint(opts.DirMode.Perm())
|
||||
copts.file_mode = C.uint(opts.FileMode.Perm())
|
||||
copts.notify_flags = C.uint(opts.NotifyFlags)
|
||||
if opts.NotifyCallback != nil || opts.ProgressCallback != nil {
|
||||
C._go_git_populate_checkout_cb(ptr)
|
||||
}
|
||||
payload := pointerHandles.Track(opts)
|
||||
if opts.NotifyCallback != nil {
|
||||
ptr.notify_payload = payload
|
||||
}
|
||||
if opts.ProgressCallback != nil {
|
||||
ptr.progress_payload = payload
|
||||
C._go_git_populate_checkout_callbacks(copts)
|
||||
data := &checkoutCallbackData{
|
||||
options: opts,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
payload := pointerHandles.Track(data)
|
||||
if opts.NotifyCallback != nil {
|
||||
copts.notify_payload = payload
|
||||
}
|
||||
if opts.ProgressCallback != nil {
|
||||
copts.progress_payload = payload
|
||||
}
|
||||
}
|
||||
if opts.TargetDirectory != "" {
|
||||
ptr.target_directory = C.CString(opts.TargetDirectory)
|
||||
copts.target_directory = C.CString(opts.TargetDirectory)
|
||||
}
|
||||
if len(opts.Paths) > 0 {
|
||||
ptr.paths.strings = makeCStringsFromStrings(opts.Paths)
|
||||
ptr.paths.count = C.size_t(len(opts.Paths))
|
||||
copts.paths.strings = makeCStringsFromStrings(opts.Paths)
|
||||
copts.paths.count = C.size_t(len(opts.Paths))
|
||||
}
|
||||
|
||||
if opts.Baseline != nil {
|
||||
ptr.baseline = opts.Baseline.cast_ptr
|
||||
copts.baseline = opts.Baseline.cast_ptr
|
||||
}
|
||||
|
||||
return ptr
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeCheckoutOpts(ptr *C.git_checkout_options) {
|
||||
if ptr == nil {
|
||||
func freeCheckoutOptions(copts *C.git_checkout_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
C.free(unsafe.Pointer(ptr.target_directory))
|
||||
if ptr.paths.count > 0 {
|
||||
freeStrarray(&ptr.paths)
|
||||
C.free(unsafe.Pointer(copts.target_directory))
|
||||
if copts.paths.count > 0 {
|
||||
freeStrarray(&copts.paths)
|
||||
}
|
||||
if ptr.notify_payload != nil {
|
||||
pointerHandles.Untrack(ptr.notify_payload)
|
||||
if copts.notify_payload != nil {
|
||||
pointerHandles.Untrack(copts.notify_payload)
|
||||
} else if copts.progress_payload != nil {
|
||||
pointerHandles.Untrack(copts.progress_payload)
|
||||
}
|
||||
}
|
||||
|
||||
// Updates files in the index and the working tree to match the content of
|
||||
// the commit pointed at by HEAD. opts may be nil.
|
||||
func (v *Repository) CheckoutHead(opts *CheckoutOpts) error {
|
||||
func (v *Repository) CheckoutHead(opts *CheckoutOptions) error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cOpts := opts.toC()
|
||||
defer freeCheckoutOpts(cOpts)
|
||||
var err error
|
||||
cOpts := populateCheckoutOptions(&C.git_checkout_options{}, opts, &err)
|
||||
defer freeCheckoutOptions(cOpts)
|
||||
|
||||
ret := C.git_checkout_head(v.ptr, cOpts)
|
||||
runtime.KeepAlive(v)
|
||||
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
@ -199,7 +220,7 @@ func (v *Repository) CheckoutHead(opts *CheckoutOpts) error {
|
|||
// Updates files in the working tree to match the content of the given
|
||||
// index. If index is nil, the repository's index will be used. opts
|
||||
// may be nil.
|
||||
func (v *Repository) CheckoutIndex(index *Index, opts *CheckoutOpts) error {
|
||||
func (v *Repository) CheckoutIndex(index *Index, opts *CheckoutOptions) error {
|
||||
var iptr *C.git_index = nil
|
||||
if index != nil {
|
||||
iptr = index.ptr
|
||||
|
@ -208,11 +229,15 @@ func (v *Repository) CheckoutIndex(index *Index, opts *CheckoutOpts) error {
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cOpts := opts.toC()
|
||||
defer freeCheckoutOpts(cOpts)
|
||||
var err error
|
||||
cOpts := populateCheckoutOptions(&C.git_checkout_options{}, opts, &err)
|
||||
defer freeCheckoutOptions(cOpts)
|
||||
|
||||
ret := C.git_checkout_index(v.ptr, iptr, cOpts)
|
||||
runtime.KeepAlive(v)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
@ -220,15 +245,20 @@ func (v *Repository) CheckoutIndex(index *Index, opts *CheckoutOpts) error {
|
|||
return nil
|
||||
}
|
||||
|
||||
func (v *Repository) CheckoutTree(tree *Tree, opts *CheckoutOpts) error {
|
||||
func (v *Repository) CheckoutTree(tree *Tree, opts *CheckoutOptions) error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cOpts := opts.toC()
|
||||
defer freeCheckoutOpts(cOpts)
|
||||
var err error
|
||||
cOpts := populateCheckoutOptions(&C.git_checkout_options{}, opts, &err)
|
||||
defer freeCheckoutOptions(cOpts)
|
||||
|
||||
ret := C.git_checkout_tree(v.ptr, tree.ptr, cOpts)
|
||||
runtime.KeepAlive(v)
|
||||
runtime.KeepAlive(tree)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
|
|
@ -9,39 +9,37 @@ import (
|
|||
)
|
||||
|
||||
type CherrypickOptions struct {
|
||||
Version uint
|
||||
Mainline uint
|
||||
MergeOpts MergeOptions
|
||||
CheckoutOpts CheckoutOpts
|
||||
Mainline uint
|
||||
MergeOptions MergeOptions
|
||||
CheckoutOptions CheckoutOptions
|
||||
}
|
||||
|
||||
func cherrypickOptionsFromC(c *C.git_cherrypick_options) CherrypickOptions {
|
||||
opts := CherrypickOptions{
|
||||
Version: uint(c.version),
|
||||
Mainline: uint(c.mainline),
|
||||
MergeOpts: mergeOptionsFromC(&c.merge_opts),
|
||||
CheckoutOpts: checkoutOptionsFromC(&c.checkout_opts),
|
||||
Mainline: uint(c.mainline),
|
||||
MergeOptions: mergeOptionsFromC(&c.merge_opts),
|
||||
CheckoutOptions: checkoutOptionsFromC(&c.checkout_opts),
|
||||
}
|
||||
return opts
|
||||
}
|
||||
|
||||
func (opts *CherrypickOptions) toC() *C.git_cherrypick_options {
|
||||
func populateCherrypickOptions(copts *C.git_cherrypick_options, opts *CherrypickOptions, errorTarget *error) *C.git_cherrypick_options {
|
||||
C.git_cherrypick_options_init(copts, C.GIT_CHERRYPICK_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
c := C.git_cherrypick_options{}
|
||||
c.version = C.uint(opts.Version)
|
||||
c.mainline = C.uint(opts.Mainline)
|
||||
c.merge_opts = *opts.MergeOpts.toC()
|
||||
c.checkout_opts = *opts.CheckoutOpts.toC()
|
||||
return &c
|
||||
copts.mainline = C.uint(opts.Mainline)
|
||||
populateMergeOptions(&copts.merge_opts, &opts.MergeOptions)
|
||||
populateCheckoutOptions(&copts.checkout_opts, &opts.CheckoutOptions, errorTarget)
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeCherrypickOpts(ptr *C.git_cherrypick_options) {
|
||||
if ptr == nil {
|
||||
func freeCherrypickOpts(copts *C.git_cherrypick_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
freeCheckoutOpts(&ptr.checkout_opts)
|
||||
freeMergeOptions(&copts.merge_opts)
|
||||
freeCheckoutOptions(&copts.checkout_opts)
|
||||
}
|
||||
|
||||
func DefaultCherrypickOptions() (CherrypickOptions, error) {
|
||||
|
@ -62,14 +60,18 @@ func (v *Repository) Cherrypick(commit *Commit, opts CherrypickOptions) error {
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cOpts := opts.toC()
|
||||
var err error
|
||||
cOpts := populateCherrypickOptions(&C.git_cherrypick_options{}, &opts, &err)
|
||||
defer freeCherrypickOpts(cOpts)
|
||||
|
||||
ecode := C.git_cherrypick(v.ptr, commit.cast_ptr, cOpts)
|
||||
ret := C.git_cherrypick(v.ptr, commit.cast_ptr, cOpts)
|
||||
runtime.KeepAlive(v)
|
||||
runtime.KeepAlive(commit)
|
||||
if ecode < 0 {
|
||||
return MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
@ -78,7 +80,8 @@ func (r *Repository) CherrypickCommit(pick, our *Commit, opts CherrypickOptions)
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cOpts := opts.MergeOpts.toC()
|
||||
cOpts := populateMergeOptions(&C.git_merge_options{}, &opts.MergeOptions)
|
||||
defer freeMergeOptions(cOpts)
|
||||
|
||||
var ptr *C.git_index
|
||||
ret := C.git_cherrypick_commit(&ptr, r.ptr, pick.cast_ptr, our.cast_ptr, C.uint(opts.Mainline), cOpts)
|
||||
|
|
|
@ -11,7 +11,7 @@ func checkout(t *testing.T, repo *Repository, commit *Commit) {
|
|||
t.Fatal(err)
|
||||
}
|
||||
|
||||
err = repo.CheckoutTree(tree, &CheckoutOpts{Strategy: CheckoutSafe})
|
||||
err = repo.CheckoutTree(tree, &CheckoutOptions{Strategy: CheckoutSafe})
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
|
110
clone.go
110
clone.go
|
@ -3,7 +3,7 @@ package git
|
|||
/*
|
||||
#include <git2.h>
|
||||
|
||||
extern void _go_git_populate_remote_cb(git_clone_options *opts);
|
||||
extern void _go_git_populate_clone_callbacks(git_clone_options *opts);
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
|
@ -11,11 +11,11 @@ import (
|
|||
"unsafe"
|
||||
)
|
||||
|
||||
type RemoteCreateCallback func(repo *Repository, name, url string) (*Remote, ErrorCode)
|
||||
type RemoteCreateCallback func(repo *Repository, name, url string) (*Remote, error)
|
||||
|
||||
type CloneOptions struct {
|
||||
*CheckoutOpts
|
||||
*FetchOptions
|
||||
CheckoutOptions CheckoutOptions
|
||||
FetchOptions FetchOptions
|
||||
Bare bool
|
||||
CheckoutBranch string
|
||||
RemoteCreateCallback RemoteCreateCallback
|
||||
|
@ -28,20 +28,23 @@ func Clone(url string, path string, options *CloneOptions) (*Repository, error)
|
|||
cpath := C.CString(path)
|
||||
defer C.free(unsafe.Pointer(cpath))
|
||||
|
||||
copts := (*C.git_clone_options)(C.calloc(1, C.size_t(unsafe.Sizeof(C.git_clone_options{}))))
|
||||
populateCloneOptions(copts, options)
|
||||
defer freeCloneOptions(copts)
|
||||
var err error
|
||||
cOptions := populateCloneOptions(&C.git_clone_options{}, options, &err)
|
||||
defer freeCloneOptions(cOptions)
|
||||
|
||||
if len(options.CheckoutBranch) != 0 {
|
||||
copts.checkout_branch = C.CString(options.CheckoutBranch)
|
||||
cOptions.checkout_branch = C.CString(options.CheckoutBranch)
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var ptr *C.git_repository
|
||||
ret := C.git_clone(&ptr, curl, cpath, copts)
|
||||
ret := C.git_clone(&ptr, curl, cpath, cOptions)
|
||||
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
@ -50,60 +53,81 @@ func Clone(url string, path string, options *CloneOptions) (*Repository, error)
|
|||
}
|
||||
|
||||
//export remoteCreateCallback
|
||||
func remoteCreateCallback(cremote unsafe.Pointer, crepo unsafe.Pointer, cname, curl *C.char, payload unsafe.Pointer) C.int {
|
||||
func remoteCreateCallback(
|
||||
out **C.git_remote,
|
||||
crepo *C.git_repository,
|
||||
cname, curl *C.char,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
name := C.GoString(cname)
|
||||
url := C.GoString(curl)
|
||||
repo := newRepositoryFromC((*C.git_repository)(crepo))
|
||||
// We don't own this repository, so make sure we don't try to free it
|
||||
runtime.SetFinalizer(repo, nil)
|
||||
repo := newRepositoryFromC(crepo)
|
||||
repo.weak = true
|
||||
defer repo.Free()
|
||||
|
||||
if opts, ok := pointerHandles.Get(payload).(CloneOptions); ok {
|
||||
remote, err := opts.RemoteCreateCallback(repo, name, url)
|
||||
// clear finalizer as the calling C function will
|
||||
// free the remote itself
|
||||
runtime.SetFinalizer(remote, nil)
|
||||
|
||||
if err == ErrOk && remote != nil {
|
||||
cptr := (**C.git_remote)(cremote)
|
||||
*cptr = remote.ptr
|
||||
} else if err == ErrOk && remote == nil {
|
||||
panic("no remote created by callback")
|
||||
}
|
||||
|
||||
return C.int(err)
|
||||
} else {
|
||||
data, ok := pointerHandles.Get(handle).(*cloneCallbackData)
|
||||
if !ok {
|
||||
panic("invalid remote create callback")
|
||||
}
|
||||
|
||||
remote, err := data.options.RemoteCreateCallback(repo, name, url)
|
||||
|
||||
if err != nil {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
if remote == nil {
|
||||
panic("no remote created by callback")
|
||||
}
|
||||
|
||||
*out = remote.ptr
|
||||
|
||||
// clear finalizer as the calling C function will
|
||||
// free the remote itself
|
||||
runtime.SetFinalizer(remote, nil)
|
||||
remote.repo.Remotes.untrackRemote(remote)
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func populateCloneOptions(ptr *C.git_clone_options, opts *CloneOptions) {
|
||||
C.git_clone_options_init(ptr, C.GIT_CLONE_OPTIONS_VERSION)
|
||||
type cloneCallbackData struct {
|
||||
options *CloneOptions
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
func populateCloneOptions(copts *C.git_clone_options, opts *CloneOptions, errorTarget *error) *C.git_clone_options {
|
||||
C.git_clone_options_init(copts, C.GIT_CLONE_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return
|
||||
return nil
|
||||
}
|
||||
populateCheckoutOpts(&ptr.checkout_opts, opts.CheckoutOpts)
|
||||
populateFetchOptions(&ptr.fetch_opts, opts.FetchOptions)
|
||||
ptr.bare = cbool(opts.Bare)
|
||||
populateCheckoutOptions(&copts.checkout_opts, &opts.CheckoutOptions, errorTarget)
|
||||
populateFetchOptions(&copts.fetch_opts, &opts.FetchOptions, errorTarget)
|
||||
copts.bare = cbool(opts.Bare)
|
||||
|
||||
if opts.RemoteCreateCallback != nil {
|
||||
data := &cloneCallbackData{
|
||||
options: opts,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
// Go v1.1 does not allow to assign a C function pointer
|
||||
C._go_git_populate_remote_cb(ptr)
|
||||
ptr.remote_cb_payload = pointerHandles.Track(*opts)
|
||||
C._go_git_populate_clone_callbacks(copts)
|
||||
copts.remote_cb_payload = pointerHandles.Track(data)
|
||||
}
|
||||
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeCloneOptions(ptr *C.git_clone_options) {
|
||||
if ptr == nil {
|
||||
func freeCloneOptions(copts *C.git_clone_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
|
||||
freeCheckoutOpts(&ptr.checkout_opts)
|
||||
freeCheckoutOptions(&copts.checkout_opts)
|
||||
freeFetchOptions(&copts.fetch_opts)
|
||||
|
||||
if ptr.remote_cb_payload != nil {
|
||||
pointerHandles.Untrack(ptr.remote_cb_payload)
|
||||
if copts.remote_cb_payload != nil {
|
||||
pointerHandles.Untrack(copts.remote_cb_payload)
|
||||
}
|
||||
|
||||
C.free(unsafe.Pointer(ptr.checkout_branch))
|
||||
C.free(unsafe.Pointer(ptr))
|
||||
C.free(unsafe.Pointer(copts.checkout_branch))
|
||||
}
|
||||
|
|
|
@ -2,6 +2,7 @@ package git
|
|||
|
||||
import (
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"testing"
|
||||
)
|
||||
|
||||
|
@ -49,15 +50,9 @@ func TestCloneWithCallback(t *testing.T) {
|
|||
|
||||
opts := CloneOptions{
|
||||
Bare: true,
|
||||
RemoteCreateCallback: func(r *Repository, name, url string) (*Remote, ErrorCode) {
|
||||
RemoteCreateCallback: func(r *Repository, name, url string) (*Remote, error) {
|
||||
testPayload += 1
|
||||
|
||||
remote, err := r.Remotes.Create(REMOTENAME, url)
|
||||
if err != nil {
|
||||
return nil, ErrGeneric
|
||||
}
|
||||
|
||||
return remote, ErrOk
|
||||
return r.Remotes.Create(REMOTENAME, url)
|
||||
},
|
||||
}
|
||||
|
||||
|
@ -74,4 +69,19 @@ func TestCloneWithCallback(t *testing.T) {
|
|||
if err != nil || remote == nil {
|
||||
t.Fatal("Remote was not created properly")
|
||||
}
|
||||
defer remote.Free()
|
||||
}
|
||||
|
||||
// TestCloneWithExternalHTTPUrl
|
||||
func TestCloneWithExternalHTTPUrl(t *testing.T) {
|
||||
|
||||
path, err := ioutil.TempDir("", "git2go")
|
||||
defer os.RemoveAll(path)
|
||||
|
||||
// clone the repo
|
||||
url := "https://github.com/libgit2/TestGitRepository"
|
||||
_, err = Clone(url, path, &CloneOptions{})
|
||||
if err != nil {
|
||||
t.Fatal("cannot clone remote repo via https, error: ", err)
|
||||
}
|
||||
}
|
||||
|
|
69
commit.go
69
commit.go
|
@ -12,8 +12,17 @@ import (
|
|||
"unsafe"
|
||||
)
|
||||
|
||||
// MessageEncoding is the encoding of commit messages.
|
||||
type MessageEncoding string
|
||||
|
||||
const (
|
||||
// MessageEncodingUTF8 is the default message encoding.
|
||||
MessageEncodingUTF8 MessageEncoding = "UTF-8"
|
||||
)
|
||||
|
||||
// Commit
|
||||
type Commit struct {
|
||||
doNotCompare
|
||||
Object
|
||||
cast_ptr *C.git_commit
|
||||
}
|
||||
|
@ -28,10 +37,14 @@ func (c *Commit) Message() string {
|
|||
return ret
|
||||
}
|
||||
|
||||
func (c *Commit) MessageEncoding() string {
|
||||
ret := C.GoString(C.git_commit_message_encoding(c.cast_ptr))
|
||||
func (c *Commit) MessageEncoding() MessageEncoding {
|
||||
ptr := C.git_commit_message_encoding(c.cast_ptr)
|
||||
if ptr == nil {
|
||||
return MessageEncodingUTF8
|
||||
}
|
||||
ret := C.GoString(ptr)
|
||||
runtime.KeepAlive(c)
|
||||
return ret
|
||||
return MessageEncoding(ret)
|
||||
}
|
||||
|
||||
func (c *Commit) RawMessage() string {
|
||||
|
@ -55,6 +68,19 @@ func (c *Commit) ContentToSign() string {
|
|||
// CommitSigningCallback defines a function type that takes some data to sign and returns (signature, signature_field, error)
|
||||
type CommitSigningCallback func(string) (signature, signatureField string, err error)
|
||||
|
||||
// CommitCreateCallback defines a function type that is called when another
|
||||
// function is going to create commits (for example, Rebase) to allow callers
|
||||
// to override the commit creation behavior. For example, users may wish to
|
||||
// sign commits by providing this information to Repository.CreateCommitBuffer,
|
||||
// signing that buffer, then calling Repository.CreateCommitWithSignature.
|
||||
type CommitCreateCallback func(
|
||||
author, committer *Signature,
|
||||
messageEncoding MessageEncoding,
|
||||
message string,
|
||||
tree *Tree,
|
||||
parents ...*Commit,
|
||||
) (oid *Oid, err error)
|
||||
|
||||
// WithSignatureUsing creates a new signed commit from this one using the given signing callback
|
||||
func (c *Commit) WithSignatureUsing(f CommitSigningCallback) (*Oid, error) {
|
||||
signature, signatureField, err := f(c.ContentToSign())
|
||||
|
@ -67,40 +93,11 @@ func (c *Commit) WithSignatureUsing(f CommitSigningCallback) (*Oid, error) {
|
|||
|
||||
// WithSignature creates a new signed commit from the given signature and signature field
|
||||
func (c *Commit) WithSignature(signature string, signatureField string) (*Oid, error) {
|
||||
totalCommit := c.ContentToSign()
|
||||
|
||||
oid := new(Oid)
|
||||
|
||||
var csf *C.char = nil
|
||||
if signatureField != "" {
|
||||
csf = C.CString(signatureField)
|
||||
defer C.free(unsafe.Pointer(csf))
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cTotalCommit := C.CString(totalCommit)
|
||||
cSignature := C.CString(signature)
|
||||
defer C.free(unsafe.Pointer(cTotalCommit))
|
||||
defer C.free(unsafe.Pointer(cSignature))
|
||||
|
||||
ret := C.git_commit_create_with_signature(
|
||||
oid.toC(),
|
||||
c.Owner().ptr,
|
||||
cTotalCommit,
|
||||
cSignature,
|
||||
csf,
|
||||
return c.Owner().CreateCommitWithSignature(
|
||||
c.ContentToSign(),
|
||||
signature,
|
||||
signatureField,
|
||||
)
|
||||
|
||||
runtime.KeepAlive(c)
|
||||
runtime.KeepAlive(oid)
|
||||
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return oid, nil
|
||||
}
|
||||
|
||||
func (c *Commit) ExtractSignature() (string, string, error) {
|
||||
|
|
16
config.go
16
config.go
|
@ -52,6 +52,7 @@ func newConfigEntryFromC(centry *C.git_config_entry) *ConfigEntry {
|
|||
}
|
||||
|
||||
type Config struct {
|
||||
doNotCompare
|
||||
ptr *C.git_config
|
||||
}
|
||||
|
||||
|
@ -361,6 +362,7 @@ func OpenOndisk(path string) (*Config, error) {
|
|||
}
|
||||
|
||||
type ConfigIterator struct {
|
||||
doNotCompare
|
||||
ptr *C.git_config_iterator
|
||||
cfg *Config
|
||||
}
|
||||
|
@ -450,3 +452,17 @@ func ConfigFindProgramdata() (string, error) {
|
|||
|
||||
return C.GoString(buf.ptr), nil
|
||||
}
|
||||
|
||||
// OpenDefault opens the default config according to git rules
|
||||
func OpenDefault() (*Config, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
config := new(Config)
|
||||
|
||||
if ret := C.git_config_open_default(&config.ptr); ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return config, nil
|
||||
}
|
||||
|
|
|
@ -107,3 +107,13 @@ func TestConfigLookups(t *testing.T) {
|
|||
test(c, t)
|
||||
}
|
||||
}
|
||||
|
||||
func TestOpenDefault(t *testing.T) {
|
||||
|
||||
c, err := OpenDefault()
|
||||
if err != nil {
|
||||
t.Errorf("OpenDefault error: '%v'. Expected none\n", err)
|
||||
return
|
||||
}
|
||||
defer c.Free()
|
||||
}
|
||||
|
|
236
credentials.go
236
credentials.go
|
@ -2,62 +2,146 @@ package git
|
|||
|
||||
/*
|
||||
#include <git2.h>
|
||||
#include <git2/sys/cred.h>
|
||||
#include <git2/credential.h>
|
||||
#include <git2/sys/credential.h>
|
||||
|
||||
git_credtype_t _go_git_cred_credtype(git_cred *cred);
|
||||
git_credential_t _go_git_credential_credtype(git_credential *cred);
|
||||
void _go_git_populate_credential_ssh_custom(git_credential_ssh_custom *cred);
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"crypto/rand"
|
||||
"errors"
|
||||
"fmt"
|
||||
"runtime"
|
||||
"strings"
|
||||
"unsafe"
|
||||
|
||||
"golang.org/x/crypto/ssh"
|
||||
)
|
||||
|
||||
type CredType uint
|
||||
// CredentialType is a bitmask of supported credential types.
|
||||
//
|
||||
// This represents the various types of authentication methods supported by the
|
||||
// library.
|
||||
type CredentialType uint
|
||||
|
||||
const (
|
||||
CredTypeUserpassPlaintext CredType = C.GIT_CREDTYPE_USERPASS_PLAINTEXT
|
||||
CredTypeSshKey CredType = C.GIT_CREDTYPE_SSH_KEY
|
||||
CredTypeSshCustom CredType = C.GIT_CREDTYPE_SSH_CUSTOM
|
||||
CredTypeDefault CredType = C.GIT_CREDTYPE_DEFAULT
|
||||
CredentialTypeUserpassPlaintext CredentialType = C.GIT_CREDENTIAL_USERPASS_PLAINTEXT
|
||||
CredentialTypeSSHKey CredentialType = C.GIT_CREDENTIAL_SSH_KEY
|
||||
CredentialTypeSSHCustom CredentialType = C.GIT_CREDENTIAL_SSH_CUSTOM
|
||||
CredentialTypeDefault CredentialType = C.GIT_CREDENTIAL_DEFAULT
|
||||
CredentialTypeSSHInteractive CredentialType = C.GIT_CREDENTIAL_SSH_INTERACTIVE
|
||||
CredentialTypeUsername CredentialType = C.GIT_CREDENTIAL_USERNAME
|
||||
CredentialTypeSSHMemory CredentialType = C.GIT_CREDENTIAL_SSH_MEMORY
|
||||
)
|
||||
|
||||
type Cred struct {
|
||||
ptr *C.git_cred
|
||||
func (t CredentialType) String() string {
|
||||
if t == 0 {
|
||||
return "CredentialType(0)"
|
||||
}
|
||||
|
||||
var parts []string
|
||||
|
||||
if (t & CredentialTypeUserpassPlaintext) != 0 {
|
||||
parts = append(parts, "UserpassPlaintext")
|
||||
t &= ^CredentialTypeUserpassPlaintext
|
||||
}
|
||||
if (t & CredentialTypeSSHKey) != 0 {
|
||||
parts = append(parts, "SSHKey")
|
||||
t &= ^CredentialTypeSSHKey
|
||||
}
|
||||
if (t & CredentialTypeSSHCustom) != 0 {
|
||||
parts = append(parts, "SSHCustom")
|
||||
t &= ^CredentialTypeSSHCustom
|
||||
}
|
||||
if (t & CredentialTypeDefault) != 0 {
|
||||
parts = append(parts, "Default")
|
||||
t &= ^CredentialTypeDefault
|
||||
}
|
||||
if (t & CredentialTypeSSHInteractive) != 0 {
|
||||
parts = append(parts, "SSHInteractive")
|
||||
t &= ^CredentialTypeSSHInteractive
|
||||
}
|
||||
if (t & CredentialTypeUsername) != 0 {
|
||||
parts = append(parts, "Username")
|
||||
t &= ^CredentialTypeUsername
|
||||
}
|
||||
if (t & CredentialTypeSSHMemory) != 0 {
|
||||
parts = append(parts, "SSHMemory")
|
||||
t &= ^CredentialTypeSSHMemory
|
||||
}
|
||||
|
||||
if t != 0 {
|
||||
parts = append(parts, fmt.Sprintf("CredentialType(%#x)", t))
|
||||
}
|
||||
|
||||
return strings.Join(parts, "|")
|
||||
}
|
||||
|
||||
func newCred() *Cred {
|
||||
cred := &Cred{}
|
||||
runtime.SetFinalizer(cred, (*Cred).Free)
|
||||
type Credential struct {
|
||||
doNotCompare
|
||||
ptr *C.git_credential
|
||||
}
|
||||
|
||||
func newCredential() *Credential {
|
||||
cred := &Credential{}
|
||||
runtime.SetFinalizer(cred, (*Credential).Free)
|
||||
return cred
|
||||
}
|
||||
|
||||
func (o *Cred) HasUsername() bool {
|
||||
if C.git_cred_has_username(o.ptr) == 1 {
|
||||
func (o *Credential) HasUsername() bool {
|
||||
if C.git_credential_has_username(o.ptr) == 1 {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func (o *Cred) Type() CredType {
|
||||
return (CredType)(C._go_git_cred_credtype(o.ptr))
|
||||
func (o *Credential) Type() CredentialType {
|
||||
return (CredentialType)(C._go_git_credential_credtype(o.ptr))
|
||||
}
|
||||
|
||||
func (o *Cred) Free() {
|
||||
C.git_cred_free(o.ptr)
|
||||
func (o *Credential) Free() {
|
||||
C.git_credential_free(o.ptr)
|
||||
runtime.SetFinalizer(o, nil)
|
||||
o.ptr = nil
|
||||
}
|
||||
|
||||
func NewCredUserpassPlaintext(username string, password string) (*Cred, error) {
|
||||
// GetUserpassPlaintext returns the plaintext username/password combination stored in the Cred.
|
||||
func (o *Credential) GetUserpassPlaintext() (username, password string, err error) {
|
||||
if o.Type() != CredentialTypeUserpassPlaintext {
|
||||
err = errors.New("credential is not userpass plaintext")
|
||||
return
|
||||
}
|
||||
|
||||
plaintextCredPtr := (*C.git_cred_userpass_plaintext)(unsafe.Pointer(o.ptr))
|
||||
username = C.GoString(plaintextCredPtr.username)
|
||||
password = C.GoString(plaintextCredPtr.password)
|
||||
return
|
||||
}
|
||||
|
||||
// GetSSHKey returns the SSH-specific key information from the Cred object.
|
||||
func (o *Credential) GetSSHKey() (username, publickey, privatekey, passphrase string, err error) {
|
||||
if o.Type() != CredentialTypeSSHKey && o.Type() != CredentialTypeSSHMemory {
|
||||
err = fmt.Errorf("credential is not an SSH key: %v", o.Type())
|
||||
return
|
||||
}
|
||||
|
||||
sshKeyCredPtr := (*C.git_cred_ssh_key)(unsafe.Pointer(o.ptr))
|
||||
username = C.GoString(sshKeyCredPtr.username)
|
||||
publickey = C.GoString(sshKeyCredPtr.publickey)
|
||||
privatekey = C.GoString(sshKeyCredPtr.privatekey)
|
||||
passphrase = C.GoString(sshKeyCredPtr.passphrase)
|
||||
return
|
||||
}
|
||||
|
||||
func NewCredentialUsername(username string) (*Credential, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cred := newCred()
|
||||
cred := newCredential()
|
||||
cusername := C.CString(username)
|
||||
defer C.free(unsafe.Pointer(cusername))
|
||||
cpassword := C.CString(password)
|
||||
defer C.free(unsafe.Pointer(cpassword))
|
||||
ret := C.git_cred_userpass_plaintext_new(&cred.ptr, cusername, cpassword)
|
||||
ret := C.git_credential_username_new(&cred.ptr, cusername)
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
|
@ -65,13 +149,30 @@ func NewCredUserpassPlaintext(username string, password string) (*Cred, error) {
|
|||
return cred, nil
|
||||
}
|
||||
|
||||
// NewCredSshKey creates new ssh credentials reading the public and private keys
|
||||
// from the file system.
|
||||
func NewCredSshKey(username string, publicKeyPath string, privateKeyPath string, passphrase string) (*Cred, error) {
|
||||
func NewCredentialUserpassPlaintext(username string, password string) (*Credential, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cred := newCred()
|
||||
cred := newCredential()
|
||||
cusername := C.CString(username)
|
||||
defer C.free(unsafe.Pointer(cusername))
|
||||
cpassword := C.CString(password)
|
||||
defer C.free(unsafe.Pointer(cpassword))
|
||||
ret := C.git_credential_userpass_plaintext_new(&cred.ptr, cusername, cpassword)
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
return cred, nil
|
||||
}
|
||||
|
||||
// NewCredentialSSHKey creates new ssh credentials reading the public and private keys
|
||||
// from the file system.
|
||||
func NewCredentialSSHKey(username string, publicKeyPath string, privateKeyPath string, passphrase string) (*Credential, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cred := newCredential()
|
||||
cusername := C.CString(username)
|
||||
defer C.free(unsafe.Pointer(cusername))
|
||||
cpublickey := C.CString(publicKeyPath)
|
||||
|
@ -80,7 +181,7 @@ func NewCredSshKey(username string, publicKeyPath string, privateKeyPath string,
|
|||
defer C.free(unsafe.Pointer(cprivatekey))
|
||||
cpassphrase := C.CString(passphrase)
|
||||
defer C.free(unsafe.Pointer(cpassphrase))
|
||||
ret := C.git_cred_ssh_key_new(&cred.ptr, cusername, cpublickey, cprivatekey, cpassphrase)
|
||||
ret := C.git_credential_ssh_key_new(&cred.ptr, cusername, cpublickey, cprivatekey, cpassphrase)
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
|
@ -88,13 +189,13 @@ func NewCredSshKey(username string, publicKeyPath string, privateKeyPath string,
|
|||
return cred, nil
|
||||
}
|
||||
|
||||
// NewCredSshKeyFromMemory creates new ssh credentials using the publicKey and privateKey
|
||||
// NewCredentialSSHKeyFromMemory creates new ssh credentials using the publicKey and privateKey
|
||||
// arguments as the values for the public and private keys.
|
||||
func NewCredSshKeyFromMemory(username string, publicKey string, privateKey string, passphrase string) (*Cred, error) {
|
||||
func NewCredentialSSHKeyFromMemory(username string, publicKey string, privateKey string, passphrase string) (*Credential, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cred := newCred()
|
||||
cred := newCredential()
|
||||
cusername := C.CString(username)
|
||||
defer C.free(unsafe.Pointer(cusername))
|
||||
cpublickey := C.CString(publicKey)
|
||||
|
@ -103,7 +204,7 @@ func NewCredSshKeyFromMemory(username string, publicKey string, privateKey strin
|
|||
defer C.free(unsafe.Pointer(cprivatekey))
|
||||
cpassphrase := C.CString(passphrase)
|
||||
defer C.free(unsafe.Pointer(cpassphrase))
|
||||
ret := C.git_cred_ssh_key_memory_new(&cred.ptr, cusername, cpublickey, cprivatekey, cpassphrase)
|
||||
ret := C.git_credential_ssh_key_memory_new(&cred.ptr, cusername, cpublickey, cprivatekey, cpassphrase)
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
|
@ -111,14 +212,14 @@ func NewCredSshKeyFromMemory(username string, publicKey string, privateKey strin
|
|||
return cred, nil
|
||||
}
|
||||
|
||||
func NewCredSshKeyFromAgent(username string) (*Cred, error) {
|
||||
func NewCredentialSSHKeyFromAgent(username string) (*Credential, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cred := newCred()
|
||||
cred := newCredential()
|
||||
cusername := C.CString(username)
|
||||
defer C.free(unsafe.Pointer(cusername))
|
||||
ret := C.git_cred_ssh_key_from_agent(&cred.ptr, cusername)
|
||||
ret := C.git_credential_ssh_key_from_agent(&cred.ptr, cusername)
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
|
@ -126,12 +227,69 @@ func NewCredSshKeyFromAgent(username string) (*Cred, error) {
|
|||
return cred, nil
|
||||
}
|
||||
|
||||
func NewCredDefault() (*Cred, error) {
|
||||
type credentialSSHCustomData struct {
|
||||
signer ssh.Signer
|
||||
}
|
||||
|
||||
//export credentialSSHCustomFree
|
||||
func credentialSSHCustomFree(cred *C.git_credential_ssh_custom) {
|
||||
if cred == nil {
|
||||
return
|
||||
}
|
||||
|
||||
C.free(unsafe.Pointer(cred.username))
|
||||
C.free(unsafe.Pointer(cred.publickey))
|
||||
pointerHandles.Untrack(cred.payload)
|
||||
C.free(unsafe.Pointer(cred))
|
||||
}
|
||||
|
||||
//export credentialSSHSignCallback
|
||||
func credentialSSHSignCallback(
|
||||
errorMessage **C.char,
|
||||
sig **C.uchar,
|
||||
sig_len *C.size_t,
|
||||
data *C.uchar,
|
||||
data_len C.size_t,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
signer := pointerHandles.Get(handle).(*credentialSSHCustomData).signer
|
||||
signature, err := signer.Sign(rand.Reader, C.GoBytes(unsafe.Pointer(data), C.int(data_len)))
|
||||
if err != nil {
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
*sig = (*C.uchar)(C.CBytes(signature.Blob))
|
||||
*sig_len = C.size_t(len(signature.Blob))
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// NewCredentialSSHKeyFromSigner creates new SSH credentials using the provided signer.
|
||||
func NewCredentialSSHKeyFromSigner(username string, signer ssh.Signer) (*Credential, error) {
|
||||
publicKey := signer.PublicKey().Marshal()
|
||||
|
||||
ccred := (*C.git_credential_ssh_custom)(C.calloc(1, C.size_t(unsafe.Sizeof(C.git_credential_ssh_custom{}))))
|
||||
ccred.parent.credtype = C.GIT_CREDENTIAL_SSH_CUSTOM
|
||||
ccred.username = C.CString(username)
|
||||
ccred.publickey = (*C.char)(C.CBytes(publicKey))
|
||||
ccred.publickey_len = C.size_t(len(publicKey))
|
||||
C._go_git_populate_credential_ssh_custom(ccred)
|
||||
|
||||
data := credentialSSHCustomData{
|
||||
signer: signer,
|
||||
}
|
||||
ccred.payload = pointerHandles.Track(&data)
|
||||
|
||||
cred := newCredential()
|
||||
cred.ptr = &ccred.parent
|
||||
|
||||
return cred, nil
|
||||
}
|
||||
|
||||
func NewCredentialDefault() (*Credential, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cred := newCred()
|
||||
ret := C.git_cred_default_new(&cred.ptr)
|
||||
cred := newCredential()
|
||||
ret := C.git_credential_default_new(&cred.ptr)
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
|
|
|
@ -0,0 +1,271 @@
|
|||
package git
|
||||
|
||||
/*
|
||||
#include <git2.h>
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
// The constants, functions, and types in this files are slated for deprecation
|
||||
// in the next major version.
|
||||
|
||||
// blob.go
|
||||
|
||||
// Deprecated: BlobChunkCallback is not used.
|
||||
type BlobChunkCallback func(maxLen int) ([]byte, error)
|
||||
|
||||
// Deprecated: BlobCallbackData is not used.
|
||||
type BlobCallbackData struct {
|
||||
Callback BlobChunkCallback
|
||||
Error error
|
||||
}
|
||||
|
||||
// checkout.go
|
||||
|
||||
// Deprecated: CheckoutOpts is a deprecated alias of CheckoutOptions.
|
||||
type CheckoutOpts = CheckoutOptions
|
||||
|
||||
// credentials.go
|
||||
|
||||
// Deprecated: CredType is a deprecated alias of CredentialType
|
||||
type CredType = CredentialType
|
||||
|
||||
const (
|
||||
CredTypeUserpassPlaintext = CredentialTypeUserpassPlaintext
|
||||
CredTypeSshKey = CredentialTypeSSHKey
|
||||
CredTypeSshCustom = CredentialTypeSSHCustom
|
||||
CredTypeDefault = CredentialTypeDefault
|
||||
)
|
||||
|
||||
// Deprecated: Cred is a deprecated alias of Credential
|
||||
type Cred = Credential
|
||||
|
||||
// Deprecated: NewCredUsername is a deprecated alias of NewCredentialUsername.
|
||||
func NewCredUsername(username string) (*Cred, error) {
|
||||
return NewCredentialUsername(username)
|
||||
}
|
||||
|
||||
// Deprecated: NewCredUserpassPlaintext is a deprecated alias of NewCredentialUserpassPlaintext.
|
||||
func NewCredUserpassPlaintext(username string, password string) (*Cred, error) {
|
||||
return NewCredentialUserpassPlaintext(username, password)
|
||||
}
|
||||
|
||||
// Deprecated: NewCredSshKey is a deprecated alias of NewCredentialSshKey.
|
||||
func NewCredSshKey(username string, publicKeyPath string, privateKeyPath string, passphrase string) (*Cred, error) {
|
||||
return NewCredentialSSHKey(username, publicKeyPath, privateKeyPath, passphrase)
|
||||
}
|
||||
|
||||
// Deprecated: NewCredSshKeyFromMemory is a deprecated alias of NewCredentialSSHKeyFromMemory.
|
||||
func NewCredSshKeyFromMemory(username string, publicKey string, privateKey string, passphrase string) (*Cred, error) {
|
||||
return NewCredentialSSHKeyFromMemory(username, publicKey, privateKey, passphrase)
|
||||
}
|
||||
|
||||
// Deprecated: NewCredSshKeyFromAgent is a deprecated alias of NewCredentialSSHFromAgent.
|
||||
func NewCredSshKeyFromAgent(username string) (*Cred, error) {
|
||||
return NewCredentialSSHKeyFromAgent(username)
|
||||
}
|
||||
|
||||
// Deprecated: NewCredDefault is a deprecated alias fof NewCredentialDefault.
|
||||
func NewCredDefault() (*Cred, error) {
|
||||
return NewCredentialDefault()
|
||||
}
|
||||
|
||||
// diff.go
|
||||
|
||||
const (
|
||||
// Deprecated: DiffIgnoreWhitespaceEol is a deprecated alias of DiffIgnoreWhitespaceEOL.
|
||||
DiffIgnoreWitespaceEol = DiffIgnoreWhitespaceEOL
|
||||
)
|
||||
|
||||
// features.go
|
||||
|
||||
const (
|
||||
// Deprecated: FeatureHttps is a deprecated alias of FeatureHTTPS.
|
||||
FeatureHttps = FeatureHTTPS
|
||||
|
||||
// Deprecated: FeatureSsh is a deprecated alias of FeatureSSH.
|
||||
FeatureSsh = FeatureSSH
|
||||
)
|
||||
|
||||
// git.go
|
||||
|
||||
const (
|
||||
// Deprecated: ErrClassNone is a deprecated alias of ErrorClassNone.
|
||||
ErrClassNone = ErrorClassNone
|
||||
// Deprecated: ErrClassNoMemory is a deprecated alias of ErrorClassNoMemory.
|
||||
ErrClassNoMemory = ErrorClassNoMemory
|
||||
// Deprecated: ErrClassOs is a deprecated alias of ErrorClassOS.
|
||||
ErrClassOs = ErrorClassOS
|
||||
// Deprecated: ErrClassInvalid is a deprecated alias of ErrorClassInvalid.
|
||||
ErrClassInvalid = ErrorClassInvalid
|
||||
// Deprecated: ErrClassReference is a deprecated alias of ErrorClassReference.
|
||||
ErrClassReference = ErrorClassReference
|
||||
// Deprecated: ErrClassZlib is a deprecated alias of ErrorClassZlib.
|
||||
ErrClassZlib = ErrorClassZlib
|
||||
// Deprecated: ErrClassRepository is a deprecated alias of ErrorClassRepository.
|
||||
ErrClassRepository = ErrorClassRepository
|
||||
// Deprecated: ErrClassConfig is a deprecated alias of ErrorClassConfig.
|
||||
ErrClassConfig = ErrorClassConfig
|
||||
// Deprecated: ErrClassRegex is a deprecated alias of ErrorClassRegex.
|
||||
ErrClassRegex = ErrorClassRegex
|
||||
// Deprecated: ErrClassOdb is a deprecated alias of ErrorClassOdb.
|
||||
ErrClassOdb = ErrorClassOdb
|
||||
// Deprecated: ErrClassIndex is a deprecated alias of ErrorClassIndex.
|
||||
ErrClassIndex = ErrorClassIndex
|
||||
// Deprecated: ErrClassObject is a deprecated alias of ErrorClassObject.
|
||||
ErrClassObject = ErrorClassObject
|
||||
// Deprecated: ErrClassNet is a deprecated alias of ErrorClassNet.
|
||||
ErrClassNet = ErrorClassNet
|
||||
// Deprecated: ErrClassTag is a deprecated alias of ErrorClassTag.
|
||||
ErrClassTag = ErrorClassTag
|
||||
// Deprecated: ErrClassTree is a deprecated alias of ErrorClassTree.
|
||||
ErrClassTree = ErrorClassTree
|
||||
// Deprecated: ErrClassIndexer is a deprecated alias of ErrorClassIndexer.
|
||||
ErrClassIndexer = ErrorClassIndexer
|
||||
// Deprecated: ErrClassSSL is a deprecated alias of ErrorClassSSL.
|
||||
ErrClassSSL = ErrorClassSSL
|
||||
// Deprecated: ErrClassSubmodule is a deprecated alias of ErrorClassSubmodule.
|
||||
ErrClassSubmodule = ErrorClassSubmodule
|
||||
// Deprecated: ErrClassThread is a deprecated alias of ErrorClassThread.
|
||||
ErrClassThread = ErrorClassThread
|
||||
// Deprecated: ErrClassStash is a deprecated alias of ErrorClassStash.
|
||||
ErrClassStash = ErrorClassStash
|
||||
// Deprecated: ErrClassCheckout is a deprecated alias of ErrorClassCheckout.
|
||||
ErrClassCheckout = ErrorClassCheckout
|
||||
// Deprecated: ErrClassFetchHead is a deprecated alias of ErrorClassFetchHead.
|
||||
ErrClassFetchHead = ErrorClassFetchHead
|
||||
// Deprecated: ErrClassMerge is a deprecated alias of ErrorClassMerge.
|
||||
ErrClassMerge = ErrorClassMerge
|
||||
// Deprecated: ErrClassSsh is a deprecated alias of ErrorClassSSH.
|
||||
ErrClassSsh = ErrorClassSSH
|
||||
// Deprecated: ErrClassFilter is a deprecated alias of ErrorClassFilter.
|
||||
ErrClassFilter = ErrorClassFilter
|
||||
// Deprecated: ErrClassRevert is a deprecated alias of ErrorClassRevert.
|
||||
ErrClassRevert = ErrorClassRevert
|
||||
// Deprecated: ErrClassCallback is a deprecated alias of ErrorClassCallback.
|
||||
ErrClassCallback = ErrorClassCallback
|
||||
// Deprecated: ErrClassRebase is a deprecated alias of ErrorClassRebase.
|
||||
ErrClassRebase = ErrorClassRebase
|
||||
// Deprecated: ErrClassPatch is a deprecated alias of ErrorClassPatch.
|
||||
ErrClassPatch = ErrorClassPatch
|
||||
)
|
||||
|
||||
const (
|
||||
// Deprecated: ErrOk is a deprecated alias of ErrorCodeOK.
|
||||
ErrOk = ErrorCodeOK
|
||||
// Deprecated: ErrGeneric is a deprecated alias of ErrorCodeGeneric.
|
||||
ErrGeneric = ErrorCodeGeneric
|
||||
// Deprecated: ErrNotFound is a deprecated alias of ErrorCodeNotFound.
|
||||
ErrNotFound = ErrorCodeNotFound
|
||||
// Deprecated: ErrExists is a deprecated alias of ErrorCodeExists.
|
||||
ErrExists = ErrorCodeExists
|
||||
// Deprecated: ErrAmbiguous is a deprecated alias of ErrorCodeAmbiguous.
|
||||
ErrAmbiguous = ErrorCodeAmbiguous
|
||||
// Deprecated: ErrAmbigious is a deprecated alias of ErrorCodeAmbiguous.
|
||||
ErrAmbigious = ErrorCodeAmbiguous
|
||||
// Deprecated: ErrBuffs is a deprecated alias of ErrorCodeBuffs.
|
||||
ErrBuffs = ErrorCodeBuffs
|
||||
// Deprecated: ErrUser is a deprecated alias of ErrorCodeUser.
|
||||
ErrUser = ErrorCodeUser
|
||||
// Deprecated: ErrBareRepo is a deprecated alias of ErrorCodeBareRepo.
|
||||
ErrBareRepo = ErrorCodeBareRepo
|
||||
// Deprecated: ErrUnbornBranch is a deprecated alias of ErrorCodeUnbornBranch.
|
||||
ErrUnbornBranch = ErrorCodeUnbornBranch
|
||||
// Deprecated: ErrUnmerged is a deprecated alias of ErrorCodeUnmerged.
|
||||
ErrUnmerged = ErrorCodeUnmerged
|
||||
// Deprecated: ErrNonFastForward is a deprecated alias of ErrorCodeNonFastForward.
|
||||
ErrNonFastForward = ErrorCodeNonFastForward
|
||||
// Deprecated: ErrInvalidSpec is a deprecated alias of ErrorCodeInvalidSpec.
|
||||
ErrInvalidSpec = ErrorCodeInvalidSpec
|
||||
// Deprecated: ErrConflict is a deprecated alias of ErrorCodeConflict.
|
||||
ErrConflict = ErrorCodeConflict
|
||||
// Deprecated: ErrLocked is a deprecated alias of ErrorCodeLocked.
|
||||
ErrLocked = ErrorCodeLocked
|
||||
// Deprecated: ErrModified is a deprecated alias of ErrorCodeModified.
|
||||
ErrModified = ErrorCodeModified
|
||||
// Deprecated: ErrAuth is a deprecated alias of ErrorCodeAuth.
|
||||
ErrAuth = ErrorCodeAuth
|
||||
// Deprecated: ErrCertificate is a deprecated alias of ErrorCodeCertificate.
|
||||
ErrCertificate = ErrorCodeCertificate
|
||||
// Deprecated: ErrApplied is a deprecated alias of ErrorCodeApplied.
|
||||
ErrApplied = ErrorCodeApplied
|
||||
// Deprecated: ErrPeel is a deprecated alias of ErrorCodePeel.
|
||||
ErrPeel = ErrorCodePeel
|
||||
// Deprecated: ErrEOF is a deprecated alias of ErrorCodeEOF.
|
||||
ErrEOF = ErrorCodeEOF
|
||||
// Deprecated: ErrUncommitted is a deprecated alias of ErrorCodeUncommitted.
|
||||
ErrUncommitted = ErrorCodeUncommitted
|
||||
// Deprecated: ErrDirectory is a deprecated alias of ErrorCodeDirectory.
|
||||
ErrDirectory = ErrorCodeDirectory
|
||||
// Deprecated: ErrMergeConflict is a deprecated alias of ErrorCodeMergeConflict.
|
||||
ErrMergeConflict = ErrorCodeMergeConflict
|
||||
// Deprecated: ErrPassthrough is a deprecated alias of ErrorCodePassthrough.
|
||||
ErrPassthrough = ErrorCodePassthrough
|
||||
// Deprecated: ErrIterOver is a deprecated alias of ErrorCodeIterOver.
|
||||
ErrIterOver = ErrorCodeIterOver
|
||||
// Deprecated: ErrApplyFail is a deprecated alias of ErrorCodeApplyFail.
|
||||
ErrApplyFail = ErrorCodeApplyFail
|
||||
)
|
||||
|
||||
// index.go
|
||||
|
||||
// Deprecated: IndexAddOpts is a deprecated alias of IndexAddOption.
|
||||
type IndexAddOpts = IndexAddOption
|
||||
|
||||
// Deprecated: IndexStageOpts is a deprecated alias of IndexStageState.
|
||||
type IndexStageOpts = IndexStageState
|
||||
|
||||
// submodule.go
|
||||
|
||||
// Deprecated: SubmoduleCbk is a deprecated alias of SubmoduleCallback.
|
||||
type SubmoduleCbk = SubmoduleCallback
|
||||
|
||||
// Deprecated: SubmoduleVisitor is not used.
|
||||
func SubmoduleVisitor(csub unsafe.Pointer, name *C.char, handle unsafe.Pointer) C.int {
|
||||
sub := &Submodule{ptr: (*C.git_submodule)(csub)}
|
||||
|
||||
callback, ok := pointerHandles.Get(handle).(SubmoduleCallback)
|
||||
if !ok {
|
||||
panic("invalid submodule visitor callback")
|
||||
}
|
||||
err := callback(sub, C.GoString(name))
|
||||
if err != nil {
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// reference.go
|
||||
|
||||
// Deprecated: ReferenceIsValidName is a deprecated alias of ReferenceNameIsValid.
|
||||
func ReferenceIsValidName(name string) bool {
|
||||
valid, _ := ReferenceNameIsValid(name)
|
||||
return valid
|
||||
}
|
||||
|
||||
// remote.go
|
||||
|
||||
// Deprecated: RemoteIsValidName is a deprecated alias of RemoteNameIsValid.
|
||||
func RemoteIsValidName(name string) bool {
|
||||
valid, _ := RemoteNameIsValid(name)
|
||||
return valid
|
||||
}
|
||||
|
||||
// tree.go
|
||||
|
||||
// Deprecated: CallbackGitTreeWalk is not used.
|
||||
func CallbackGitTreeWalk(_root *C.char, entry *C.git_tree_entry, ptr unsafe.Pointer) C.int {
|
||||
root := C.GoString(_root)
|
||||
|
||||
callback, ok := pointerHandles.Get(ptr).(TreeWalkCallback)
|
||||
if !ok {
|
||||
panic("invalid treewalk callback")
|
||||
}
|
||||
err := callback(root, newTreeEntry(entry))
|
||||
if err != nil {
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
|
@ -176,6 +176,7 @@ func (repo *Repository) DescribeWorkdir(opts *DescribeOptions) (*DescribeResult,
|
|||
//
|
||||
// Use Format() to get a string out of it.
|
||||
type DescribeResult struct {
|
||||
doNotCompare
|
||||
ptr *C.git_describe_result
|
||||
}
|
||||
|
||||
|
|
426
diff.go
426
diff.go
|
@ -3,7 +3,7 @@ package git
|
|||
/*
|
||||
#include <git2.h>
|
||||
|
||||
extern void _go_git_populate_apply_cb(git_apply_options *options);
|
||||
extern void _go_git_populate_apply_callbacks(git_apply_options *options);
|
||||
extern int _go_git_diff_foreach(git_diff *diff, int eachFile, int eachHunk, int eachLine, void *payload);
|
||||
extern void _go_git_setup_diff_notify_callbacks(git_diff_options* opts);
|
||||
extern int _go_git_diff_blobs(git_blob *old, const char *old_path, git_blob *new, const char *new_path, git_diff_options *opts, int eachFile, int eachHunk, int eachLine, void *payload);
|
||||
|
@ -132,6 +132,7 @@ func diffLineFromC(line *C.git_diff_line) DiffLine {
|
|||
}
|
||||
|
||||
type Diff struct {
|
||||
doNotCompare
|
||||
ptr *C.git_diff
|
||||
repo *Repository
|
||||
runFinalizer bool
|
||||
|
@ -219,6 +220,7 @@ func (diff *Diff) FindSimilar(opts *DiffFindOptions) error {
|
|||
}
|
||||
|
||||
type DiffStats struct {
|
||||
doNotCompare
|
||||
ptr *C.git_diff_stats
|
||||
}
|
||||
|
||||
|
@ -294,11 +296,11 @@ func (diff *Diff) Stats() (*DiffStats, error) {
|
|||
return stats, nil
|
||||
}
|
||||
|
||||
type diffForEachData struct {
|
||||
FileCallback DiffForEachFileCallback
|
||||
HunkCallback DiffForEachHunkCallback
|
||||
LineCallback DiffForEachLineCallback
|
||||
Error error
|
||||
type diffForEachCallbackData struct {
|
||||
fileCallback DiffForEachFileCallback
|
||||
hunkCallback DiffForEachHunkCallback
|
||||
lineCallback DiffForEachLineCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
type DiffForEachFileCallback func(delta DiffDelta, progress float64) (DiffForEachHunkCallback, error)
|
||||
|
@ -326,82 +328,91 @@ func (diff *Diff) ForEach(cbFile DiffForEachFileCallback, detail DiffDetail) err
|
|||
intLines = C.int(1)
|
||||
}
|
||||
|
||||
data := &diffForEachData{
|
||||
FileCallback: cbFile,
|
||||
var err error
|
||||
data := &diffForEachCallbackData{
|
||||
fileCallback: cbFile,
|
||||
errorTarget: &err,
|
||||
}
|
||||
|
||||
handle := pointerHandles.Track(data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
|
||||
ecode := C._go_git_diff_foreach(diff.ptr, 1, intHunks, intLines, handle)
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C._go_git_diff_foreach(diff.ptr, 1, intHunks, intLines, handle)
|
||||
runtime.KeepAlive(diff)
|
||||
if ecode < 0 {
|
||||
return data.Error
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
//export diffForEachFileCb
|
||||
func diffForEachFileCb(delta *C.git_diff_delta, progress C.float, handle unsafe.Pointer) int {
|
||||
//export diffForEachFileCallback
|
||||
func diffForEachFileCallback(delta *C.git_diff_delta, progress C.float, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*diffForEachData)
|
||||
data, ok := payload.(*diffForEachCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve data for handle")
|
||||
}
|
||||
|
||||
data.HunkCallback = nil
|
||||
if data.FileCallback != nil {
|
||||
cb, err := data.FileCallback(diffDeltaFromC(delta), float64(progress))
|
||||
data.hunkCallback = nil
|
||||
if data.fileCallback != nil {
|
||||
cb, err := data.fileCallback(diffDeltaFromC(delta), float64(progress))
|
||||
if err != nil {
|
||||
data.Error = err
|
||||
return -1
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
data.HunkCallback = cb
|
||||
data.hunkCallback = cb
|
||||
}
|
||||
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
type DiffForEachHunkCallback func(DiffHunk) (DiffForEachLineCallback, error)
|
||||
|
||||
//export diffForEachHunkCb
|
||||
func diffForEachHunkCb(delta *C.git_diff_delta, hunk *C.git_diff_hunk, handle unsafe.Pointer) int {
|
||||
//export diffForEachHunkCallback
|
||||
func diffForEachHunkCallback(delta *C.git_diff_delta, hunk *C.git_diff_hunk, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*diffForEachData)
|
||||
data, ok := payload.(*diffForEachCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve data for handle")
|
||||
}
|
||||
|
||||
data.LineCallback = nil
|
||||
if data.HunkCallback != nil {
|
||||
cb, err := data.HunkCallback(diffHunkFromC(hunk))
|
||||
data.lineCallback = nil
|
||||
if data.hunkCallback != nil {
|
||||
cb, err := data.hunkCallback(diffHunkFromC(hunk))
|
||||
if err != nil {
|
||||
data.Error = err
|
||||
return -1
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
data.LineCallback = cb
|
||||
data.lineCallback = cb
|
||||
}
|
||||
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
type DiffForEachLineCallback func(DiffLine) error
|
||||
|
||||
//export diffForEachLineCb
|
||||
func diffForEachLineCb(delta *C.git_diff_delta, hunk *C.git_diff_hunk, line *C.git_diff_line, handle unsafe.Pointer) int {
|
||||
//export diffForEachLineCallback
|
||||
func diffForEachLineCallback(delta *C.git_diff_delta, hunk *C.git_diff_hunk, line *C.git_diff_line, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*diffForEachData)
|
||||
data, ok := payload.(*diffForEachCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve data for handle")
|
||||
}
|
||||
|
||||
err := data.LineCallback(diffLineFromC(line))
|
||||
err := data.lineCallback(diffLineFromC(line))
|
||||
if err != nil {
|
||||
data.Error = err
|
||||
return -1
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func (diff *Diff) Patch(deltaIndex int) (*Patch, error) {
|
||||
|
@ -447,7 +458,7 @@ func (diff *Diff) ToBuf(format DiffFormat) ([]byte, error) {
|
|||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
defer C.git_buf_free(&diffBuf)
|
||||
defer C.git_buf_dispose(&diffBuf)
|
||||
|
||||
return C.GoBytes(unsafe.Pointer(diffBuf.ptr), C.int(diffBuf.size)), nil
|
||||
}
|
||||
|
@ -478,7 +489,7 @@ const (
|
|||
|
||||
DiffIgnoreWhitespace DiffOptionsFlag = C.GIT_DIFF_IGNORE_WHITESPACE
|
||||
DiffIgnoreWhitespaceChange DiffOptionsFlag = C.GIT_DIFF_IGNORE_WHITESPACE_CHANGE
|
||||
DiffIgnoreWitespaceEol DiffOptionsFlag = C.GIT_DIFF_IGNORE_WHITESPACE_EOL
|
||||
DiffIgnoreWhitespaceEOL DiffOptionsFlag = C.GIT_DIFF_IGNORE_WHITESPACE_EOL
|
||||
|
||||
DiffShowUntrackedContent DiffOptionsFlag = C.GIT_DIFF_SHOW_UNTRACKED_CONTENT
|
||||
DiffShowUnmodified DiffOptionsFlag = C.GIT_DIFF_SHOW_UNMODIFIED
|
||||
|
@ -586,93 +597,92 @@ var (
|
|||
ErrDeltaSkip = errors.New("Skip delta")
|
||||
)
|
||||
|
||||
type diffNotifyData struct {
|
||||
Callback DiffNotifyCallback
|
||||
Repository *Repository
|
||||
Error error
|
||||
type diffNotifyCallbackData struct {
|
||||
callback DiffNotifyCallback
|
||||
repository *Repository
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export diffNotifyCb
|
||||
func diffNotifyCb(_diff_so_far unsafe.Pointer, delta_to_add *C.git_diff_delta, matched_pathspec *C.char, handle unsafe.Pointer) int {
|
||||
//export diffNotifyCallback
|
||||
func diffNotifyCallback(_diff_so_far unsafe.Pointer, delta_to_add *C.git_diff_delta, matched_pathspec *C.char, handle unsafe.Pointer) C.int {
|
||||
diff_so_far := (*C.git_diff)(_diff_so_far)
|
||||
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*diffNotifyData)
|
||||
data, ok := payload.(*diffNotifyCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve data for handle")
|
||||
}
|
||||
|
||||
if data != nil {
|
||||
// We are not taking ownership of this diff pointer, so no finalizer is set.
|
||||
diff := &Diff{
|
||||
ptr: diff_so_far,
|
||||
repo: data.Repository,
|
||||
runFinalizer: false,
|
||||
}
|
||||
|
||||
err := data.Callback(diff, diffDeltaFromC(delta_to_add), C.GoString(matched_pathspec))
|
||||
|
||||
// Since the callback could theoretically keep a reference to the diff
|
||||
// (which could be freed by libgit2 if an error occurs later during the
|
||||
// diffing process), this converts a use-after-free (terrible!) into a nil
|
||||
// dereference ("just" pretty bad).
|
||||
diff.ptr = nil
|
||||
|
||||
if err == ErrDeltaSkip {
|
||||
return 1
|
||||
} else if err != nil {
|
||||
data.Error = err
|
||||
return -1
|
||||
} else {
|
||||
return 0
|
||||
}
|
||||
if data == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
return 0
|
||||
|
||||
// We are not taking ownership of this diff pointer, so no finalizer is set.
|
||||
diff := &Diff{
|
||||
ptr: diff_so_far,
|
||||
repo: data.repository,
|
||||
runFinalizer: false,
|
||||
}
|
||||
|
||||
err := data.callback(diff, diffDeltaFromC(delta_to_add), C.GoString(matched_pathspec))
|
||||
|
||||
// Since the callback could theoretically keep a reference to the diff
|
||||
// (which could be freed by libgit2 if an error occurs later during the
|
||||
// diffing process), this converts a use-after-free (terrible!) into a nil
|
||||
// dereference ("just" pretty bad).
|
||||
diff.ptr = nil
|
||||
|
||||
if err == ErrDeltaSkip {
|
||||
return 1
|
||||
}
|
||||
if err != nil {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func diffOptionsToC(opts *DiffOptions, repo *Repository) (copts *C.git_diff_options) {
|
||||
cpathspec := C.git_strarray{}
|
||||
if opts != nil {
|
||||
notifyData := &diffNotifyData{
|
||||
Callback: opts.NotifyCallback,
|
||||
Repository: repo,
|
||||
}
|
||||
|
||||
if opts.Pathspec != nil {
|
||||
cpathspec.count = C.size_t(len(opts.Pathspec))
|
||||
cpathspec.strings = makeCStringsFromStrings(opts.Pathspec)
|
||||
}
|
||||
|
||||
copts = &C.git_diff_options{
|
||||
version: C.GIT_DIFF_OPTIONS_VERSION,
|
||||
flags: C.uint32_t(opts.Flags),
|
||||
ignore_submodules: C.git_submodule_ignore_t(opts.IgnoreSubmodules),
|
||||
pathspec: cpathspec,
|
||||
context_lines: C.uint32_t(opts.ContextLines),
|
||||
interhunk_lines: C.uint32_t(opts.InterhunkLines),
|
||||
id_abbrev: C.uint16_t(opts.IdAbbrev),
|
||||
max_size: C.git_off_t(opts.MaxSize),
|
||||
old_prefix: C.CString(opts.OldPrefix),
|
||||
new_prefix: C.CString(opts.NewPrefix),
|
||||
}
|
||||
|
||||
if opts.NotifyCallback != nil {
|
||||
C._go_git_setup_diff_notify_callbacks(copts)
|
||||
copts.payload = pointerHandles.Track(notifyData)
|
||||
}
|
||||
func populateDiffOptions(copts *C.git_diff_options, opts *DiffOptions, repo *Repository, errorTarget *error) *C.git_diff_options {
|
||||
C.git_diff_options_init(copts, C.GIT_DIFF_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
return
|
||||
|
||||
copts.flags = C.uint32_t(opts.Flags)
|
||||
copts.ignore_submodules = C.git_submodule_ignore_t(opts.IgnoreSubmodules)
|
||||
if len(opts.Pathspec) > 0 {
|
||||
copts.pathspec.count = C.size_t(len(opts.Pathspec))
|
||||
copts.pathspec.strings = makeCStringsFromStrings(opts.Pathspec)
|
||||
}
|
||||
copts.context_lines = C.uint32_t(opts.ContextLines)
|
||||
copts.interhunk_lines = C.uint32_t(opts.InterhunkLines)
|
||||
copts.id_abbrev = C.uint16_t(opts.IdAbbrev)
|
||||
copts.max_size = C.git_off_t(opts.MaxSize)
|
||||
copts.old_prefix = C.CString(opts.OldPrefix)
|
||||
copts.new_prefix = C.CString(opts.NewPrefix)
|
||||
|
||||
if opts.NotifyCallback != nil {
|
||||
notifyData := &diffNotifyCallbackData{
|
||||
callback: opts.NotifyCallback,
|
||||
repository: repo,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
C._go_git_setup_diff_notify_callbacks(copts)
|
||||
copts.payload = pointerHandles.Track(notifyData)
|
||||
}
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeDiffOptions(copts *C.git_diff_options) {
|
||||
if copts != nil {
|
||||
cpathspec := copts.pathspec
|
||||
freeStrarray(&cpathspec)
|
||||
C.free(unsafe.Pointer(copts.old_prefix))
|
||||
C.free(unsafe.Pointer(copts.new_prefix))
|
||||
if copts.payload != nil {
|
||||
pointerHandles.Untrack(copts.payload)
|
||||
}
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
freeStrarray(&copts.pathspec)
|
||||
C.free(unsafe.Pointer(copts.old_prefix))
|
||||
C.free(unsafe.Pointer(copts.new_prefix))
|
||||
if copts.payload != nil {
|
||||
pointerHandles.Untrack(copts.payload)
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -688,17 +698,21 @@ func (v *Repository) DiffTreeToTree(oldTree, newTree *Tree, opts *DiffOptions) (
|
|||
newPtr = newTree.cast_ptr
|
||||
}
|
||||
|
||||
copts := diffOptionsToC(opts, v)
|
||||
var err error
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, v, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_diff_tree_to_tree(&diffPtr, v.ptr, oldPtr, newPtr, copts)
|
||||
ret := C.git_diff_tree_to_tree(&diffPtr, v.ptr, oldPtr, newPtr, copts)
|
||||
runtime.KeepAlive(oldTree)
|
||||
runtime.KeepAlive(newTree)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
return newDiffFromC(diffPtr, v), nil
|
||||
}
|
||||
|
@ -711,17 +725,22 @@ func (v *Repository) DiffTreeToWorkdir(oldTree *Tree, opts *DiffOptions) (*Diff,
|
|||
oldPtr = oldTree.cast_ptr
|
||||
}
|
||||
|
||||
copts := diffOptionsToC(opts, v)
|
||||
var err error
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, v, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_diff_tree_to_workdir(&diffPtr, v.ptr, oldPtr, copts)
|
||||
ret := C.git_diff_tree_to_workdir(&diffPtr, v.ptr, oldPtr, copts)
|
||||
runtime.KeepAlive(oldTree)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newDiffFromC(diffPtr, v), nil
|
||||
}
|
||||
|
||||
|
@ -738,18 +757,23 @@ func (v *Repository) DiffTreeToIndex(oldTree *Tree, index *Index, opts *DiffOpti
|
|||
indexPtr = index.ptr
|
||||
}
|
||||
|
||||
copts := diffOptionsToC(opts, v)
|
||||
var err error
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, v, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_diff_tree_to_index(&diffPtr, v.ptr, oldPtr, indexPtr, copts)
|
||||
ret := C.git_diff_tree_to_index(&diffPtr, v.ptr, oldPtr, indexPtr, copts)
|
||||
runtime.KeepAlive(oldTree)
|
||||
runtime.KeepAlive(index)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newDiffFromC(diffPtr, v), nil
|
||||
}
|
||||
|
||||
|
@ -761,17 +785,22 @@ func (v *Repository) DiffTreeToWorkdirWithIndex(oldTree *Tree, opts *DiffOptions
|
|||
oldPtr = oldTree.cast_ptr
|
||||
}
|
||||
|
||||
copts := diffOptionsToC(opts, v)
|
||||
var err error
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, v, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_diff_tree_to_workdir_with_index(&diffPtr, v.ptr, oldPtr, copts)
|
||||
ret := C.git_diff_tree_to_workdir_with_index(&diffPtr, v.ptr, oldPtr, copts)
|
||||
runtime.KeepAlive(oldTree)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newDiffFromC(diffPtr, v), nil
|
||||
}
|
||||
|
||||
|
@ -783,25 +812,32 @@ func (v *Repository) DiffIndexToWorkdir(index *Index, opts *DiffOptions) (*Diff,
|
|||
indexPtr = index.ptr
|
||||
}
|
||||
|
||||
copts := diffOptionsToC(opts, v)
|
||||
var err error
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, v, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_diff_index_to_workdir(&diffPtr, v.ptr, indexPtr, copts)
|
||||
ret := C.git_diff_index_to_workdir(&diffPtr, v.ptr, indexPtr, copts)
|
||||
runtime.KeepAlive(index)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newDiffFromC(diffPtr, v), nil
|
||||
}
|
||||
|
||||
// DiffBlobs performs a diff between two arbitrary blobs. You can pass
|
||||
// whatever file names you'd like for them to appear as in the diff.
|
||||
func DiffBlobs(oldBlob *Blob, oldAsPath string, newBlob *Blob, newAsPath string, opts *DiffOptions, fileCallback DiffForEachFileCallback, detail DiffDetail) error {
|
||||
data := &diffForEachData{
|
||||
FileCallback: fileCallback,
|
||||
var err error
|
||||
data := &diffForEachCallbackData{
|
||||
fileCallback: fileCallback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
|
||||
intHunks := C.int(0)
|
||||
|
@ -833,17 +869,20 @@ func DiffBlobs(oldBlob *Blob, oldAsPath string, newBlob *Blob, newAsPath string,
|
|||
newBlobPath := C.CString(newAsPath)
|
||||
defer C.free(unsafe.Pointer(newBlobPath))
|
||||
|
||||
copts := diffOptionsToC(opts, repo)
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, repo, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C._go_git_diff_blobs(oldBlobPtr, oldBlobPath, newBlobPtr, newBlobPath, copts, 1, intHunks, intLines, handle)
|
||||
ret := C._go_git_diff_blobs(oldBlobPtr, oldBlobPath, newBlobPtr, newBlobPath, copts, 1, intHunks, intLines, handle)
|
||||
runtime.KeepAlive(oldBlob)
|
||||
runtime.KeepAlive(newBlob)
|
||||
if ecode < 0 {
|
||||
return MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
|
@ -864,56 +903,59 @@ type ApplyOptions struct {
|
|||
Flags uint
|
||||
}
|
||||
|
||||
type applyCallbackData struct {
|
||||
options *ApplyOptions
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export hunkApplyCallback
|
||||
func hunkApplyCallback(_hunk *C.git_diff_hunk, _payload unsafe.Pointer) C.int {
|
||||
opts, ok := pointerHandles.Get(_payload).(*ApplyOptions)
|
||||
data, ok := pointerHandles.Get(_payload).(*applyCallbackData)
|
||||
if !ok {
|
||||
panic("invalid apply options payload")
|
||||
}
|
||||
|
||||
if opts.ApplyHunkCallback == nil {
|
||||
return 0
|
||||
if data.options.ApplyHunkCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
hunk := diffHunkFromC(_hunk)
|
||||
|
||||
apply, err := opts.ApplyHunkCallback(&hunk)
|
||||
apply, err := data.options.ApplyHunkCallback(&hunk)
|
||||
if err != nil {
|
||||
if gitError, ok := err.(*GitError); ok {
|
||||
return C.int(gitError.Code)
|
||||
}
|
||||
return -1
|
||||
} else if apply {
|
||||
return 0
|
||||
} else {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
if !apply {
|
||||
return 1
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export deltaApplyCallback
|
||||
func deltaApplyCallback(_delta *C.git_diff_delta, _payload unsafe.Pointer) C.int {
|
||||
opts, ok := pointerHandles.Get(_payload).(*ApplyOptions)
|
||||
data, ok := pointerHandles.Get(_payload).(*applyCallbackData)
|
||||
if !ok {
|
||||
panic("invalid apply options payload")
|
||||
}
|
||||
|
||||
if opts.ApplyDeltaCallback == nil {
|
||||
return 0
|
||||
if data.options.ApplyDeltaCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
delta := diffDeltaFromC(_delta)
|
||||
|
||||
apply, err := opts.ApplyDeltaCallback(&delta)
|
||||
apply, err := data.options.ApplyDeltaCallback(&delta)
|
||||
if err != nil {
|
||||
if gitError, ok := err.(*GitError); ok {
|
||||
return C.int(gitError.Code)
|
||||
}
|
||||
return -1
|
||||
} else if apply {
|
||||
return 0
|
||||
} else {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
if !apply {
|
||||
return 1
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// DefaultApplyOptions returns default options for applying diffs or patches.
|
||||
|
@ -925,34 +967,44 @@ func DefaultApplyOptions() (*ApplyOptions, error) {
|
|||
|
||||
ecode := C.git_apply_options_init(&opts, C.GIT_APPLY_OPTIONS_VERSION)
|
||||
if int(ecode) != 0 {
|
||||
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
|
||||
return applyOptionsFromC(&opts), nil
|
||||
}
|
||||
|
||||
func (a *ApplyOptions) toC() *C.git_apply_options {
|
||||
if a == nil {
|
||||
func populateApplyOptions(copts *C.git_apply_options, opts *ApplyOptions, errorTarget *error) *C.git_apply_options {
|
||||
C.git_apply_options_init(copts, C.GIT_APPLY_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
opts := &C.git_apply_options{
|
||||
version: C.GIT_APPLY_OPTIONS_VERSION,
|
||||
flags: C.uint(a.Flags),
|
||||
copts.flags = C.uint(opts.Flags)
|
||||
|
||||
if opts.ApplyDeltaCallback != nil || opts.ApplyHunkCallback != nil {
|
||||
data := &applyCallbackData{
|
||||
options: opts,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
C._go_git_populate_apply_callbacks(copts)
|
||||
copts.payload = pointerHandles.Track(data)
|
||||
}
|
||||
|
||||
if a.ApplyDeltaCallback != nil || a.ApplyHunkCallback != nil {
|
||||
C._go_git_populate_apply_cb(opts)
|
||||
opts.payload = pointerHandles.Track(a)
|
||||
}
|
||||
|
||||
return opts
|
||||
return copts
|
||||
}
|
||||
|
||||
func applyOptionsFromC(opts *C.git_apply_options) *ApplyOptions {
|
||||
func freeApplyOptions(copts *C.git_apply_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
if copts.payload != nil {
|
||||
pointerHandles.Untrack(copts.payload)
|
||||
}
|
||||
}
|
||||
|
||||
func applyOptionsFromC(copts *C.git_apply_options) *ApplyOptions {
|
||||
return &ApplyOptions{
|
||||
Flags: uint(opts.flags),
|
||||
Flags: uint(copts.flags),
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -979,13 +1031,19 @@ func (v *Repository) ApplyDiff(diff *Diff, location ApplyLocation, opts *ApplyOp
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cOpts := opts.toC()
|
||||
ecode := C.git_apply(v.ptr, diff.ptr, C.git_apply_location_t(location), cOpts)
|
||||
var err error
|
||||
cOpts := populateApplyOptions(&C.git_apply_options{}, opts, &err)
|
||||
defer freeApplyOptions(cOpts)
|
||||
|
||||
ret := C.git_apply(v.ptr, diff.ptr, C.git_apply_location_t(location), cOpts)
|
||||
runtime.KeepAlive(v)
|
||||
runtime.KeepAlive(diff)
|
||||
runtime.KeepAlive(cOpts)
|
||||
if ecode < 0 {
|
||||
return MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
|
@ -996,14 +1054,20 @@ func (v *Repository) ApplyToTree(diff *Diff, tree *Tree, opts *ApplyOptions) (*I
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var err error
|
||||
cOpts := populateApplyOptions(&C.git_apply_options{}, opts, &err)
|
||||
defer freeApplyOptions(cOpts)
|
||||
|
||||
var indexPtr *C.git_index
|
||||
cOpts := opts.toC()
|
||||
ecode := C.git_apply_to_tree(&indexPtr, v.ptr, tree.cast_ptr, diff.ptr, cOpts)
|
||||
ret := C.git_apply_to_tree(&indexPtr, v.ptr, tree.cast_ptr, diff.ptr, cOpts)
|
||||
runtime.KeepAlive(diff)
|
||||
runtime.KeepAlive(tree)
|
||||
runtime.KeepAlive(cOpts)
|
||||
if ecode != 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newIndexFromC(indexPtr, v), nil
|
||||
|
|
41
diff_test.go
41
diff_test.go
|
@ -173,9 +173,8 @@ func TestDiffTreeToTree(t *testing.T) {
|
|||
}, DiffDetailLines)
|
||||
|
||||
if err != errTest {
|
||||
t.Fatal("Expected custom error to be returned")
|
||||
t.Fatalf("Expected custom error to be returned, got %v, want %v", err, errTest)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
func createTestTrees(t *testing.T, repo *Repository) (originalTree *Tree, newTree *Tree) {
|
||||
|
@ -259,7 +258,7 @@ func TestApplyDiffAddfile(t *testing.T) {
|
|||
defer diff.Free()
|
||||
|
||||
t.Run("check does not apply to current tree because file exists", func(t *testing.T) {
|
||||
err = repo.ResetToCommit(addSecondFileCommit, ResetHard, &CheckoutOpts{})
|
||||
err = repo.ResetToCommit(addSecondFileCommit, ResetHard, &CheckoutOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
err = repo.ApplyDiff(diff, ApplyLocationBoth, nil)
|
||||
|
@ -269,7 +268,7 @@ func TestApplyDiffAddfile(t *testing.T) {
|
|||
})
|
||||
|
||||
t.Run("check apply to correct commit", func(t *testing.T) {
|
||||
err = repo.ResetToCommit(addFirstFileCommit, ResetHard, &CheckoutOpts{})
|
||||
err = repo.ResetToCommit(addFirstFileCommit, ResetHard, &CheckoutOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
err = repo.ApplyDiff(diff, ApplyLocationBoth, nil)
|
||||
|
@ -324,7 +323,7 @@ func TestApplyDiffAddfile(t *testing.T) {
|
|||
})
|
||||
|
||||
t.Run("check convert to raw buffer and apply", func(t *testing.T) {
|
||||
err = repo.ResetToCommit(addFirstFileCommit, ResetHard, &CheckoutOpts{})
|
||||
err = repo.ResetToCommit(addFirstFileCommit, ResetHard, &CheckoutOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
raw, err := diff.ToBuf(DiffFormatPatch)
|
||||
|
@ -345,7 +344,7 @@ func TestApplyDiffAddfile(t *testing.T) {
|
|||
t.Run("check apply callbacks work", func(t *testing.T) {
|
||||
// reset the state and get new default options for test
|
||||
resetAndGetOpts := func(t *testing.T) *ApplyOptions {
|
||||
err = repo.ResetToCommit(addFirstFileCommit, ResetHard, &CheckoutOpts{})
|
||||
err = repo.ResetToCommit(addFirstFileCommit, ResetHard, &CheckoutOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
opts, err := DefaultApplyOptions()
|
||||
|
@ -486,13 +485,15 @@ func TestApplyToTree(t *testing.T) {
|
|||
diffAC, err := repo.DiffTreeToTree(treeA, treeC, nil)
|
||||
checkFatal(t, err)
|
||||
|
||||
errMessageDropped := errors.New("message dropped")
|
||||
|
||||
for _, tc := range []struct {
|
||||
name string
|
||||
tree *Tree
|
||||
diff *Diff
|
||||
applyHunkCallback ApplyHunkCallback
|
||||
applyDeltaCallback ApplyDeltaCallback
|
||||
error error
|
||||
err error
|
||||
expectedDiff *Diff
|
||||
}{
|
||||
{
|
||||
|
@ -505,10 +506,10 @@ func TestApplyToTree(t *testing.T) {
|
|||
name: "applying a conflicting patch errors",
|
||||
tree: treeB,
|
||||
diff: diffAC,
|
||||
error: &GitError{
|
||||
err: &GitError{
|
||||
Message: "hunk at line 1 did not apply",
|
||||
Code: ErrApplyFail,
|
||||
Class: ErrClassPatch,
|
||||
Code: ErrorCodeApplyFail,
|
||||
Class: ErrorClassPatch,
|
||||
},
|
||||
},
|
||||
{
|
||||
|
@ -529,11 +530,8 @@ func TestApplyToTree(t *testing.T) {
|
|||
name: "hunk callback erroring fails the call",
|
||||
tree: treeA,
|
||||
diff: diffAB,
|
||||
applyHunkCallback: func(*DiffHunk) (bool, error) { return true, errors.New("message dropped") },
|
||||
error: &GitError{
|
||||
Code: ErrGeneric,
|
||||
Class: ErrClassInvalid,
|
||||
},
|
||||
applyHunkCallback: func(*DiffHunk) (bool, error) { return true, errMessageDropped },
|
||||
err: errMessageDropped,
|
||||
},
|
||||
{
|
||||
name: "delta callback returning false does not apply",
|
||||
|
@ -545,11 +543,8 @@ func TestApplyToTree(t *testing.T) {
|
|||
name: "delta callback erroring fails the call",
|
||||
tree: treeA,
|
||||
diff: diffAB,
|
||||
applyDeltaCallback: func(*DiffDelta) (bool, error) { return true, errors.New("message dropped") },
|
||||
error: &GitError{
|
||||
Code: ErrGeneric,
|
||||
Class: ErrClassInvalid,
|
||||
},
|
||||
applyDeltaCallback: func(*DiffDelta) (bool, error) { return true, errMessageDropped },
|
||||
err: errMessageDropped,
|
||||
},
|
||||
} {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
|
@ -560,9 +555,9 @@ func TestApplyToTree(t *testing.T) {
|
|||
opts.ApplyDeltaCallback = tc.applyDeltaCallback
|
||||
|
||||
index, err := repo.ApplyToTree(tc.diff, tc.tree, opts)
|
||||
if tc.error != nil {
|
||||
if !reflect.DeepEqual(err, tc.error) {
|
||||
t.Fatalf("expected error %q but got %q", tc.error, err)
|
||||
if tc.err != nil {
|
||||
if !reflect.DeepEqual(tc.err, err) {
|
||||
t.Fatalf("expected error %q but got %q", tc.err, err)
|
||||
}
|
||||
|
||||
return
|
||||
|
|
|
@ -0,0 +1,63 @@
|
|||
// Code generated by "stringer -type ErrorClass -trimprefix ErrorClass -tags static"; DO NOT EDIT.
|
||||
|
||||
package git
|
||||
|
||||
import "strconv"
|
||||
|
||||
func _() {
|
||||
// An "invalid array index" compiler error signifies that the constant values have changed.
|
||||
// Re-run the stringer command to generate them again.
|
||||
var x [1]struct{}
|
||||
_ = x[ErrorClassNone-0]
|
||||
_ = x[ErrorClassNoMemory-1]
|
||||
_ = x[ErrorClassOS-2]
|
||||
_ = x[ErrorClassInvalid-3]
|
||||
_ = x[ErrorClassReference-4]
|
||||
_ = x[ErrorClassZlib-5]
|
||||
_ = x[ErrorClassRepository-6]
|
||||
_ = x[ErrorClassConfig-7]
|
||||
_ = x[ErrorClassRegex-8]
|
||||
_ = x[ErrorClassOdb-9]
|
||||
_ = x[ErrorClassIndex-10]
|
||||
_ = x[ErrorClassObject-11]
|
||||
_ = x[ErrorClassNet-12]
|
||||
_ = x[ErrorClassTag-13]
|
||||
_ = x[ErrorClassTree-14]
|
||||
_ = x[ErrorClassIndexer-15]
|
||||
_ = x[ErrorClassSSL-16]
|
||||
_ = x[ErrorClassSubmodule-17]
|
||||
_ = x[ErrorClassThread-18]
|
||||
_ = x[ErrorClassStash-19]
|
||||
_ = x[ErrorClassCheckout-20]
|
||||
_ = x[ErrorClassFetchHead-21]
|
||||
_ = x[ErrorClassMerge-22]
|
||||
_ = x[ErrorClassSSH-23]
|
||||
_ = x[ErrorClassFilter-24]
|
||||
_ = x[ErrorClassRevert-25]
|
||||
_ = x[ErrorClassCallback-26]
|
||||
_ = x[ErrorClassRebase-29]
|
||||
_ = x[ErrorClassPatch-31]
|
||||
}
|
||||
|
||||
const (
|
||||
_ErrorClass_name_0 = "NoneNoMemoryOSInvalidReferenceZlibRepositoryConfigRegexOdbIndexObjectNetTagTreeIndexerSSLSubmoduleThreadStashCheckoutFetchHeadMergeSSHFilterRevertCallback"
|
||||
_ErrorClass_name_1 = "Rebase"
|
||||
_ErrorClass_name_2 = "Patch"
|
||||
)
|
||||
|
||||
var (
|
||||
_ErrorClass_index_0 = [...]uint8{0, 4, 12, 14, 21, 30, 34, 44, 50, 55, 58, 63, 69, 72, 75, 79, 86, 89, 98, 104, 109, 117, 126, 131, 134, 140, 146, 154}
|
||||
)
|
||||
|
||||
func (i ErrorClass) String() string {
|
||||
switch {
|
||||
case 0 <= i && i <= 26:
|
||||
return _ErrorClass_name_0[_ErrorClass_index_0[i]:_ErrorClass_index_0[i+1]]
|
||||
case i == 29:
|
||||
return _ErrorClass_name_1
|
||||
case i == 31:
|
||||
return _ErrorClass_name_2
|
||||
default:
|
||||
return "ErrorClass(" + strconv.FormatInt(int64(i), 10) + ")"
|
||||
}
|
||||
}
|
|
@ -0,0 +1,69 @@
|
|||
// Code generated by "stringer -type ErrorCode -trimprefix ErrorCode -tags static"; DO NOT EDIT.
|
||||
|
||||
package git
|
||||
|
||||
import "strconv"
|
||||
|
||||
func _() {
|
||||
// An "invalid array index" compiler error signifies that the constant values have changed.
|
||||
// Re-run the stringer command to generate them again.
|
||||
var x [1]struct{}
|
||||
_ = x[ErrorCodeOK-0]
|
||||
_ = x[ErrorCodeGeneric - -1]
|
||||
_ = x[ErrorCodeNotFound - -3]
|
||||
_ = x[ErrorCodeExists - -4]
|
||||
_ = x[ErrorCodeAmbiguous - -5]
|
||||
_ = x[ErrorCodeBuffs - -6]
|
||||
_ = x[ErrorCodeUser - -7]
|
||||
_ = x[ErrorCodeBareRepo - -8]
|
||||
_ = x[ErrorCodeUnbornBranch - -9]
|
||||
_ = x[ErrorCodeUnmerged - -10]
|
||||
_ = x[ErrorCodeNonFastForward - -11]
|
||||
_ = x[ErrorCodeInvalidSpec - -12]
|
||||
_ = x[ErrorCodeConflict - -13]
|
||||
_ = x[ErrorCodeLocked - -14]
|
||||
_ = x[ErrorCodeModified - -15]
|
||||
_ = x[ErrorCodeAuth - -16]
|
||||
_ = x[ErrorCodeCertificate - -17]
|
||||
_ = x[ErrorCodeApplied - -18]
|
||||
_ = x[ErrorCodePeel - -19]
|
||||
_ = x[ErrorCodeEOF - -20]
|
||||
_ = x[ErrorCodeInvalid - -21]
|
||||
_ = x[ErrorCodeUncommitted - -22]
|
||||
_ = x[ErrorCodeDirectory - -23]
|
||||
_ = x[ErrorCodeMergeConflict - -24]
|
||||
_ = x[ErrorCodePassthrough - -30]
|
||||
_ = x[ErrorCodeIterOver - -31]
|
||||
_ = x[ErrorCodeRetry - -32]
|
||||
_ = x[ErrorCodeMismatch - -33]
|
||||
_ = x[ErrorCodeIndexDirty - -34]
|
||||
_ = x[ErrorCodeApplyFail - -35]
|
||||
}
|
||||
|
||||
const (
|
||||
_ErrorCode_name_0 = "ApplyFailIndexDirtyMismatchRetryIterOverPassthrough"
|
||||
_ErrorCode_name_1 = "MergeConflictDirectoryUncommittedInvalidEOFPeelAppliedCertificateAuthModifiedLockedConflictInvalidSpecNonFastForwardUnmergedUnbornBranchBareRepoUserBuffsAmbiguousExistsNotFound"
|
||||
_ErrorCode_name_2 = "GenericOK"
|
||||
)
|
||||
|
||||
var (
|
||||
_ErrorCode_index_0 = [...]uint8{0, 9, 19, 27, 32, 40, 51}
|
||||
_ErrorCode_index_1 = [...]uint8{0, 13, 22, 33, 40, 43, 47, 54, 65, 69, 77, 83, 91, 102, 116, 124, 136, 144, 148, 153, 162, 168, 176}
|
||||
_ErrorCode_index_2 = [...]uint8{0, 7, 9}
|
||||
)
|
||||
|
||||
func (i ErrorCode) String() string {
|
||||
switch {
|
||||
case -35 <= i && i <= -30:
|
||||
i -= -35
|
||||
return _ErrorCode_name_0[_ErrorCode_index_0[i]:_ErrorCode_index_0[i+1]]
|
||||
case -24 <= i && i <= -3:
|
||||
i -= -24
|
||||
return _ErrorCode_name_1[_ErrorCode_index_1[i]:_ErrorCode_index_1[i+1]]
|
||||
case -1 <= i && i <= 0:
|
||||
i -= -1
|
||||
return _ErrorCode_name_2[_ErrorCode_index_2[i]:_ErrorCode_index_2[i+1]]
|
||||
default:
|
||||
return "ErrorCode(" + strconv.FormatInt(int64(i), 10) + ")"
|
||||
}
|
||||
}
|
|
@ -12,10 +12,10 @@ const (
|
|||
FeatureThreads Feature = C.GIT_FEATURE_THREADS
|
||||
|
||||
// libgit2 was built with HTTPS support built-in
|
||||
FeatureHttps Feature = C.GIT_FEATURE_HTTPS
|
||||
FeatureHTTPS Feature = C.GIT_FEATURE_HTTPS
|
||||
|
||||
// libgit2 was build with SSH support built-in
|
||||
FeatureSsh Feature = C.GIT_FEATURE_SSH
|
||||
FeatureSSH Feature = C.GIT_FEATURE_SSH
|
||||
|
||||
// libgit2 was built with nanosecond support for files
|
||||
FeatureNSec Feature = C.GIT_FEATURE_NSEC
|
||||
|
|
247
git.go
247
git.go
|
@ -14,111 +14,123 @@ import (
|
|||
"unsafe"
|
||||
)
|
||||
|
||||
//go:generate stringer -type ErrorClass -trimprefix ErrorClass -tags static
|
||||
type ErrorClass int
|
||||
|
||||
const (
|
||||
ErrClassNone ErrorClass = C.GITERR_NONE
|
||||
ErrClassNoMemory ErrorClass = C.GITERR_NOMEMORY
|
||||
ErrClassOs ErrorClass = C.GITERR_OS
|
||||
ErrClassInvalid ErrorClass = C.GITERR_INVALID
|
||||
ErrClassReference ErrorClass = C.GITERR_REFERENCE
|
||||
ErrClassZlib ErrorClass = C.GITERR_ZLIB
|
||||
ErrClassRepository ErrorClass = C.GITERR_REPOSITORY
|
||||
ErrClassConfig ErrorClass = C.GITERR_CONFIG
|
||||
ErrClassRegex ErrorClass = C.GITERR_REGEX
|
||||
ErrClassOdb ErrorClass = C.GITERR_ODB
|
||||
ErrClassIndex ErrorClass = C.GITERR_INDEX
|
||||
ErrClassObject ErrorClass = C.GITERR_OBJECT
|
||||
ErrClassNet ErrorClass = C.GITERR_NET
|
||||
ErrClassTag ErrorClass = C.GITERR_TAG
|
||||
ErrClassTree ErrorClass = C.GITERR_TREE
|
||||
ErrClassIndexer ErrorClass = C.GITERR_INDEXER
|
||||
ErrClassSSL ErrorClass = C.GITERR_SSL
|
||||
ErrClassSubmodule ErrorClass = C.GITERR_SUBMODULE
|
||||
ErrClassThread ErrorClass = C.GITERR_THREAD
|
||||
ErrClassStash ErrorClass = C.GITERR_STASH
|
||||
ErrClassCheckout ErrorClass = C.GITERR_CHECKOUT
|
||||
ErrClassFetchHead ErrorClass = C.GITERR_FETCHHEAD
|
||||
ErrClassMerge ErrorClass = C.GITERR_MERGE
|
||||
ErrClassSsh ErrorClass = C.GITERR_SSH
|
||||
ErrClassFilter ErrorClass = C.GITERR_FILTER
|
||||
ErrClassRevert ErrorClass = C.GITERR_REVERT
|
||||
ErrClassCallback ErrorClass = C.GITERR_CALLBACK
|
||||
ErrClassRebase ErrorClass = C.GITERR_REBASE
|
||||
ErrClassPatch ErrorClass = C.GITERR_PATCH
|
||||
ErrorClassNone ErrorClass = C.GIT_ERROR_NONE
|
||||
ErrorClassNoMemory ErrorClass = C.GIT_ERROR_NOMEMORY
|
||||
ErrorClassOS ErrorClass = C.GIT_ERROR_OS
|
||||
ErrorClassInvalid ErrorClass = C.GIT_ERROR_INVALID
|
||||
ErrorClassReference ErrorClass = C.GIT_ERROR_REFERENCE
|
||||
ErrorClassZlib ErrorClass = C.GIT_ERROR_ZLIB
|
||||
ErrorClassRepository ErrorClass = C.GIT_ERROR_REPOSITORY
|
||||
ErrorClassConfig ErrorClass = C.GIT_ERROR_CONFIG
|
||||
ErrorClassRegex ErrorClass = C.GIT_ERROR_REGEX
|
||||
ErrorClassOdb ErrorClass = C.GIT_ERROR_ODB
|
||||
ErrorClassIndex ErrorClass = C.GIT_ERROR_INDEX
|
||||
ErrorClassObject ErrorClass = C.GIT_ERROR_OBJECT
|
||||
ErrorClassNet ErrorClass = C.GIT_ERROR_NET
|
||||
ErrorClassTag ErrorClass = C.GIT_ERROR_TAG
|
||||
ErrorClassTree ErrorClass = C.GIT_ERROR_TREE
|
||||
ErrorClassIndexer ErrorClass = C.GIT_ERROR_INDEXER
|
||||
ErrorClassSSL ErrorClass = C.GIT_ERROR_SSL
|
||||
ErrorClassSubmodule ErrorClass = C.GIT_ERROR_SUBMODULE
|
||||
ErrorClassThread ErrorClass = C.GIT_ERROR_THREAD
|
||||
ErrorClassStash ErrorClass = C.GIT_ERROR_STASH
|
||||
ErrorClassCheckout ErrorClass = C.GIT_ERROR_CHECKOUT
|
||||
ErrorClassFetchHead ErrorClass = C.GIT_ERROR_FETCHHEAD
|
||||
ErrorClassMerge ErrorClass = C.GIT_ERROR_MERGE
|
||||
ErrorClassSSH ErrorClass = C.GIT_ERROR_SSH
|
||||
ErrorClassFilter ErrorClass = C.GIT_ERROR_FILTER
|
||||
ErrorClassRevert ErrorClass = C.GIT_ERROR_REVERT
|
||||
ErrorClassCallback ErrorClass = C.GIT_ERROR_CALLBACK
|
||||
ErrorClassRebase ErrorClass = C.GIT_ERROR_REBASE
|
||||
ErrorClassPatch ErrorClass = C.GIT_ERROR_PATCH
|
||||
)
|
||||
|
||||
//go:generate stringer -type ErrorCode -trimprefix ErrorCode -tags static
|
||||
type ErrorCode int
|
||||
|
||||
const (
|
||||
// ErrorCodeOK indicates that the operation completed successfully.
|
||||
ErrorCodeOK ErrorCode = C.GIT_OK
|
||||
|
||||
// No error
|
||||
ErrOk ErrorCode = C.GIT_OK
|
||||
// ErrorCodeGeneric represents a generic error.
|
||||
ErrorCodeGeneric ErrorCode = C.GIT_ERROR
|
||||
// ErrorCodeNotFound represents that the requested object could not be found
|
||||
ErrorCodeNotFound ErrorCode = C.GIT_ENOTFOUND
|
||||
// ErrorCodeExists represents that the object exists preventing operation.
|
||||
ErrorCodeExists ErrorCode = C.GIT_EEXISTS
|
||||
// ErrorCodeAmbiguous represents that more than one object matches.
|
||||
ErrorCodeAmbiguous ErrorCode = C.GIT_EAMBIGUOUS
|
||||
// ErrorCodeBuffs represents that the output buffer is too short to hold data.
|
||||
ErrorCodeBuffs ErrorCode = C.GIT_EBUFS
|
||||
|
||||
// Generic error
|
||||
ErrGeneric ErrorCode = C.GIT_ERROR
|
||||
// Requested object could not be found
|
||||
ErrNotFound ErrorCode = C.GIT_ENOTFOUND
|
||||
// Object exists preventing operation
|
||||
ErrExists ErrorCode = C.GIT_EEXISTS
|
||||
// More than one object matches
|
||||
ErrAmbiguous ErrorCode = C.GIT_EAMBIGUOUS
|
||||
// (backwards compatibility misspelling)
|
||||
ErrAmbigious ErrorCode = C.GIT_EAMBIGUOUS
|
||||
// Output buffer too short to hold data
|
||||
ErrBuffs ErrorCode = C.GIT_EBUFS
|
||||
|
||||
// GIT_EUSER is a special error that is never generated by libgit2
|
||||
// ErrorCodeUser is a special error that is never generated by libgit2
|
||||
// code. You can return it from a callback (e.g to stop an iteration)
|
||||
// to know that it was generated by the callback and not by libgit2.
|
||||
ErrUser ErrorCode = C.GIT_EUSER
|
||||
ErrorCodeUser ErrorCode = C.GIT_EUSER
|
||||
|
||||
// Operation not allowed on bare repository
|
||||
ErrBareRepo ErrorCode = C.GIT_EBAREREPO
|
||||
// HEAD refers to branch with no commits
|
||||
ErrUnbornBranch ErrorCode = C.GIT_EUNBORNBRANCH
|
||||
// Merge in progress prevented operation
|
||||
ErrUnmerged ErrorCode = C.GIT_EUNMERGED
|
||||
// Reference was not fast-forwardable
|
||||
ErrNonFastForward ErrorCode = C.GIT_ENONFASTFORWARD
|
||||
// Name/ref spec was not in a valid format
|
||||
ErrInvalidSpec ErrorCode = C.GIT_EINVALIDSPEC
|
||||
// Checkout conflicts prevented operation
|
||||
ErrConflict ErrorCode = C.GIT_ECONFLICT
|
||||
// Lock file prevented operation
|
||||
ErrLocked ErrorCode = C.GIT_ELOCKED
|
||||
// Reference value does not match expected
|
||||
ErrModified ErrorCode = C.GIT_EMODIFIED
|
||||
// Authentication failed
|
||||
ErrAuth ErrorCode = C.GIT_EAUTH
|
||||
// Server certificate is invalid
|
||||
ErrCertificate ErrorCode = C.GIT_ECERTIFICATE
|
||||
// Patch/merge has already been applied
|
||||
ErrApplied ErrorCode = C.GIT_EAPPLIED
|
||||
// The requested peel operation is not possible
|
||||
ErrPeel ErrorCode = C.GIT_EPEEL
|
||||
// Unexpected EOF
|
||||
ErrEOF ErrorCode = C.GIT_EEOF
|
||||
// Uncommitted changes in index prevented operation
|
||||
ErrUncommitted ErrorCode = C.GIT_EUNCOMMITTED
|
||||
// The operation is not valid for a directory
|
||||
ErrDirectory ErrorCode = C.GIT_EDIRECTORY
|
||||
// A merge conflict exists and cannot continue
|
||||
ErrMergeConflict ErrorCode = C.GIT_EMERGECONFLICT
|
||||
// ErrorCodeBareRepo represents that the operation not allowed on bare repository
|
||||
ErrorCodeBareRepo ErrorCode = C.GIT_EBAREREPO
|
||||
// ErrorCodeUnbornBranch represents that HEAD refers to branch with no commits.
|
||||
ErrorCodeUnbornBranch ErrorCode = C.GIT_EUNBORNBRANCH
|
||||
// ErrorCodeUnmerged represents that a merge in progress prevented operation.
|
||||
ErrorCodeUnmerged ErrorCode = C.GIT_EUNMERGED
|
||||
// ErrorCodeNonFastForward represents that the reference was not fast-forwardable.
|
||||
ErrorCodeNonFastForward ErrorCode = C.GIT_ENONFASTFORWARD
|
||||
// ErrorCodeInvalidSpec represents that the name/ref spec was not in a valid format.
|
||||
ErrorCodeInvalidSpec ErrorCode = C.GIT_EINVALIDSPEC
|
||||
// ErrorCodeConflict represents that checkout conflicts prevented operation.
|
||||
ErrorCodeConflict ErrorCode = C.GIT_ECONFLICT
|
||||
// ErrorCodeLocked represents that lock file prevented operation.
|
||||
ErrorCodeLocked ErrorCode = C.GIT_ELOCKED
|
||||
// ErrorCodeModified represents that the reference value does not match expected.
|
||||
ErrorCodeModified ErrorCode = C.GIT_EMODIFIED
|
||||
// ErrorCodeAuth represents that the authentication failed.
|
||||
ErrorCodeAuth ErrorCode = C.GIT_EAUTH
|
||||
// ErrorCodeCertificate represents that the server certificate is invalid.
|
||||
ErrorCodeCertificate ErrorCode = C.GIT_ECERTIFICATE
|
||||
// ErrorCodeApplied represents that the patch/merge has already been applied.
|
||||
ErrorCodeApplied ErrorCode = C.GIT_EAPPLIED
|
||||
// ErrorCodePeel represents that the requested peel operation is not possible.
|
||||
ErrorCodePeel ErrorCode = C.GIT_EPEEL
|
||||
// ErrorCodeEOF represents an unexpected EOF.
|
||||
ErrorCodeEOF ErrorCode = C.GIT_EEOF
|
||||
// ErrorCodeInvalid represents an invalid operation or input.
|
||||
ErrorCodeInvalid ErrorCode = C.GIT_EINVALID
|
||||
// ErrorCodeUIncommitted represents that uncommitted changes in index prevented operation.
|
||||
ErrorCodeUncommitted ErrorCode = C.GIT_EUNCOMMITTED
|
||||
// ErrorCodeDirectory represents that the operation is not valid for a directory.
|
||||
ErrorCodeDirectory ErrorCode = C.GIT_EDIRECTORY
|
||||
// ErrorCodeMergeConflict represents that a merge conflict exists and cannot continue.
|
||||
ErrorCodeMergeConflict ErrorCode = C.GIT_EMERGECONFLICT
|
||||
|
||||
// Internal only
|
||||
ErrPassthrough ErrorCode = C.GIT_PASSTHROUGH
|
||||
// Signals end of iteration with iterator
|
||||
ErrIterOver ErrorCode = C.GIT_ITEROVER
|
||||
// Patch application failed
|
||||
ErrApplyFail ErrorCode = C.GIT_EAPPLYFAIL
|
||||
// ErrorCodePassthrough represents that a user-configured callback refused to act.
|
||||
ErrorCodePassthrough ErrorCode = C.GIT_PASSTHROUGH
|
||||
// ErrorCodeIterOver signals end of iteration with iterator.
|
||||
ErrorCodeIterOver ErrorCode = C.GIT_ITEROVER
|
||||
// ErrorCodeRetry is an internal-only error code.
|
||||
ErrorCodeRetry ErrorCode = C.GIT_RETRY
|
||||
// ErrorCodeMismatch represents a hashsum mismatch in object.
|
||||
ErrorCodeMismatch ErrorCode = C.GIT_EMISMATCH
|
||||
// ErrorCodeIndexDirty represents that unsaved changes in the index would be overwritten.
|
||||
ErrorCodeIndexDirty ErrorCode = C.GIT_EINDEXDIRTY
|
||||
// ErrorCodeApplyFail represents that a patch application failed.
|
||||
ErrorCodeApplyFail ErrorCode = C.GIT_EAPPLYFAIL
|
||||
)
|
||||
|
||||
var (
|
||||
ErrInvalid = errors.New("Invalid state for operation")
|
||||
)
|
||||
|
||||
// doNotCompare is an idiomatic way of making structs non-comparable to avoid
|
||||
// future field additions to make them non-comparable.
|
||||
type doNotCompare [0]func()
|
||||
|
||||
var pointerHandles *HandleList
|
||||
var remotePointers *remotePointerList
|
||||
|
||||
func init() {
|
||||
initLibGit2()
|
||||
|
@ -126,32 +138,48 @@ func init() {
|
|||
|
||||
func initLibGit2() {
|
||||
pointerHandles = NewHandleList()
|
||||
remotePointers = newRemotePointerList()
|
||||
|
||||
C.git_libgit2_init()
|
||||
features := Features()
|
||||
|
||||
// Due to the multithreaded nature of Go and its interaction with
|
||||
// calling C functions, we cannot work with a library that was not built
|
||||
// with multi-threading support. The most likely outcome is a segfault
|
||||
// or panic at an incomprehensible time, so let's make it easy by
|
||||
// panicking right here.
|
||||
if Features()&FeatureThreads == 0 {
|
||||
if features&FeatureThreads == 0 {
|
||||
panic("libgit2 was not built with threading support")
|
||||
}
|
||||
|
||||
// This is not something we should be doing, as we may be
|
||||
// stomping all over someone else's setup. The user should do
|
||||
// this themselves or use some binding/wrapper which does it
|
||||
// in such a way that they can be sure they're the only ones
|
||||
// setting it up.
|
||||
C.git_openssl_set_locking()
|
||||
if features&FeatureHTTPS == 0 {
|
||||
if err := registerManagedHTTP(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
} else {
|
||||
// This is not something we should be doing, as we may be stomping all over
|
||||
// someone else's setup. The user should do this themselves or use some
|
||||
// binding/wrapper which does it in such a way that they can be sure
|
||||
// they're the only ones setting it up.
|
||||
C.git_openssl_set_locking()
|
||||
}
|
||||
if features&FeatureSSH == 0 {
|
||||
if err := registerManagedSSH(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Shutdown frees all the resources acquired by libgit2. Make sure no
|
||||
// references to any git2go objects are live before calling this.
|
||||
// references to any libgit2 go objects are live before calling this.
|
||||
// After this is called, invoking any function from this library will result in
|
||||
// undefined behavior, so make sure this is called carefully.
|
||||
func Shutdown() {
|
||||
if err := unregisterManagedTransports(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
pointerHandles.Clear()
|
||||
remotePointers.clear()
|
||||
|
||||
C.git_libgit2_shutdown()
|
||||
}
|
||||
|
@ -195,13 +223,13 @@ func NewOid(s string) (*Oid, error) {
|
|||
|
||||
o := new(Oid)
|
||||
|
||||
slice, error := hex.DecodeString(s)
|
||||
if error != nil {
|
||||
return nil, error
|
||||
slice, err := hex.DecodeString(s)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if len(slice) != 20 {
|
||||
return nil, &GitError{"Invalid Oid", ErrClassNone, ErrGeneric}
|
||||
return nil, &GitError{"invalid oid", ErrorClassNone, ErrorCodeGeneric}
|
||||
}
|
||||
|
||||
copy(o[:], slice[:20])
|
||||
|
@ -269,7 +297,6 @@ func (e GitError) Error() string {
|
|||
}
|
||||
|
||||
func IsErrorClass(err error, c ErrorClass) bool {
|
||||
|
||||
if err == nil {
|
||||
return false
|
||||
}
|
||||
|
@ -289,20 +316,23 @@ func IsErrorCode(err error, c ErrorCode) bool {
|
|||
return false
|
||||
}
|
||||
|
||||
func MakeGitError(errorCode C.int) error {
|
||||
|
||||
func MakeGitError(c C.int) error {
|
||||
var errMessage string
|
||||
var errClass ErrorClass
|
||||
if errorCode != C.GIT_ITEROVER {
|
||||
err := C.giterr_last()
|
||||
errorCode := ErrorCode(c)
|
||||
if errorCode != ErrorCodeIterOver {
|
||||
err := C.git_error_last()
|
||||
if err != nil {
|
||||
errMessage = C.GoString(err.message)
|
||||
errClass = ErrorClass(err.klass)
|
||||
} else {
|
||||
errClass = ErrClassInvalid
|
||||
errClass = ErrorClassInvalid
|
||||
}
|
||||
}
|
||||
return &GitError{errMessage, errClass, ErrorCode(errorCode)}
|
||||
if errMessage == "" {
|
||||
errMessage = errorCode.String()
|
||||
}
|
||||
return &GitError{errMessage, errClass, errorCode}
|
||||
}
|
||||
|
||||
func MakeGitError2(err int) error {
|
||||
|
@ -323,6 +353,17 @@ func ucbool(b bool) C.uint {
|
|||
return C.uint(0)
|
||||
}
|
||||
|
||||
func setCallbackError(errorMessage **C.char, err error) C.int {
|
||||
if err != nil {
|
||||
*errorMessage = C.CString(err.Error())
|
||||
if gitError, ok := err.(*GitError); ok {
|
||||
return C.int(gitError.Code)
|
||||
}
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func Discover(start string, across_fs bool, ceiling_dirs []string) (string, error) {
|
||||
ceildirs := C.CString(strings.Join(ceiling_dirs, string(C.GIT_PATH_LIST_SEPARATOR)))
|
||||
defer C.free(unsafe.Pointer(ceildirs))
|
||||
|
|
|
@ -1,14 +0,0 @@
|
|||
// +build !static
|
||||
|
||||
package git
|
||||
|
||||
/*
|
||||
#cgo pkg-config: libgit2
|
||||
#cgo CFLAGS: -DLIBGIT2_DYNAMIC
|
||||
#include <git2.h>
|
||||
|
||||
#if LIBGIT2_VER_MAJOR != 1 || LIBGIT2_VER_MINOR != 0
|
||||
# error "Invalid libgit2 version; this git2go supports libgit2 v1.0"
|
||||
#endif
|
||||
*/
|
||||
import "C"
|
|
@ -1,14 +0,0 @@
|
|||
// +build static,system_libgit2
|
||||
|
||||
package git
|
||||
|
||||
/*
|
||||
#cgo pkg-config: libgit2 --static
|
||||
#cgo CFLAGS: -DLIBGIT2_STATIC
|
||||
#include <git2.h>
|
||||
|
||||
#if LIBGIT2_VER_MAJOR != 1 || LIBGIT2_VER_MINOR != 0
|
||||
# error "Invalid libgit2 version; this git2go supports libgit2 v1.0"
|
||||
#endif
|
||||
*/
|
||||
import "C"
|
48
git_test.go
48
git_test.go
|
@ -1,13 +1,51 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"path"
|
||||
"reflect"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestMain(m *testing.M) {
|
||||
if err := registerManagedHTTP(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
ret := m.Run()
|
||||
|
||||
if err := unregisterManagedTransports(); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
|
||||
// Ensure that we are not leaking any pointer handles.
|
||||
pointerHandles.Lock()
|
||||
if len(pointerHandles.handles) > 0 {
|
||||
for h, ptr := range pointerHandles.handles {
|
||||
fmt.Printf("%016p: %v %+v\n", h, reflect.TypeOf(ptr), ptr)
|
||||
}
|
||||
panic("pointer handle list not empty")
|
||||
}
|
||||
pointerHandles.Unlock()
|
||||
|
||||
// Or remote pointers.
|
||||
remotePointers.Lock()
|
||||
if len(remotePointers.pointers) > 0 {
|
||||
for ptr, remote := range remotePointers.pointers {
|
||||
fmt.Printf("%016p: %+v\n", ptr, remote)
|
||||
}
|
||||
panic("remote pointer list not empty")
|
||||
}
|
||||
remotePointers.Unlock()
|
||||
|
||||
Shutdown()
|
||||
|
||||
os.Exit(ret)
|
||||
}
|
||||
|
||||
func cleanupTestRepo(t *testing.T, r *Repository) {
|
||||
var err error
|
||||
if r.IsBare() {
|
||||
|
@ -45,16 +83,16 @@ func createBareTestRepo(t *testing.T) *Repository {
|
|||
return repo
|
||||
}
|
||||
|
||||
// commitOpts contains any extra options for creating commits in the seed repo
|
||||
type commitOpts struct {
|
||||
// commitOptions contains any extra options for creating commits in the seed repo
|
||||
type commitOptions struct {
|
||||
CommitSigningCallback
|
||||
}
|
||||
|
||||
func seedTestRepo(t *testing.T, repo *Repository) (*Oid, *Oid) {
|
||||
return seedTestRepoOpt(t, repo, commitOpts{})
|
||||
return seedTestRepoOpt(t, repo, commitOptions{})
|
||||
}
|
||||
|
||||
func seedTestRepoOpt(t *testing.T, repo *Repository, opts commitOpts) (*Oid, *Oid) {
|
||||
func seedTestRepoOpt(t *testing.T, repo *Repository, opts commitOptions) (*Oid, *Oid) {
|
||||
loc, err := time.LoadLocation("Europe/Berlin")
|
||||
checkFatal(t, err)
|
||||
sig := &Signature{
|
||||
|
@ -155,7 +193,7 @@ func TestOidZero(t *testing.T) {
|
|||
func TestEmptyOid(t *testing.T) {
|
||||
t.Parallel()
|
||||
_, err := NewOid("")
|
||||
if err == nil || !IsErrorCode(err, ErrGeneric) {
|
||||
if err == nil || !IsErrorCode(err, ErrorCodeGeneric) {
|
||||
t.Fatal("Should have returned invalid error")
|
||||
}
|
||||
}
|
||||
|
|
5
go.mod
5
go.mod
|
@ -1,5 +0,0 @@
|
|||
module github.com/libgit2/git2go/v30
|
||||
|
||||
go 1.13
|
||||
|
||||
require golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de
|
7
go.sum
7
go.sum
|
@ -1,7 +0,0 @@
|
|||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||
golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de h1:ikNHVSjEfnvz6sxdSPCaPt572qowuyMDMJLLm3Db3ig=
|
||||
golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
27
graph.go
27
graph.go
|
@ -40,3 +40,30 @@ func (repo *Repository) AheadBehind(local, upstream *Oid) (ahead, behind int, er
|
|||
|
||||
return int(aheadT), int(behindT), nil
|
||||
}
|
||||
|
||||
// ReachableFromAny returns whether a commit is reachable from any of a list of
|
||||
// commits by following parent edges.
|
||||
func (repo *Repository) ReachableFromAny(commit *Oid, descendants []*Oid) (bool, error) {
|
||||
if len(descendants) == 0 {
|
||||
return false, nil
|
||||
}
|
||||
|
||||
coids := make([]C.git_oid, len(descendants))
|
||||
for i := 0; i < len(descendants); i++ {
|
||||
coids[i] = *descendants[i].toC()
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_graph_reachable_from_any(repo.ptr, commit.toC(), &coids[0], C.size_t(len(descendants)))
|
||||
runtime.KeepAlive(repo)
|
||||
runtime.KeepAlive(commit)
|
||||
runtime.KeepAlive(coids)
|
||||
runtime.KeepAlive(descendants)
|
||||
if ret < 0 {
|
||||
return false, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return (ret > 0), nil
|
||||
}
|
||||
|
|
|
@ -0,0 +1,75 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestReachableFromAny(t *testing.T) {
|
||||
repo, err := OpenRepository("testdata/TestGitRepository.git")
|
||||
checkFatal(t, err)
|
||||
defer repo.Free()
|
||||
|
||||
for name, tc := range map[string]struct {
|
||||
reachable bool
|
||||
commit string
|
||||
descendants []string
|
||||
}{
|
||||
"empty": {
|
||||
reachable: false,
|
||||
commit: "49322bb17d3acc9146f98c97d078513228bbf3c0",
|
||||
},
|
||||
"same": {
|
||||
reachable: true,
|
||||
commit: "49322bb17d3acc9146f98c97d078513228bbf3c0",
|
||||
descendants: []string{"49322bb17d3acc9146f98c97d078513228bbf3c0"},
|
||||
},
|
||||
"unreachable": {
|
||||
reachable: false,
|
||||
commit: "ac7e7e44c1885efb472ad54a78327d66bfc4ecef",
|
||||
descendants: []string{"58be4659bb571194ed4562d04b359d26216f526e"},
|
||||
},
|
||||
"unreachable-reverse": {
|
||||
reachable: false,
|
||||
commit: "58be4659bb571194ed4562d04b359d26216f526e",
|
||||
descendants: []string{"ac7e7e44c1885efb472ad54a78327d66bfc4ecef"},
|
||||
},
|
||||
"root": {
|
||||
reachable: false,
|
||||
commit: "42e4e7c5e507e113ebbb7801b16b52cf867b7ce1",
|
||||
descendants: []string{
|
||||
"ac7e7e44c1885efb472ad54a78327d66bfc4ecef",
|
||||
"d86a2aada2f5e7ccf6f11880bfb9ab404e8a8864",
|
||||
"f73b95671f326616d66b2afb3bdfcdbbce110b44",
|
||||
"d0114ab8ac326bab30e3a657a0397578c5a1af88",
|
||||
},
|
||||
},
|
||||
"head": {
|
||||
reachable: false,
|
||||
commit: "49322bb17d3acc9146f98c97d078513228bbf3c0",
|
||||
descendants: []string{
|
||||
"ac7e7e44c1885efb472ad54a78327d66bfc4ecef",
|
||||
"d86a2aada2f5e7ccf6f11880bfb9ab404e8a8864",
|
||||
"f73b95671f326616d66b2afb3bdfcdbbce110b44",
|
||||
"d0114ab8ac326bab30e3a657a0397578c5a1af88",
|
||||
},
|
||||
},
|
||||
} {
|
||||
tc := tc
|
||||
t.Run(name, func(t *testing.T) {
|
||||
commit, err := NewOid(tc.commit)
|
||||
checkFatal(t, err)
|
||||
|
||||
descendants := make([]*Oid, len(tc.descendants))
|
||||
for i, o := range tc.descendants {
|
||||
descendants[i], err = NewOid(o)
|
||||
checkFatal(t, err)
|
||||
}
|
||||
reachable, err := repo.ReachableFromAny(commit, descendants)
|
||||
checkFatal(t, err)
|
||||
|
||||
if reachable != tc.reachable {
|
||||
t.Errorf("ReachableFromAny(%s, %v) = %v, wanted %v", tc.commit, tc.descendants, reachable, tc.reachable)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
|
@ -11,6 +11,7 @@ import (
|
|||
)
|
||||
|
||||
type HandleList struct {
|
||||
doNotCompare
|
||||
sync.RWMutex
|
||||
// stores the Go pointers
|
||||
handles map[unsafe.Pointer]interface{}
|
||||
|
|
|
@ -0,0 +1,243 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"sync"
|
||||
)
|
||||
|
||||
// RegisterManagedHTTPTransport registers a Go-native implementation of an
|
||||
// HTTP/S transport that doesn't rely on any system libraries (e.g.
|
||||
// libopenssl/libmbedtls).
|
||||
//
|
||||
// If Shutdown or ReInit are called, make sure that the smart transports are
|
||||
// freed before it.
|
||||
func RegisterManagedHTTPTransport(protocol string) (*RegisteredSmartTransport, error) {
|
||||
return NewRegisteredSmartTransport(protocol, true, httpSmartSubtransportFactory)
|
||||
}
|
||||
|
||||
func registerManagedHTTP() error {
|
||||
globalRegisteredSmartTransports.Lock()
|
||||
defer globalRegisteredSmartTransports.Unlock()
|
||||
|
||||
for _, protocol := range []string{"http", "https"} {
|
||||
if _, ok := globalRegisteredSmartTransports.transports[protocol]; ok {
|
||||
continue
|
||||
}
|
||||
managed, err := newRegisteredSmartTransport(protocol, true, httpSmartSubtransportFactory, true)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to register transport for %q: %v", protocol, err)
|
||||
}
|
||||
globalRegisteredSmartTransports.transports[protocol] = managed
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func httpSmartSubtransportFactory(remote *Remote, transport *Transport) (SmartSubtransport, error) {
|
||||
var proxyFn func(*http.Request) (*url.URL, error)
|
||||
remoteConnectOpts, err := transport.SmartRemoteConnectOptions()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
switch remoteConnectOpts.ProxyOptions.Type {
|
||||
case ProxyTypeNone:
|
||||
proxyFn = nil
|
||||
case ProxyTypeAuto:
|
||||
proxyFn = http.ProxyFromEnvironment
|
||||
case ProxyTypeSpecified:
|
||||
parsedUrl, err := url.Parse(remoteConnectOpts.ProxyOptions.Url)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
proxyFn = http.ProxyURL(parsedUrl)
|
||||
}
|
||||
|
||||
return &httpSmartSubtransport{
|
||||
transport: transport,
|
||||
client: &http.Client{
|
||||
Transport: &http.Transport{
|
||||
Proxy: proxyFn,
|
||||
},
|
||||
},
|
||||
}, nil
|
||||
}
|
||||
|
||||
type httpSmartSubtransport struct {
|
||||
transport *Transport
|
||||
client *http.Client
|
||||
}
|
||||
|
||||
func (t *httpSmartSubtransport) Action(url string, action SmartServiceAction) (SmartSubtransportStream, error) {
|
||||
var req *http.Request
|
||||
var err error
|
||||
switch action {
|
||||
case SmartServiceActionUploadpackLs:
|
||||
req, err = http.NewRequest("GET", url+"/info/refs?service=git-upload-pack", nil)
|
||||
|
||||
case SmartServiceActionUploadpack:
|
||||
req, err = http.NewRequest("POST", url+"/git-upload-pack", nil)
|
||||
if err != nil {
|
||||
break
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-git-upload-pack-request")
|
||||
|
||||
case SmartServiceActionReceivepackLs:
|
||||
req, err = http.NewRequest("GET", url+"/info/refs?service=git-receive-pack", nil)
|
||||
|
||||
case SmartServiceActionReceivepack:
|
||||
req, err = http.NewRequest("POST", url+"/info/refs?service=git-upload-pack", nil)
|
||||
if err != nil {
|
||||
break
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-git-receive-pack-request")
|
||||
|
||||
default:
|
||||
err = errors.New("unknown action")
|
||||
}
|
||||
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
req.Header.Set("User-Agent", "git/2.0 (libgit2)")
|
||||
|
||||
stream := newManagedHttpStream(t, req)
|
||||
if req.Method == "POST" {
|
||||
stream.recvReply.Add(1)
|
||||
stream.sendRequestBackground()
|
||||
}
|
||||
|
||||
return stream, nil
|
||||
}
|
||||
|
||||
func (t *httpSmartSubtransport) Close() error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func (t *httpSmartSubtransport) Free() {
|
||||
t.client = nil
|
||||
}
|
||||
|
||||
type httpSmartSubtransportStream struct {
|
||||
owner *httpSmartSubtransport
|
||||
req *http.Request
|
||||
resp *http.Response
|
||||
reader *io.PipeReader
|
||||
writer *io.PipeWriter
|
||||
sentRequest bool
|
||||
recvReply sync.WaitGroup
|
||||
httpError error
|
||||
}
|
||||
|
||||
func newManagedHttpStream(owner *httpSmartSubtransport, req *http.Request) *httpSmartSubtransportStream {
|
||||
r, w := io.Pipe()
|
||||
return &httpSmartSubtransportStream{
|
||||
owner: owner,
|
||||
req: req,
|
||||
reader: r,
|
||||
writer: w,
|
||||
}
|
||||
}
|
||||
|
||||
func (self *httpSmartSubtransportStream) Read(buf []byte) (int, error) {
|
||||
if !self.sentRequest {
|
||||
self.recvReply.Add(1)
|
||||
if err := self.sendRequest(); err != nil {
|
||||
return 0, err
|
||||
}
|
||||
}
|
||||
|
||||
if err := self.writer.Close(); err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
self.recvReply.Wait()
|
||||
|
||||
if self.httpError != nil {
|
||||
return 0, self.httpError
|
||||
}
|
||||
|
||||
return self.resp.Body.Read(buf)
|
||||
}
|
||||
|
||||
func (self *httpSmartSubtransportStream) Write(buf []byte) (int, error) {
|
||||
if self.httpError != nil {
|
||||
return 0, self.httpError
|
||||
}
|
||||
return self.writer.Write(buf)
|
||||
}
|
||||
|
||||
func (self *httpSmartSubtransportStream) Free() {
|
||||
if self.resp != nil {
|
||||
self.resp.Body.Close()
|
||||
}
|
||||
}
|
||||
|
||||
func (self *httpSmartSubtransportStream) sendRequestBackground() {
|
||||
go func() {
|
||||
self.httpError = self.sendRequest()
|
||||
}()
|
||||
self.sentRequest = true
|
||||
}
|
||||
|
||||
func (self *httpSmartSubtransportStream) sendRequest() error {
|
||||
defer self.recvReply.Done()
|
||||
self.resp = nil
|
||||
|
||||
var resp *http.Response
|
||||
var err error
|
||||
var userName string
|
||||
var password string
|
||||
for {
|
||||
req := &http.Request{
|
||||
Method: self.req.Method,
|
||||
URL: self.req.URL,
|
||||
Header: self.req.Header,
|
||||
}
|
||||
if req.Method == "POST" {
|
||||
req.Body = self.reader
|
||||
req.ContentLength = -1
|
||||
}
|
||||
|
||||
if userName != "" && password != "" {
|
||||
req.SetBasicAuth(userName, password)
|
||||
}
|
||||
resp, err = http.DefaultClient.Do(req)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if resp.StatusCode == http.StatusOK {
|
||||
break
|
||||
}
|
||||
|
||||
if resp.StatusCode == http.StatusUnauthorized {
|
||||
resp.Body.Close()
|
||||
|
||||
cred, err := self.owner.transport.SmartCredentials("", CredentialTypeUserpassPlaintext)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer cred.Free()
|
||||
|
||||
userName, password, err = cred.GetUserpassPlaintext()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
continue
|
||||
}
|
||||
|
||||
// Any other error we treat as a hard error and punt back to the caller
|
||||
resp.Body.Close()
|
||||
return fmt.Errorf("Unhandled HTTP error %s", resp.Status)
|
||||
}
|
||||
|
||||
self.sentRequest = true
|
||||
self.resp = resp
|
||||
return nil
|
||||
}
|
98
index.go
98
index.go
|
@ -15,43 +15,50 @@ import (
|
|||
"unsafe"
|
||||
)
|
||||
|
||||
type IndexMatchedPathCallback func(string, string) int
|
||||
type IndexMatchedPathCallback func(string, string) error
|
||||
type indexMatchedPathCallbackData struct {
|
||||
callback IndexMatchedPathCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
type IndexAddOpts uint
|
||||
// IndexAddOption is a set of flags for APIs that add files matching pathspec.
|
||||
type IndexAddOption uint
|
||||
|
||||
const (
|
||||
IndexAddDefault IndexAddOpts = C.GIT_INDEX_ADD_DEFAULT
|
||||
IndexAddForce IndexAddOpts = C.GIT_INDEX_ADD_FORCE
|
||||
IndexAddDisablePathspecMatch IndexAddOpts = C.GIT_INDEX_ADD_DISABLE_PATHSPEC_MATCH
|
||||
IndexAddCheckPathspec IndexAddOpts = C.GIT_INDEX_ADD_CHECK_PATHSPEC
|
||||
IndexAddDefault IndexAddOption = C.GIT_INDEX_ADD_DEFAULT
|
||||
IndexAddForce IndexAddOption = C.GIT_INDEX_ADD_FORCE
|
||||
IndexAddDisablePathspecMatch IndexAddOption = C.GIT_INDEX_ADD_DISABLE_PATHSPEC_MATCH
|
||||
IndexAddCheckPathspec IndexAddOption = C.GIT_INDEX_ADD_CHECK_PATHSPEC
|
||||
)
|
||||
|
||||
type IndexStageOpts int
|
||||
// IndexStageState indicates the state of the git index.
|
||||
type IndexStageState int
|
||||
|
||||
const (
|
||||
// IndexStageAny matches any index stage.
|
||||
//
|
||||
// Some index APIs take a stage to match; pass this value to match
|
||||
// any entry matching the path regardless of stage.
|
||||
IndexStageAny IndexStageOpts = C.GIT_INDEX_STAGE_ANY
|
||||
IndexStageAny IndexStageState = C.GIT_INDEX_STAGE_ANY
|
||||
// IndexStageNormal is a normal staged file in the index.
|
||||
IndexStageNormal IndexStageOpts = C.GIT_INDEX_STAGE_NORMAL
|
||||
IndexStageNormal IndexStageState = C.GIT_INDEX_STAGE_NORMAL
|
||||
// IndexStageAncestor is the ancestor side of a conflict.
|
||||
IndexStageAncestor IndexStageOpts = C.GIT_INDEX_STAGE_ANCESTOR
|
||||
IndexStageAncestor IndexStageState = C.GIT_INDEX_STAGE_ANCESTOR
|
||||
// IndexStageOurs is the "ours" side of a conflict.
|
||||
IndexStageOurs IndexStageOpts = C.GIT_INDEX_STAGE_OURS
|
||||
IndexStageOurs IndexStageState = C.GIT_INDEX_STAGE_OURS
|
||||
// IndexStageTheirs is the "theirs" side of a conflict.
|
||||
IndexStageTheirs IndexStageOpts = C.GIT_INDEX_STAGE_THEIRS
|
||||
IndexStageTheirs IndexStageState = C.GIT_INDEX_STAGE_THEIRS
|
||||
)
|
||||
|
||||
type Index struct {
|
||||
doNotCompare
|
||||
ptr *C.git_index
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
type IndexTime struct {
|
||||
seconds int32
|
||||
nanoseconds uint32
|
||||
Seconds int32
|
||||
Nanoseconds uint32
|
||||
}
|
||||
|
||||
type IndexEntry struct {
|
||||
|
@ -82,10 +89,10 @@ func newIndexEntryFromC(entry *C.git_index_entry) *IndexEntry {
|
|||
}
|
||||
|
||||
func populateCIndexEntry(source *IndexEntry, dest *C.git_index_entry) {
|
||||
dest.ctime.seconds = C.int32_t(source.Ctime.seconds)
|
||||
dest.ctime.nanoseconds = C.uint32_t(source.Ctime.nanoseconds)
|
||||
dest.mtime.seconds = C.int32_t(source.Mtime.seconds)
|
||||
dest.mtime.nanoseconds = C.uint32_t(source.Mtime.nanoseconds)
|
||||
dest.ctime.seconds = C.int32_t(source.Ctime.Seconds)
|
||||
dest.ctime.nanoseconds = C.uint32_t(source.Ctime.Nanoseconds)
|
||||
dest.mtime.seconds = C.int32_t(source.Mtime.Seconds)
|
||||
dest.mtime.nanoseconds = C.uint32_t(source.Mtime.Nanoseconds)
|
||||
dest.mode = C.uint32_t(source.Mode)
|
||||
dest.uid = C.uint32_t(source.Uid)
|
||||
dest.gid = C.uint32_t(source.Gid)
|
||||
|
@ -101,7 +108,7 @@ func freeCIndexEntry(entry *C.git_index_entry) {
|
|||
}
|
||||
|
||||
func newIndexFromC(ptr *C.git_index, repo *Repository) *Index {
|
||||
idx := &Index{ptr, repo}
|
||||
idx := &Index{ptr: ptr, repo: repo}
|
||||
runtime.SetFinalizer(idx, (*Index).Free)
|
||||
return idx
|
||||
}
|
||||
|
@ -219,18 +226,23 @@ func (v *Index) AddFromBuffer(entry *IndexEntry, buffer []byte) error {
|
|||
return nil
|
||||
}
|
||||
|
||||
func (v *Index) AddAll(pathspecs []string, flags IndexAddOpts, callback IndexMatchedPathCallback) error {
|
||||
func (v *Index) AddAll(pathspecs []string, flags IndexAddOption, callback IndexMatchedPathCallback) error {
|
||||
cpathspecs := C.git_strarray{}
|
||||
cpathspecs.count = C.size_t(len(pathspecs))
|
||||
cpathspecs.strings = makeCStringsFromStrings(pathspecs)
|
||||
defer freeStrarray(&cpathspecs)
|
||||
|
||||
var err error
|
||||
data := indexMatchedPathCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var handle unsafe.Pointer
|
||||
if callback != nil {
|
||||
handle = pointerHandles.Track(callback)
|
||||
handle = pointerHandles.Track(&data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
}
|
||||
|
||||
|
@ -241,9 +253,13 @@ func (v *Index) AddAll(pathspecs []string, flags IndexAddOpts, callback IndexMat
|
|||
handle,
|
||||
)
|
||||
runtime.KeepAlive(v)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
@ -253,12 +269,17 @@ func (v *Index) UpdateAll(pathspecs []string, callback IndexMatchedPathCallback)
|
|||
cpathspecs.strings = makeCStringsFromStrings(pathspecs)
|
||||
defer freeStrarray(&cpathspecs)
|
||||
|
||||
var err error
|
||||
data := indexMatchedPathCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var handle unsafe.Pointer
|
||||
if callback != nil {
|
||||
handle = pointerHandles.Track(callback)
|
||||
handle = pointerHandles.Track(&data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
}
|
||||
|
||||
|
@ -268,9 +289,13 @@ func (v *Index) UpdateAll(pathspecs []string, callback IndexMatchedPathCallback)
|
|||
handle,
|
||||
)
|
||||
runtime.KeepAlive(v)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
@ -280,12 +305,17 @@ func (v *Index) RemoveAll(pathspecs []string, callback IndexMatchedPathCallback)
|
|||
cpathspecs.strings = makeCStringsFromStrings(pathspecs)
|
||||
defer freeStrarray(&cpathspecs)
|
||||
|
||||
var err error
|
||||
data := indexMatchedPathCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var handle unsafe.Pointer
|
||||
if callback != nil {
|
||||
handle = pointerHandles.Track(callback)
|
||||
handle = pointerHandles.Track(&data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
}
|
||||
|
||||
|
@ -295,19 +325,30 @@ func (v *Index) RemoveAll(pathspecs []string, callback IndexMatchedPathCallback)
|
|||
handle,
|
||||
)
|
||||
runtime.KeepAlive(v)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
//export indexMatchedPathCallback
|
||||
func indexMatchedPathCallback(cPath, cMatchedPathspec *C.char, payload unsafe.Pointer) int {
|
||||
if callback, ok := pointerHandles.Get(payload).(IndexMatchedPathCallback); ok {
|
||||
return callback(C.GoString(cPath), C.GoString(cMatchedPathspec))
|
||||
} else {
|
||||
func indexMatchedPathCallback(cPath, cMatchedPathspec *C.char, payload unsafe.Pointer) C.int {
|
||||
data, ok := pointerHandles.Get(payload).(*indexMatchedPathCallbackData)
|
||||
if !ok {
|
||||
panic("invalid matched path callback")
|
||||
}
|
||||
|
||||
err := data.callback(C.GoString(cPath), C.GoString(cMatchedPathspec))
|
||||
if err != nil {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func (v *Index) RemoveByPath(path string) error {
|
||||
|
@ -433,7 +474,7 @@ func (v *Index) EntryByPath(path string, stage int) (*IndexEntry, error) {
|
|||
|
||||
centry := C.git_index_get_bypath(v.ptr, cpath, C.int(stage))
|
||||
if centry == nil {
|
||||
return nil, MakeGitError(C.GIT_ENOTFOUND)
|
||||
return nil, MakeGitError(C.int(ErrorCodeNotFound))
|
||||
}
|
||||
ret := newIndexEntryFromC(centry)
|
||||
runtime.KeepAlive(v)
|
||||
|
@ -575,6 +616,7 @@ func (v *Index) RemoveConflict(path string) error {
|
|||
}
|
||||
|
||||
type IndexConflictIterator struct {
|
||||
doNotCompare
|
||||
ptr *C.git_index_conflict_iterator
|
||||
index *Index
|
||||
}
|
||||
|
|
|
@ -223,9 +223,9 @@ func TestIndexAddAllCallback(t *testing.T) {
|
|||
checkFatal(t, err)
|
||||
|
||||
cbPath := ""
|
||||
err = idx.AddAll([]string{}, IndexAddDefault, func(p, mP string) int {
|
||||
err = idx.AddAll([]string{}, IndexAddDefault, func(p, mP string) error {
|
||||
cbPath = p
|
||||
return 0
|
||||
return nil
|
||||
})
|
||||
checkFatal(t, err)
|
||||
if cbPath != "README" {
|
||||
|
|
26
indexer.go
26
indexer.go
|
@ -19,34 +19,32 @@ import (
|
|||
// Indexer can post-process packfiles and create an .idx file for efficient
|
||||
// lookup.
|
||||
type Indexer struct {
|
||||
ptr *C.git_indexer
|
||||
stats C.git_transfer_progress
|
||||
callbacks RemoteCallbacks
|
||||
callbacksHandle unsafe.Pointer
|
||||
doNotCompare
|
||||
ptr *C.git_indexer
|
||||
stats C.git_transfer_progress
|
||||
ccallbacks C.git_remote_callbacks
|
||||
}
|
||||
|
||||
// NewIndexer creates a new indexer instance.
|
||||
func NewIndexer(packfilePath string, odb *Odb, callback TransferProgressCallback) (indexer *Indexer, err error) {
|
||||
indexer = new(Indexer)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var odbPtr *C.git_odb = nil
|
||||
if odb != nil {
|
||||
odbPtr = odb.ptr
|
||||
}
|
||||
|
||||
indexer.callbacks.TransferProgressCallback = callback
|
||||
indexer.callbacksHandle = pointerHandles.Track(&indexer.callbacks)
|
||||
indexer = new(Indexer)
|
||||
populateRemoteCallbacks(&indexer.ccallbacks, &RemoteCallbacks{TransferProgressCallback: callback}, nil)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cstr := C.CString(packfilePath)
|
||||
defer C.free(unsafe.Pointer(cstr))
|
||||
|
||||
ret := C._go_git_indexer_new(&indexer.ptr, cstr, 0, odbPtr, indexer.callbacksHandle)
|
||||
ret := C._go_git_indexer_new(&indexer.ptr, cstr, 0, odbPtr, indexer.ccallbacks.payload)
|
||||
runtime.KeepAlive(odb)
|
||||
if ret < 0 {
|
||||
pointerHandles.Untrack(indexer.callbacksHandle)
|
||||
untrackCallbacksPayload(&indexer.ccallbacks)
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
|
@ -93,7 +91,7 @@ func (indexer *Indexer) Commit() (*Oid, error) {
|
|||
|
||||
// Free frees the indexer and its resources.
|
||||
func (indexer *Indexer) Free() {
|
||||
pointerHandles.Untrack(indexer.callbacksHandle)
|
||||
untrackCallbacksPayload(&indexer.ccallbacks)
|
||||
runtime.SetFinalizer(indexer, nil)
|
||||
C.git_indexer_free(indexer.ptr)
|
||||
}
|
||||
|
|
|
@ -33,9 +33,9 @@ func TestIndexerOutOfOrder(t *testing.T) {
|
|||
defer os.RemoveAll(tmpPath)
|
||||
|
||||
var finalStats TransferProgress
|
||||
idx, err := NewIndexer(tmpPath, nil, func(stats TransferProgress) ErrorCode {
|
||||
idx, err := NewIndexer(tmpPath, nil, func(stats TransferProgress) error {
|
||||
finalStats = stats
|
||||
return ErrOk
|
||||
return nil
|
||||
})
|
||||
checkFatal(t, err)
|
||||
defer idx.Free()
|
||||
|
|
|
@ -18,6 +18,7 @@ import (
|
|||
|
||||
// Mempack is a custom ODB backend that permits packing object in-memory.
|
||||
type Mempack struct {
|
||||
doNotCompare
|
||||
ptr *C.git_odb_backend
|
||||
}
|
||||
|
||||
|
|
|
@ -53,7 +53,7 @@ func TestMempack(t *testing.T) {
|
|||
if err == nil {
|
||||
t.Errorf("object %s unexpectedly found", obj.Id().String())
|
||||
obj.Free()
|
||||
} else if !IsErrorCode(err, ErrNotFound) {
|
||||
} else if !IsErrorCode(err, ErrorCodeNotFound) {
|
||||
t.Errorf("unexpected error %v", err)
|
||||
}
|
||||
}
|
||||
|
|
164
merge.go
164
merge.go
|
@ -17,6 +17,7 @@ import (
|
|||
)
|
||||
|
||||
type AnnotatedCommit struct {
|
||||
doNotCompare
|
||||
ptr *C.git_annotated_commit
|
||||
r *Repository
|
||||
}
|
||||
|
@ -137,7 +138,6 @@ const (
|
|||
)
|
||||
|
||||
type MergeOptions struct {
|
||||
Version uint
|
||||
TreeFlags MergeTreeFlag
|
||||
|
||||
RenameThreshold uint
|
||||
|
@ -150,7 +150,6 @@ type MergeOptions struct {
|
|||
|
||||
func mergeOptionsFromC(opts *C.git_merge_options) MergeOptions {
|
||||
return MergeOptions{
|
||||
Version: uint(opts.version),
|
||||
TreeFlags: MergeTreeFlag(opts.flags),
|
||||
RenameThreshold: uint(opts.rename_threshold),
|
||||
TargetLimit: uint(opts.target_limit),
|
||||
|
@ -172,18 +171,20 @@ func DefaultMergeOptions() (MergeOptions, error) {
|
|||
return mergeOptionsFromC(&opts), nil
|
||||
}
|
||||
|
||||
func (mo *MergeOptions) toC() *C.git_merge_options {
|
||||
if mo == nil {
|
||||
func populateMergeOptions(copts *C.git_merge_options, opts *MergeOptions) *C.git_merge_options {
|
||||
C.git_merge_options_init(copts, C.GIT_MERGE_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
return &C.git_merge_options{
|
||||
version: C.uint(mo.Version),
|
||||
flags: C.uint32_t(mo.TreeFlags),
|
||||
rename_threshold: C.uint(mo.RenameThreshold),
|
||||
target_limit: C.uint(mo.TargetLimit),
|
||||
recursion_limit: C.uint(mo.RecursionLimit),
|
||||
file_favor: C.git_merge_file_favor_t(mo.FileFavor),
|
||||
}
|
||||
copts.flags = C.uint32_t(opts.TreeFlags)
|
||||
copts.rename_threshold = C.uint(opts.RenameThreshold)
|
||||
copts.target_limit = C.uint(opts.TargetLimit)
|
||||
copts.recursion_limit = C.uint(opts.RecursionLimit)
|
||||
copts.file_favor = C.git_merge_file_favor_t(opts.FileFavor)
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeMergeOptions(copts *C.git_merge_options) {
|
||||
}
|
||||
|
||||
type MergeFileFavor int
|
||||
|
@ -195,23 +196,28 @@ const (
|
|||
MergeFileFavorUnion MergeFileFavor = C.GIT_MERGE_FILE_FAVOR_UNION
|
||||
)
|
||||
|
||||
func (r *Repository) Merge(theirHeads []*AnnotatedCommit, mergeOptions *MergeOptions, checkoutOptions *CheckoutOpts) error {
|
||||
func (r *Repository) Merge(theirHeads []*AnnotatedCommit, mergeOptions *MergeOptions, checkoutOptions *CheckoutOptions) error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cMergeOpts := mergeOptions.toC()
|
||||
cCheckoutOpts := checkoutOptions.toC()
|
||||
defer freeCheckoutOpts(cCheckoutOpts)
|
||||
var err error
|
||||
cMergeOpts := populateMergeOptions(&C.git_merge_options{}, mergeOptions)
|
||||
defer freeMergeOptions(cMergeOpts)
|
||||
cCheckoutOptions := populateCheckoutOptions(&C.git_checkout_options{}, checkoutOptions, &err)
|
||||
defer freeCheckoutOptions(cCheckoutOptions)
|
||||
|
||||
gmerge_head_array := make([]*C.git_annotated_commit, len(theirHeads))
|
||||
for i := 0; i < len(theirHeads); i++ {
|
||||
gmerge_head_array[i] = theirHeads[i].ptr
|
||||
}
|
||||
ptr := unsafe.Pointer(&gmerge_head_array[0])
|
||||
err := C.git_merge(r.ptr, (**C.git_annotated_commit)(ptr), C.size_t(len(theirHeads)), cMergeOpts, cCheckoutOpts)
|
||||
ret := C.git_merge(r.ptr, (**C.git_annotated_commit)(ptr), C.size_t(len(theirHeads)), cMergeOpts, cCheckoutOptions)
|
||||
runtime.KeepAlive(theirHeads)
|
||||
if err < 0 {
|
||||
return MakeGitError(err)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
@ -261,7 +267,8 @@ func (r *Repository) MergeCommits(ours *Commit, theirs *Commit, options *MergeOp
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
copts := options.toC()
|
||||
copts := populateMergeOptions(&C.git_merge_options{}, options)
|
||||
defer freeMergeOptions(copts)
|
||||
|
||||
var ptr *C.git_index
|
||||
ret := C.git_merge_commits(&ptr, r.ptr, ours.cast_ptr, theirs.cast_ptr, copts)
|
||||
|
@ -278,7 +285,8 @@ func (r *Repository) MergeTrees(ancestor *Tree, ours *Tree, theirs *Tree, option
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
copts := options.toC()
|
||||
copts := populateMergeOptions(&C.git_merge_options{}, options)
|
||||
defer freeMergeOptions(copts)
|
||||
|
||||
var ancestor_ptr *C.git_tree
|
||||
if ancestor != nil {
|
||||
|
@ -324,7 +332,7 @@ func (r *Repository) MergeBases(one, two *Oid) ([]*Oid, error) {
|
|||
runtime.KeepAlive(one)
|
||||
runtime.KeepAlive(two)
|
||||
if ret < 0 {
|
||||
return make([]*Oid, 0), MakeGitError(ret)
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
oids := make([]*Oid, coids.count)
|
||||
|
@ -343,15 +351,86 @@ func (r *Repository) MergeBases(one, two *Oid) ([]*Oid, error) {
|
|||
return oids, nil
|
||||
}
|
||||
|
||||
//TODO: int git_merge_base_many(git_oid *out, git_repository *repo, size_t length, const git_oid input_array[]);
|
||||
//TODO: GIT_EXTERN(int) git_merge_base_octopus(git_oid *out,git_repository *repo,size_t length,const git_oid input_array[]);
|
||||
// MergeBaseMany finds a merge base given a list of commits.
|
||||
func (r *Repository) MergeBaseMany(oids []*Oid) (*Oid, error) {
|
||||
coids := make([]C.git_oid, len(oids))
|
||||
for i := 0; i < len(oids); i++ {
|
||||
coids[i] = *oids[i].toC()
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var oid C.git_oid
|
||||
ret := C.git_merge_base_many(&oid, r.ptr, C.size_t(len(oids)), &coids[0])
|
||||
runtime.KeepAlive(r)
|
||||
runtime.KeepAlive(coids)
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
return newOidFromC(&oid), nil
|
||||
}
|
||||
|
||||
// MergeBasesMany finds all merge bases given a list of commits.
|
||||
func (r *Repository) MergeBasesMany(oids []*Oid) ([]*Oid, error) {
|
||||
inCoids := make([]C.git_oid, len(oids))
|
||||
for i := 0; i < len(oids); i++ {
|
||||
inCoids[i] = *oids[i].toC()
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var outCoids C.git_oidarray
|
||||
ret := C.git_merge_bases_many(&outCoids, r.ptr, C.size_t(len(oids)), &inCoids[0])
|
||||
runtime.KeepAlive(r)
|
||||
runtime.KeepAlive(inCoids)
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
outOids := make([]*Oid, outCoids.count)
|
||||
hdr := reflect.SliceHeader{
|
||||
Data: uintptr(unsafe.Pointer(outCoids.ids)),
|
||||
Len: int(outCoids.count),
|
||||
Cap: int(outCoids.count),
|
||||
}
|
||||
goSlice := *(*[]C.git_oid)(unsafe.Pointer(&hdr))
|
||||
|
||||
for i, cid := range goSlice {
|
||||
outOids[i] = newOidFromC(&cid)
|
||||
}
|
||||
|
||||
return outOids, nil
|
||||
}
|
||||
|
||||
// MergeBaseOctopus finds a merge base in preparation for an octopus merge.
|
||||
func (r *Repository) MergeBaseOctopus(oids []*Oid) (*Oid, error) {
|
||||
coids := make([]C.git_oid, len(oids))
|
||||
for i := 0; i < len(oids); i++ {
|
||||
coids[i] = *oids[i].toC()
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var oid C.git_oid
|
||||
ret := C.git_merge_base_octopus(&oid, r.ptr, C.size_t(len(oids)), &coids[0])
|
||||
runtime.KeepAlive(r)
|
||||
runtime.KeepAlive(coids)
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
return newOidFromC(&oid), nil
|
||||
}
|
||||
|
||||
type MergeFileResult struct {
|
||||
doNotCompare
|
||||
ptr *C.git_merge_file_result
|
||||
Automergeable bool
|
||||
Path string
|
||||
Mode uint
|
||||
Contents []byte
|
||||
ptr *C.git_merge_file_result
|
||||
}
|
||||
|
||||
func newMergeFileResultFromC(c *C.git_merge_file_result) *MergeFileResult {
|
||||
|
@ -436,19 +515,28 @@ func mergeFileOptionsFromC(c C.git_merge_file_options) MergeFileOptions {
|
|||
}
|
||||
}
|
||||
|
||||
func populateCMergeFileOptions(c *C.git_merge_file_options, options MergeFileOptions) {
|
||||
c.ancestor_label = C.CString(options.AncestorLabel)
|
||||
c.our_label = C.CString(options.OurLabel)
|
||||
c.their_label = C.CString(options.TheirLabel)
|
||||
c.favor = C.git_merge_file_favor_t(options.Favor)
|
||||
c.flags = C.uint32_t(options.Flags)
|
||||
c.marker_size = C.ushort(options.MarkerSize)
|
||||
func populateMergeFileOptions(copts *C.git_merge_file_options, opts *MergeFileOptions) *C.git_merge_file_options {
|
||||
C.git_merge_file_options_init(copts, C.GIT_MERGE_FILE_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
copts.ancestor_label = C.CString(opts.AncestorLabel)
|
||||
copts.our_label = C.CString(opts.OurLabel)
|
||||
copts.their_label = C.CString(opts.TheirLabel)
|
||||
copts.favor = C.git_merge_file_favor_t(opts.Favor)
|
||||
copts.flags = C.uint32_t(opts.Flags)
|
||||
copts.marker_size = C.ushort(opts.MarkerSize)
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeCMergeFileOptions(c *C.git_merge_file_options) {
|
||||
C.free(unsafe.Pointer(c.ancestor_label))
|
||||
C.free(unsafe.Pointer(c.our_label))
|
||||
C.free(unsafe.Pointer(c.their_label))
|
||||
func freeMergeFileOptions(copts *C.git_merge_file_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
C.free(unsafe.Pointer(copts.ancestor_label))
|
||||
C.free(unsafe.Pointer(copts.our_label))
|
||||
C.free(unsafe.Pointer(copts.their_label))
|
||||
}
|
||||
|
||||
func MergeFile(ancestor MergeFileInput, ours MergeFileInput, theirs MergeFileInput, options *MergeFileOptions) (*MergeFileResult, error) {
|
||||
|
@ -481,8 +569,8 @@ func MergeFile(ancestor MergeFileInput, ours MergeFileInput, theirs MergeFileInp
|
|||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
populateCMergeFileOptions(copts, *options)
|
||||
defer freeCMergeFileOptions(copts)
|
||||
populateMergeFileOptions(copts, options)
|
||||
defer freeMergeFileOptions(copts)
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
|
|
|
@ -43,6 +43,14 @@ func TestMergeWithSelf(t *testing.T) {
|
|||
mergeHeads[0] = mergeHead
|
||||
err = repo.Merge(mergeHeads, nil, nil)
|
||||
checkFatal(t, err)
|
||||
|
||||
mergeMessage, err := repo.Message()
|
||||
checkFatal(t, err)
|
||||
|
||||
expectedMessage := "Merge branch 'master'\n"
|
||||
if mergeMessage != expectedMessage {
|
||||
t.Errorf("merge Message = %v, want %v", mergeMessage, expectedMessage)
|
||||
}
|
||||
}
|
||||
|
||||
func TestMergeAnalysisWithSelf(t *testing.T) {
|
||||
|
@ -163,6 +171,15 @@ func TestMergeBase(t *testing.T) {
|
|||
if mergeBase.Cmp(commitAId) != 0 {
|
||||
t.Fatalf("unexpected merge base")
|
||||
}
|
||||
}
|
||||
|
||||
func TestMergeBases(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
commitAId, _ := seedTestRepo(t, repo)
|
||||
commitBId, _ := appendCommit(t, repo)
|
||||
|
||||
mergeBases, err := repo.MergeBases(commitAId, commitBId)
|
||||
checkFatal(t, err)
|
||||
|
@ -176,6 +193,58 @@ func TestMergeBase(t *testing.T) {
|
|||
}
|
||||
}
|
||||
|
||||
func TestMergeBaseMany(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
commitAId, _ := seedTestRepo(t, repo)
|
||||
commitBId, _ := appendCommit(t, repo)
|
||||
|
||||
mergeBase, err := repo.MergeBaseMany([]*Oid{commitAId, commitBId})
|
||||
checkFatal(t, err)
|
||||
|
||||
if mergeBase.Cmp(commitAId) != 0 {
|
||||
t.Fatalf("unexpected merge base")
|
||||
}
|
||||
}
|
||||
|
||||
func TestMergeBasesMany(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
commitAId, _ := seedTestRepo(t, repo)
|
||||
commitBId, _ := appendCommit(t, repo)
|
||||
|
||||
mergeBases, err := repo.MergeBasesMany([]*Oid{commitAId, commitBId})
|
||||
checkFatal(t, err)
|
||||
|
||||
if len(mergeBases) != 1 {
|
||||
t.Fatalf("expected merge bases len to be 1, got %v", len(mergeBases))
|
||||
}
|
||||
|
||||
if mergeBases[0].Cmp(commitAId) != 0 {
|
||||
t.Fatalf("unexpected merge base")
|
||||
}
|
||||
}
|
||||
|
||||
func TestMergeBaseOctopus(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
commitAId, _ := seedTestRepo(t, repo)
|
||||
commitBId, _ := appendCommit(t, repo)
|
||||
|
||||
mergeBase, err := repo.MergeBaseOctopus([]*Oid{commitAId, commitBId})
|
||||
checkFatal(t, err)
|
||||
|
||||
if mergeBase.Cmp(commitAId) != 0 {
|
||||
t.Fatalf("unexpected merge base")
|
||||
}
|
||||
}
|
||||
|
||||
func compareBytes(t *testing.T, expected, actual []byte) {
|
||||
for i, v := range expected {
|
||||
if actual[i] != v {
|
||||
|
|
3
note.go
3
note.go
|
@ -13,6 +13,7 @@ import (
|
|||
// This object represents the possible operations which can be
|
||||
// performed on the collection of notes for a repository.
|
||||
type NoteCollection struct {
|
||||
doNotCompare
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
|
@ -139,6 +140,7 @@ func (c *NoteCollection) DefaultRef() (string, error) {
|
|||
|
||||
// Note
|
||||
type Note struct {
|
||||
doNotCompare
|
||||
ptr *C.git_note
|
||||
r *Repository
|
||||
}
|
||||
|
@ -189,6 +191,7 @@ func (n *Note) Message() string {
|
|||
|
||||
// NoteIterator
|
||||
type NoteIterator struct {
|
||||
doNotCompare
|
||||
ptr *C.git_note_iterator
|
||||
r *Repository
|
||||
}
|
||||
|
|
|
@ -49,7 +49,7 @@ func TestNoteIterator(t *testing.T) {
|
|||
for {
|
||||
noteId, commitId, err := iter.Next()
|
||||
if err != nil {
|
||||
if !IsErrorCode(err, ErrIterOver) {
|
||||
if !IsErrorCode(err, ErrorCodeIterOver) {
|
||||
checkFatal(t, err)
|
||||
}
|
||||
break
|
||||
|
|
17
object.go
17
object.go
|
@ -22,6 +22,7 @@ const (
|
|||
)
|
||||
|
||||
type Object struct {
|
||||
doNotCompare
|
||||
ptr *C.git_object
|
||||
repo *Repository
|
||||
}
|
||||
|
@ -77,14 +78,14 @@ func (o *Object) Type() ObjectType {
|
|||
return ret
|
||||
}
|
||||
|
||||
// Owner returns a weak reference to the repository which owns this
|
||||
// object. This won't keep the underlying repository alive.
|
||||
// Owner returns a weak reference to the repository which owns this object.
|
||||
// This won't keep the underlying repository alive, but it should still be
|
||||
// Freed.
|
||||
func (o *Object) Owner() *Repository {
|
||||
ret := &Repository{
|
||||
ptr: C.git_object_owner(o.ptr),
|
||||
}
|
||||
repo := newRepositoryFromC(C.git_object_owner(o.ptr))
|
||||
runtime.KeepAlive(o)
|
||||
return ret
|
||||
repo.weak = true
|
||||
return repo
|
||||
}
|
||||
|
||||
func dupObject(obj *Object, kind ObjectType) (*C.git_object, error) {
|
||||
|
@ -201,13 +202,13 @@ func (o *Object) Free() {
|
|||
|
||||
// Peel recursively peels an object until an object of the specified type is met.
|
||||
//
|
||||
// If the query cannot be satisfied due to the object model, ErrInvalidSpec
|
||||
// If the query cannot be satisfied due to the object model, ErrorCodeInvalidSpec
|
||||
// will be returned (e.g. trying to peel a blob to a tree).
|
||||
//
|
||||
// If you pass ObjectAny as the target type, then the object will be peeled
|
||||
// until the type changes. A tag will be peeled until the referenced object
|
||||
// is no longer a tag, and a commit will be peeled to a tree. Any other object
|
||||
// type will return ErrInvalidSpec.
|
||||
// type will return ErrorCodeInvalidSpec.
|
||||
//
|
||||
// If peeling a tag we discover an object which cannot be peeled to the target
|
||||
// type due to the object model, an error will be returned.
|
||||
|
|
|
@ -153,8 +153,8 @@ func TestObjectPeel(t *testing.T) {
|
|||
|
||||
obj, err = commit.Peel(ObjectTag)
|
||||
|
||||
if !IsErrorCode(err, ErrInvalidSpec) {
|
||||
t.Fatalf("Wrong error when peeling a commit to a tag, expected ErrInvalidSpec, have %v", err)
|
||||
if !IsErrorCode(err, ErrorCodeInvalidSpec) {
|
||||
t.Fatalf("Wrong error when peeling a commit to a tag, expected ErrorCodeInvalidSpec, have %v", err)
|
||||
}
|
||||
|
||||
tree, err := repo.LookupTree(treeID)
|
||||
|
@ -162,8 +162,8 @@ func TestObjectPeel(t *testing.T) {
|
|||
|
||||
obj, err = tree.Peel(ObjectAny)
|
||||
|
||||
if !IsErrorCode(err, ErrInvalidSpec) {
|
||||
t.Fatalf("Wrong error when peeling a tree, expected ErrInvalidSpec, have %v", err)
|
||||
if !IsErrorCode(err, ErrorCodeInvalidSpec) {
|
||||
t.Fatalf("Wrong error when peeling a tree, expected ErrorCodeInvalidSpec, have %v", err)
|
||||
}
|
||||
|
||||
entry := tree.EntryByName("README")
|
||||
|
@ -173,8 +173,8 @@ func TestObjectPeel(t *testing.T) {
|
|||
|
||||
obj, err = blob.Peel(ObjectAny)
|
||||
|
||||
if !IsErrorCode(err, ErrInvalidSpec) {
|
||||
t.Fatalf("Wrong error when peeling a blob, expected ErrInvalidSpec, have %v", err)
|
||||
if !IsErrorCode(err, ErrorCodeInvalidSpec) {
|
||||
t.Fatalf("Wrong error when peeling a blob, expected ErrorCodeInvalidSpec, have %v", err)
|
||||
}
|
||||
|
||||
tagID := createTestTag(t, repo, commit)
|
||||
|
|
86
odb.go
86
odb.go
|
@ -22,10 +22,12 @@ import (
|
|||
)
|
||||
|
||||
type Odb struct {
|
||||
doNotCompare
|
||||
ptr *C.git_odb
|
||||
}
|
||||
|
||||
type OdbBackend struct {
|
||||
doNotCompare
|
||||
ptr *C.git_odb_backend
|
||||
}
|
||||
|
||||
|
@ -45,7 +47,7 @@ func NewOdb() (odb *Odb, err error) {
|
|||
}
|
||||
|
||||
func NewOdbBackendFromC(ptr unsafe.Pointer) (backend *OdbBackend) {
|
||||
backend = &OdbBackend{(*C.git_odb_backend)(ptr)}
|
||||
backend = &OdbBackend{ptr: (*C.git_odb_backend)(ptr)}
|
||||
return backend
|
||||
}
|
||||
|
||||
|
@ -174,36 +176,60 @@ func (v *Odb) Read(oid *Oid) (obj *OdbObject, err error) {
|
|||
return obj, nil
|
||||
}
|
||||
|
||||
type OdbForEachCallback func(id *Oid) error
|
||||
func (odb *Odb) Refresh() error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
type foreachData struct {
|
||||
callback OdbForEachCallback
|
||||
err error
|
||||
ret := C.git_odb_refresh(odb.ptr)
|
||||
runtime.KeepAlive(odb)
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
//export odbForEachCb
|
||||
func odbForEachCb(id *C.git_oid, handle unsafe.Pointer) int {
|
||||
data, ok := pointerHandles.Get(handle).(*foreachData)
|
||||
func (odb *Odb) WriteMultiPackIndex() error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_odb_write_multi_pack_index(odb.ptr)
|
||||
runtime.KeepAlive(odb)
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
type OdbForEachCallback func(id *Oid) error
|
||||
type odbForEachCallbackData struct {
|
||||
callback OdbForEachCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export odbForEachCallback
|
||||
func odbForEachCallback(id *C.git_oid, handle unsafe.Pointer) C.int {
|
||||
data, ok := pointerHandles.Get(handle).(*odbForEachCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve handle")
|
||||
}
|
||||
|
||||
err := data.callback(newOidFromC(id))
|
||||
if err != nil {
|
||||
data.err = err
|
||||
return C.GIT_EUSER
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func (v *Odb) ForEach(callback OdbForEachCallback) error {
|
||||
data := foreachData{
|
||||
callback: callback,
|
||||
err: nil,
|
||||
var err error
|
||||
data := odbForEachCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
|
@ -212,9 +238,10 @@ func (v *Odb) ForEach(callback OdbForEachCallback) error {
|
|||
|
||||
ret := C._go_git_odb_foreach(v.ptr, handle)
|
||||
runtime.KeepAlive(v)
|
||||
if ret == C.GIT_EUSER {
|
||||
return data.err
|
||||
} else if ret < 0 {
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
|
@ -292,18 +319,16 @@ func (v *Odb) NewWriteStream(size int64, otype ObjectType) (*OdbWriteStream, err
|
|||
// layer does not understand pack files, the objects will be stored in whatever
|
||||
// format the ODB layer uses.
|
||||
func (v *Odb) NewWritePack(callback TransferProgressCallback) (*OdbWritepack, error) {
|
||||
writepack := new(OdbWritepack)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
writepack.callbacks.TransferProgressCallback = callback
|
||||
writepack.callbacksHandle = pointerHandles.Track(&writepack.callbacks)
|
||||
writepack := new(OdbWritepack)
|
||||
populateRemoteCallbacks(&writepack.ccallbacks, &RemoteCallbacks{TransferProgressCallback: callback}, nil)
|
||||
|
||||
ret := C._go_git_odb_write_pack(&writepack.ptr, v.ptr, writepack.callbacksHandle)
|
||||
ret := C._go_git_odb_write_pack(&writepack.ptr, v.ptr, writepack.ccallbacks.payload)
|
||||
runtime.KeepAlive(v)
|
||||
if ret < 0 {
|
||||
pointerHandles.Untrack(writepack.callbacksHandle)
|
||||
untrackCallbacksPayload(&writepack.ccallbacks)
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
|
@ -316,6 +341,7 @@ func (v *OdbBackend) Free() {
|
|||
}
|
||||
|
||||
type OdbObject struct {
|
||||
doNotCompare
|
||||
ptr *C.git_odb_object
|
||||
}
|
||||
|
||||
|
@ -359,6 +385,7 @@ func (object *OdbObject) Data() (data []byte) {
|
|||
}
|
||||
|
||||
type OdbReadStream struct {
|
||||
doNotCompare
|
||||
ptr *C.git_odb_stream
|
||||
Size uint64
|
||||
Type ObjectType
|
||||
|
@ -399,6 +426,7 @@ func (stream *OdbReadStream) Free() {
|
|||
}
|
||||
|
||||
type OdbWriteStream struct {
|
||||
doNotCompare
|
||||
ptr *C.git_odb_stream
|
||||
Id Oid
|
||||
}
|
||||
|
@ -443,10 +471,10 @@ func (stream *OdbWriteStream) Free() {
|
|||
|
||||
// OdbWritepack is a stream to write a packfile to the ODB.
|
||||
type OdbWritepack struct {
|
||||
ptr *C.git_odb_writepack
|
||||
stats C.git_transfer_progress
|
||||
callbacks RemoteCallbacks
|
||||
callbacksHandle unsafe.Pointer
|
||||
doNotCompare
|
||||
ptr *C.git_odb_writepack
|
||||
stats C.git_transfer_progress
|
||||
ccallbacks C.git_remote_callbacks
|
||||
}
|
||||
|
||||
func (writepack *OdbWritepack) Write(data []byte) (int, error) {
|
||||
|
@ -480,7 +508,7 @@ func (writepack *OdbWritepack) Commit() error {
|
|||
}
|
||||
|
||||
func (writepack *OdbWritepack) Free() {
|
||||
pointerHandles.Untrack(writepack.callbacksHandle)
|
||||
untrackCallbacksPayload(&writepack.ccallbacks)
|
||||
runtime.SetFinalizer(writepack, nil)
|
||||
C._go_git_odb_writepack_free(writepack.ptr)
|
||||
}
|
||||
|
|
42
odb_test.go
42
odb_test.go
|
@ -61,31 +61,31 @@ func TestOdbStream(t *testing.T) {
|
|||
|
||||
_, _ = seedTestRepo(t, repo)
|
||||
|
||||
odb, error := repo.Odb()
|
||||
checkFatal(t, error)
|
||||
odb, err := repo.Odb()
|
||||
checkFatal(t, err)
|
||||
|
||||
str := "hello, world!"
|
||||
|
||||
writeStream, error := odb.NewWriteStream(int64(len(str)), ObjectBlob)
|
||||
checkFatal(t, error)
|
||||
n, error := io.WriteString(writeStream, str)
|
||||
checkFatal(t, error)
|
||||
writeStream, err := odb.NewWriteStream(int64(len(str)), ObjectBlob)
|
||||
checkFatal(t, err)
|
||||
n, err := io.WriteString(writeStream, str)
|
||||
checkFatal(t, err)
|
||||
if n != len(str) {
|
||||
t.Fatalf("Bad write length %v != %v", n, len(str))
|
||||
}
|
||||
|
||||
error = writeStream.Close()
|
||||
checkFatal(t, error)
|
||||
err = writeStream.Close()
|
||||
checkFatal(t, err)
|
||||
|
||||
expectedId, error := NewOid("30f51a3fba5274d53522d0f19748456974647b4f")
|
||||
checkFatal(t, error)
|
||||
expectedId, err := NewOid("30f51a3fba5274d53522d0f19748456974647b4f")
|
||||
checkFatal(t, err)
|
||||
if writeStream.Id.Cmp(expectedId) != 0 {
|
||||
t.Fatal("Wrong data written")
|
||||
}
|
||||
|
||||
readStream, error := odb.NewReadStream(&writeStream.Id)
|
||||
checkFatal(t, error)
|
||||
data, error := ioutil.ReadAll(readStream)
|
||||
readStream, err := odb.NewReadStream(&writeStream.Id)
|
||||
checkFatal(t, err)
|
||||
data, err := ioutil.ReadAll(readStream)
|
||||
if str != string(data) {
|
||||
t.Fatalf("Wrong data read %v != %v", str, string(data))
|
||||
}
|
||||
|
@ -98,8 +98,8 @@ func TestOdbHash(t *testing.T) {
|
|||
|
||||
_, _ = seedTestRepo(t, repo)
|
||||
|
||||
odb, error := repo.Odb()
|
||||
checkFatal(t, error)
|
||||
odb, err := repo.Odb()
|
||||
checkFatal(t, err)
|
||||
|
||||
str := `tree 115fcae49287c82eb55bb275cbbd4556fbed72b7
|
||||
parent 66e1c476199ebcd3e304659992233132c5a52c6c
|
||||
|
@ -109,11 +109,11 @@ committer John Doe <john@doe.com> 1390682018 +0000
|
|||
Initial commit.`
|
||||
|
||||
for _, data := range [][]byte{[]byte(str), doublePointerBytes()} {
|
||||
oid, error := odb.Hash(data, ObjectCommit)
|
||||
checkFatal(t, error)
|
||||
oid, err := odb.Hash(data, ObjectCommit)
|
||||
checkFatal(t, err)
|
||||
|
||||
coid, error := odb.Write(data, ObjectCommit)
|
||||
checkFatal(t, error)
|
||||
coid, err := odb.Write(data, ObjectCommit)
|
||||
checkFatal(t, err)
|
||||
|
||||
if oid.Cmp(coid) != 0 {
|
||||
t.Fatal("Hash and write Oids are different")
|
||||
|
@ -167,9 +167,9 @@ func TestOdbWritepack(t *testing.T) {
|
|||
checkFatal(t, err)
|
||||
|
||||
var finalStats TransferProgress
|
||||
writepack, err := odb.NewWritePack(func(stats TransferProgress) ErrorCode {
|
||||
writepack, err := odb.NewWritePack(func(stats TransferProgress) error {
|
||||
finalStats = stats
|
||||
return ErrOk
|
||||
return nil
|
||||
})
|
||||
checkFatal(t, err)
|
||||
defer writepack.Free()
|
||||
|
|
|
@ -16,6 +16,7 @@ import (
|
|||
)
|
||||
|
||||
type Packbuilder struct {
|
||||
doNotCompare
|
||||
ptr *C.git_packbuilder
|
||||
r *Repository
|
||||
}
|
||||
|
@ -133,15 +134,15 @@ func (pb *Packbuilder) Written() uint32 {
|
|||
}
|
||||
|
||||
type PackbuilderForeachCallback func([]byte) error
|
||||
type packbuilderCbData struct {
|
||||
callback PackbuilderForeachCallback
|
||||
err error
|
||||
type packbuilderCallbackData struct {
|
||||
callback PackbuilderForeachCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export packbuilderForEachCb
|
||||
func packbuilderForEachCb(buf unsafe.Pointer, size C.size_t, handle unsafe.Pointer) int {
|
||||
//export packbuilderForEachCallback
|
||||
func packbuilderForEachCallback(buf unsafe.Pointer, size C.size_t, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*packbuilderCbData)
|
||||
data, ok := payload.(*packbuilderCallbackData)
|
||||
if !ok {
|
||||
panic("could not get packbuilder CB data")
|
||||
}
|
||||
|
@ -150,19 +151,20 @@ func packbuilderForEachCb(buf unsafe.Pointer, size C.size_t, handle unsafe.Point
|
|||
|
||||
err := data.callback(slice)
|
||||
if err != nil {
|
||||
data.err = err
|
||||
return C.GIT_EUSER
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// ForEach repeatedly calls the callback with new packfile data until
|
||||
// there is no more data or the callback returns an error
|
||||
func (pb *Packbuilder) ForEach(callback PackbuilderForeachCallback) error {
|
||||
data := packbuilderCbData{
|
||||
callback: callback,
|
||||
err: nil,
|
||||
var err error
|
||||
data := packbuilderCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
handle := pointerHandles.Track(&data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
|
@ -170,13 +172,13 @@ func (pb *Packbuilder) ForEach(callback PackbuilderForeachCallback) error {
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
err := C._go_git_packbuilder_foreach(pb.ptr, handle)
|
||||
ret := C._go_git_packbuilder_foreach(pb.ptr, handle)
|
||||
runtime.KeepAlive(pb)
|
||||
if err == C.GIT_EUSER {
|
||||
return data.err
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if err < 0 {
|
||||
return MakeGitError(err)
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
|
|
14
patch.go
14
patch.go
|
@ -10,6 +10,7 @@ import (
|
|||
)
|
||||
|
||||
type Patch struct {
|
||||
doNotCompare
|
||||
ptr *C.git_patch
|
||||
}
|
||||
|
||||
|
@ -77,17 +78,22 @@ func (v *Repository) PatchFromBuffers(oldPath, newPath string, oldBuf, newBuf []
|
|||
cNewPath := C.CString(newPath)
|
||||
defer C.free(unsafe.Pointer(cNewPath))
|
||||
|
||||
copts := diffOptionsToC(opts, v)
|
||||
var err error
|
||||
copts := populateDiffOptions(&C.git_diff_options{}, opts, v, &err)
|
||||
defer freeDiffOptions(copts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_patch_from_buffers(&patchPtr, oldPtr, C.size_t(len(oldBuf)), cOldPath, newPtr, C.size_t(len(newBuf)), cNewPath, copts)
|
||||
ret := C.git_patch_from_buffers(&patchPtr, oldPtr, C.size_t(len(oldBuf)), cOldPath, newPtr, C.size_t(len(newBuf)), cNewPath, copts)
|
||||
runtime.KeepAlive(oldBuf)
|
||||
runtime.KeepAlive(newBuf)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newPatchFromC(patchPtr), nil
|
||||
}
|
||||
|
|
|
@ -14,15 +14,18 @@ func TestRemotePush(t *testing.T) {
|
|||
|
||||
remote, err := localRepo.Remotes.Create("test_push", repo.Path())
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
seedTestRepo(t, localRepo)
|
||||
|
||||
err = remote.Push([]string{"refs/heads/master"}, nil)
|
||||
checkFatal(t, err)
|
||||
|
||||
_, err = localRepo.References.Lookup("refs/remotes/test_push/master")
|
||||
ref, err := localRepo.References.Lookup("refs/remotes/test_push/master")
|
||||
checkFatal(t, err)
|
||||
defer ref.Free()
|
||||
|
||||
_, err = repo.References.Lookup("refs/heads/master")
|
||||
ref, err = repo.References.Lookup("refs/heads/master")
|
||||
checkFatal(t, err)
|
||||
defer ref.Free()
|
||||
}
|
||||
|
|
269
rebase.go
269
rebase.go
|
@ -3,12 +3,13 @@ package git
|
|||
/*
|
||||
#include <git2.h>
|
||||
|
||||
extern void _go_git_populate_commit_sign_cb(git_rebase_options *opts);
|
||||
extern void _go_git_populate_rebase_callbacks(git_rebase_options *opts);
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"reflect"
|
||||
"runtime"
|
||||
"unsafe"
|
||||
)
|
||||
|
@ -71,68 +72,143 @@ func newRebaseOperationFromC(c *C.git_rebase_operation) *RebaseOperation {
|
|||
return operation
|
||||
}
|
||||
|
||||
//export commitSignCallback
|
||||
func commitSignCallback(_signature *C.git_buf, _signature_field *C.git_buf, _commit_content *C.char, _payload unsafe.Pointer) C.int {
|
||||
opts, ok := pointerHandles.Get(_payload).(*RebaseOptions)
|
||||
//export commitCreateCallback
|
||||
func commitCreateCallback(
|
||||
errorMessage **C.char,
|
||||
_out *C.git_oid,
|
||||
_author, _committer *C.git_signature,
|
||||
_message_encoding, _message *C.char,
|
||||
_tree *C.git_tree,
|
||||
_parent_count C.size_t,
|
||||
_parents **C.git_commit,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
data, ok := pointerHandles.Get(handle).(*rebaseOptionsData)
|
||||
if !ok {
|
||||
panic("invalid sign payload")
|
||||
}
|
||||
|
||||
if opts.CommitSigningCallback == nil {
|
||||
return C.GIT_PASSTHROUGH
|
||||
if data.options.CommitCreateCallback == nil && data.options.CommitSigningCallback == nil {
|
||||
return C.int(ErrorCodePassthrough)
|
||||
}
|
||||
|
||||
commitContent := C.GoString(_commit_content)
|
||||
messageEncoding := MessageEncodingUTF8
|
||||
if _message_encoding != nil {
|
||||
messageEncoding = MessageEncoding(C.GoString(_message_encoding))
|
||||
}
|
||||
tree := &Tree{
|
||||
Object: Object{
|
||||
ptr: (*C.git_object)(_tree),
|
||||
repo: data.repo,
|
||||
},
|
||||
cast_ptr: _tree,
|
||||
}
|
||||
|
||||
signature, signatureField, err := opts.CommitSigningCallback(commitContent)
|
||||
if err != nil {
|
||||
if gitError, ok := err.(*GitError); ok {
|
||||
return C.int(gitError.Code)
|
||||
var goParents []*C.git_commit
|
||||
if _parent_count > 0 {
|
||||
hdr := reflect.SliceHeader{
|
||||
Data: uintptr(unsafe.Pointer(_parents)),
|
||||
Len: int(_parent_count),
|
||||
Cap: int(_parent_count),
|
||||
}
|
||||
return C.int(-1)
|
||||
goParents = *(*[]*C.git_commit)(unsafe.Pointer(&hdr))
|
||||
}
|
||||
|
||||
fillBuf := func(bufData string, buf *C.git_buf) error {
|
||||
clen := C.size_t(len(bufData))
|
||||
cstr := unsafe.Pointer(C.CString(bufData))
|
||||
defer C.free(cstr)
|
||||
|
||||
// libgit2 requires the contents of the buffer to be NULL-terminated.
|
||||
// C.CString() guarantees that the returned buffer will be
|
||||
// NULL-terminated, so we can safely copy the terminator.
|
||||
if int(C.git_buf_set(buf, cstr, clen+1)) != 0 {
|
||||
return errors.New("could not set buffer")
|
||||
parents := make([]*Commit, int(_parent_count))
|
||||
for i, p := range goParents {
|
||||
parents[i] = &Commit{
|
||||
Object: Object{
|
||||
ptr: (*C.git_object)(p),
|
||||
repo: data.repo,
|
||||
},
|
||||
cast_ptr: p,
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
if signatureField != "" {
|
||||
err := fillBuf(signatureField, _signature_field)
|
||||
if data.options.CommitCreateCallback != nil {
|
||||
oid, err := data.options.CommitCreateCallback(
|
||||
newSignatureFromC(_author),
|
||||
newSignatureFromC(_committer),
|
||||
messageEncoding,
|
||||
C.GoString(_message),
|
||||
tree,
|
||||
parents...,
|
||||
)
|
||||
if err != nil {
|
||||
return C.int(-1)
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
if oid == nil {
|
||||
return C.int(ErrorCodePassthrough)
|
||||
}
|
||||
*_out = *oid.toC()
|
||||
} else if data.options.CommitSigningCallback != nil {
|
||||
commitContent, err := data.repo.CreateCommitBuffer(
|
||||
newSignatureFromC(_author),
|
||||
newSignatureFromC(_committer),
|
||||
messageEncoding,
|
||||
C.GoString(_message),
|
||||
tree,
|
||||
parents...,
|
||||
)
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
signature, signatureField, err := data.options.CommitSigningCallback(string(commitContent))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
oid, err := data.repo.CreateCommitWithSignature(string(commitContent), signature, signatureField)
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
*_out = *oid.toC()
|
||||
}
|
||||
|
||||
err = fillBuf(signature, _signature)
|
||||
if err != nil {
|
||||
return C.int(-1)
|
||||
}
|
||||
|
||||
return C.GIT_OK
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// RebaseOptions are used to tell the rebase machinery how to operate
|
||||
// RebaseOptions are used to tell the rebase machinery how to operate.
|
||||
type RebaseOptions struct {
|
||||
Version uint
|
||||
Quiet int
|
||||
InMemory int
|
||||
RewriteNotesRef string
|
||||
MergeOptions MergeOptions
|
||||
CheckoutOptions CheckoutOpts
|
||||
Quiet int
|
||||
InMemory int
|
||||
RewriteNotesRef string
|
||||
MergeOptions MergeOptions
|
||||
CheckoutOptions CheckoutOptions
|
||||
// CommitCreateCallback is an optional callback that allows users to override
|
||||
// commit creation when rebasing. If specified, users can create
|
||||
// their own commit and provide the commit ID, which may be useful for
|
||||
// signing commits or otherwise customizing the commit creation. If this
|
||||
// callback returns a nil Oid, then the rebase will continue to create the
|
||||
// commit.
|
||||
CommitCreateCallback CommitCreateCallback
|
||||
// Deprecated: CommitSigningCallback is an optional callback that will be
|
||||
// called with the commit content, allowing a signature to be added to the
|
||||
// rebase commit. This field is only used when rebasing. This callback is
|
||||
// not invoked if a CommitCreateCallback is specified. CommitCreateCallback
|
||||
// should be used instead of this.
|
||||
CommitSigningCallback CommitSigningCallback
|
||||
}
|
||||
|
||||
type rebaseOptionsData struct {
|
||||
options *RebaseOptions
|
||||
repo *Repository
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
// DefaultRebaseOptions returns a RebaseOptions with default values.
|
||||
func DefaultRebaseOptions() (RebaseOptions, error) {
|
||||
opts := C.git_rebase_options{}
|
||||
|
@ -149,7 +225,6 @@ func DefaultRebaseOptions() (RebaseOptions, error) {
|
|||
|
||||
func rebaseOptionsFromC(opts *C.git_rebase_options) RebaseOptions {
|
||||
return RebaseOptions{
|
||||
Version: uint(opts.version),
|
||||
Quiet: int(opts.quiet),
|
||||
InMemory: int(opts.inmemory),
|
||||
RewriteNotesRef: C.GoString(opts.rewrite_notes_ref),
|
||||
|
@ -158,25 +233,41 @@ func rebaseOptionsFromC(opts *C.git_rebase_options) RebaseOptions {
|
|||
}
|
||||
}
|
||||
|
||||
func (ro *RebaseOptions) toC() *C.git_rebase_options {
|
||||
if ro == nil {
|
||||
func populateRebaseOptions(copts *C.git_rebase_options, opts *RebaseOptions, repo *Repository, errorTarget *error) *C.git_rebase_options {
|
||||
C.git_rebase_options_init(copts, C.GIT_REBASE_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
opts := &C.git_rebase_options{
|
||||
version: C.uint(ro.Version),
|
||||
quiet: C.int(ro.Quiet),
|
||||
inmemory: C.int(ro.InMemory),
|
||||
rewrite_notes_ref: mapEmptyStringToNull(ro.RewriteNotesRef),
|
||||
merge_options: *ro.MergeOptions.toC(),
|
||||
checkout_options: *ro.CheckoutOptions.toC(),
|
||||
|
||||
copts.quiet = C.int(opts.Quiet)
|
||||
copts.inmemory = C.int(opts.InMemory)
|
||||
copts.rewrite_notes_ref = mapEmptyStringToNull(opts.RewriteNotesRef)
|
||||
populateMergeOptions(&copts.merge_options, &opts.MergeOptions)
|
||||
populateCheckoutOptions(&copts.checkout_options, &opts.CheckoutOptions, errorTarget)
|
||||
|
||||
if opts.CommitCreateCallback != nil || opts.CommitSigningCallback != nil {
|
||||
data := &rebaseOptionsData{
|
||||
options: opts,
|
||||
repo: repo,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
C._go_git_populate_rebase_callbacks(copts)
|
||||
copts.payload = pointerHandles.Track(data)
|
||||
}
|
||||
|
||||
if ro.CommitSigningCallback != nil {
|
||||
C._go_git_populate_commit_sign_cb(opts)
|
||||
opts.payload = pointerHandles.Track(ro)
|
||||
}
|
||||
return copts
|
||||
}
|
||||
|
||||
return opts
|
||||
func freeRebaseOptions(copts *C.git_rebase_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
C.free(unsafe.Pointer(copts.rewrite_notes_ref))
|
||||
freeMergeOptions(&copts.merge_options)
|
||||
freeCheckoutOptions(&copts.checkout_options)
|
||||
if copts.payload != nil {
|
||||
pointerHandles.Untrack(copts.payload)
|
||||
}
|
||||
}
|
||||
|
||||
func mapEmptyStringToNull(ref string) *C.char {
|
||||
|
@ -188,8 +279,10 @@ func mapEmptyStringToNull(ref string) *C.char {
|
|||
|
||||
// Rebase is the struct representing a Rebase object.
|
||||
type Rebase struct {
|
||||
ptr *C.git_rebase
|
||||
r *Repository
|
||||
doNotCompare
|
||||
ptr *C.git_rebase
|
||||
r *Repository
|
||||
options *C.git_rebase_options
|
||||
}
|
||||
|
||||
// InitRebase initializes a rebase operation to rebase the changes in branch relative to upstream onto another branch.
|
||||
|
@ -210,15 +303,22 @@ func (r *Repository) InitRebase(branch *AnnotatedCommit, upstream *AnnotatedComm
|
|||
}
|
||||
|
||||
var ptr *C.git_rebase
|
||||
err := C.git_rebase_init(&ptr, r.ptr, branch.ptr, upstream.ptr, onto.ptr, opts.toC())
|
||||
var err error
|
||||
cOpts := populateRebaseOptions(&C.git_rebase_options{}, opts, r, &err)
|
||||
ret := C.git_rebase_init(&ptr, r.ptr, branch.ptr, upstream.ptr, onto.ptr, cOpts)
|
||||
runtime.KeepAlive(branch)
|
||||
runtime.KeepAlive(upstream)
|
||||
runtime.KeepAlive(onto)
|
||||
if err < 0 {
|
||||
return nil, MakeGitError(err)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
freeRebaseOptions(cOpts)
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
freeRebaseOptions(cOpts)
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newRebaseFromC(ptr), nil
|
||||
return newRebaseFromC(ptr, r, cOpts), nil
|
||||
}
|
||||
|
||||
// OpenRebase opens an existing rebase that was previously started by either an invocation of InitRebase or by another client.
|
||||
|
@ -227,13 +327,20 @@ func (r *Repository) OpenRebase(opts *RebaseOptions) (*Rebase, error) {
|
|||
defer runtime.UnlockOSThread()
|
||||
|
||||
var ptr *C.git_rebase
|
||||
err := C.git_rebase_open(&ptr, r.ptr, opts.toC())
|
||||
var err error
|
||||
cOpts := populateRebaseOptions(&C.git_rebase_options{}, opts, r, &err)
|
||||
ret := C.git_rebase_open(&ptr, r.ptr, cOpts)
|
||||
runtime.KeepAlive(r)
|
||||
if err < 0 {
|
||||
return nil, MakeGitError(err)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
freeRebaseOptions(cOpts)
|
||||
return nil, err
|
||||
}
|
||||
if ret < 0 {
|
||||
freeRebaseOptions(cOpts)
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return newRebaseFromC(ptr), nil
|
||||
return newRebaseFromC(ptr, r, cOpts), nil
|
||||
}
|
||||
|
||||
// OperationAt gets the rebase operation specified by the given index.
|
||||
|
@ -285,6 +392,27 @@ func (rebase *Rebase) Next() (*RebaseOperation, error) {
|
|||
return newRebaseOperationFromC(ptr), nil
|
||||
}
|
||||
|
||||
// InmemoryIndex gets the index produced by the last operation, which is the
|
||||
// result of `Next()` and which will be committed by the next invocation of
|
||||
// `Commit()`. This is useful for resolving conflicts in an in-memory rebase
|
||||
// before committing them.
|
||||
//
|
||||
// This is only applicable for in-memory rebases; for rebases within a working
|
||||
// directory, the changes were applied to the repository's index.
|
||||
func (rebase *Rebase) InmemoryIndex() (*Index, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var ptr *C.git_index
|
||||
err := C.git_rebase_inmemory_index(&ptr, rebase.ptr)
|
||||
runtime.KeepAlive(rebase)
|
||||
if err < 0 {
|
||||
return nil, MakeGitError(err)
|
||||
}
|
||||
|
||||
return newIndexFromC(ptr, rebase.r), nil
|
||||
}
|
||||
|
||||
// Commit commits the current patch.
|
||||
// You must have resolved any conflicts that were introduced during the patch application from the Next() invocation.
|
||||
func (rebase *Rebase) Commit(ID *Oid, author, committer *Signature, message string) error {
|
||||
|
@ -344,13 +472,14 @@ func (rebase *Rebase) Abort() error {
|
|||
}
|
||||
|
||||
// Free frees the Rebase object.
|
||||
func (rebase *Rebase) Free() {
|
||||
runtime.SetFinalizer(rebase, nil)
|
||||
C.git_rebase_free(rebase.ptr)
|
||||
func (r *Rebase) Free() {
|
||||
runtime.SetFinalizer(r, nil)
|
||||
C.git_rebase_free(r.ptr)
|
||||
freeRebaseOptions(r.options)
|
||||
}
|
||||
|
||||
func newRebaseFromC(ptr *C.git_rebase) *Rebase {
|
||||
rebase := &Rebase{ptr: ptr}
|
||||
func newRebaseFromC(ptr *C.git_rebase, repo *Repository, opts *C.git_rebase_options) *Rebase {
|
||||
rebase := &Rebase{ptr: ptr, r: repo, options: opts}
|
||||
runtime.SetFinalizer(rebase, (*Rebase).Free)
|
||||
return rebase
|
||||
}
|
||||
|
|
110
rebase_test.go
110
rebase_test.go
|
@ -8,12 +8,86 @@ import (
|
|||
"testing"
|
||||
"time"
|
||||
|
||||
"golang.org/x/crypto/openpgp"
|
||||
"golang.org/x/crypto/openpgp/packet"
|
||||
"github.com/ProtonMail/go-crypto/openpgp"
|
||||
"github.com/ProtonMail/go-crypto/openpgp/packet"
|
||||
)
|
||||
|
||||
// Tests
|
||||
|
||||
func TestRebaseInMemoryWithConflict(t *testing.T) {
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
seedTestRepo(t, repo)
|
||||
|
||||
// Create two branches with common history, where both modify "common-file"
|
||||
// in a conflicting way.
|
||||
_, err := commitSomething(repo, "common-file", "a\nb\nc\n", commitOptions{})
|
||||
checkFatal(t, err)
|
||||
checkFatal(t, createBranch(repo, "branch-a"))
|
||||
checkFatal(t, createBranch(repo, "branch-b"))
|
||||
|
||||
checkFatal(t, repo.SetHead("refs/heads/branch-a"))
|
||||
_, err = commitSomething(repo, "common-file", "1\nb\nc\n", commitOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
checkFatal(t, repo.SetHead("refs/heads/branch-b"))
|
||||
_, err = commitSomething(repo, "common-file", "x\nb\nc\n", commitOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
branchA, err := repo.LookupBranch("branch-a", BranchLocal)
|
||||
checkFatal(t, err)
|
||||
onto, err := repo.AnnotatedCommitFromRef(branchA.Reference)
|
||||
checkFatal(t, err)
|
||||
|
||||
// We then rebase "branch-b" onto "branch-a" in-memory, which should result
|
||||
// in a conflict.
|
||||
rebase, err := repo.InitRebase(nil, nil, onto, &RebaseOptions{InMemory: 1})
|
||||
checkFatal(t, err)
|
||||
|
||||
_, err = rebase.Next()
|
||||
checkFatal(t, err)
|
||||
|
||||
index, err := rebase.InmemoryIndex()
|
||||
checkFatal(t, err)
|
||||
|
||||
// We simply resolve the conflict and commit the rebase.
|
||||
if !index.HasConflicts() {
|
||||
t.Fatal("expected index to have conflicts")
|
||||
}
|
||||
|
||||
conflict, err := index.Conflict("common-file")
|
||||
checkFatal(t, err)
|
||||
|
||||
resolvedBlobID, err := repo.CreateBlobFromBuffer([]byte("resolved contents"))
|
||||
checkFatal(t, err)
|
||||
|
||||
resolvedEntry := *conflict.Our
|
||||
resolvedEntry.Id = resolvedBlobID
|
||||
checkFatal(t, index.Add(&resolvedEntry))
|
||||
checkFatal(t, index.RemoveConflict("common-file"))
|
||||
|
||||
var commitID Oid
|
||||
checkFatal(t, rebase.Commit(&commitID, signature(), signature(), "rebased message"))
|
||||
checkFatal(t, rebase.Finish())
|
||||
|
||||
// And then assert that we can look up the new merge commit, and that the
|
||||
// "common-file" has the expected contents.
|
||||
commit, err := repo.LookupCommit(&commitID)
|
||||
checkFatal(t, err)
|
||||
if commit.Message() != "rebased message" {
|
||||
t.Fatalf("unexpected commit message %q", commit.Message())
|
||||
}
|
||||
|
||||
tree, err := commit.Tree()
|
||||
checkFatal(t, err)
|
||||
|
||||
blob, err := repo.LookupBlob(tree.EntryByName("common-file").Id)
|
||||
checkFatal(t, err)
|
||||
if string(blob.Contents()) != "resolved contents" {
|
||||
t.Fatalf("unexpected resolved blob contents %q", string(blob.Contents()))
|
||||
}
|
||||
}
|
||||
|
||||
func TestRebaseAbort(t *testing.T) {
|
||||
// TEST DATA
|
||||
|
||||
|
@ -38,12 +112,12 @@ func TestRebaseAbort(t *testing.T) {
|
|||
seedTestRepo(t, repo)
|
||||
|
||||
// Setup a repo with 2 branches and a different tree
|
||||
err := setupRepoForRebase(repo, masterCommit, branchName, commitOpts{})
|
||||
err := setupRepoForRebase(repo, masterCommit, branchName, commitOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
// Create several commits in emile
|
||||
for _, commit := range emileCommits {
|
||||
_, err = commitSomething(repo, commit, commit, commitOpts{})
|
||||
_, err = commitSomething(repo, commit, commit, commitOptions{})
|
||||
checkFatal(t, err)
|
||||
}
|
||||
|
||||
|
@ -99,12 +173,12 @@ func TestRebaseNoConflicts(t *testing.T) {
|
|||
}
|
||||
|
||||
// Setup a repo with 2 branches and a different tree
|
||||
err = setupRepoForRebase(repo, masterCommit, branchName, commitOpts{})
|
||||
err = setupRepoForRebase(repo, masterCommit, branchName, commitOptions{})
|
||||
checkFatal(t, err)
|
||||
|
||||
// Create several commits in emile
|
||||
for _, commit := range emileCommits {
|
||||
_, err = commitSomething(repo, commit, commit, commitOpts{})
|
||||
_, err = commitSomething(repo, commit, commit, commitOptions{})
|
||||
checkFatal(t, err)
|
||||
}
|
||||
|
||||
|
@ -143,7 +217,7 @@ func TestRebaseGpgSigned(t *testing.T) {
|
|||
entity, err := openpgp.NewEntity("Namey mcnameface", "test comment", "test@example.com", nil)
|
||||
checkFatal(t, err)
|
||||
|
||||
opts, err := DefaultRebaseOptions()
|
||||
rebaseOpts, err := DefaultRebaseOptions()
|
||||
checkFatal(t, err)
|
||||
|
||||
signCommitContent := func(commitContent string) (string, string, error) {
|
||||
|
@ -155,9 +229,9 @@ func TestRebaseGpgSigned(t *testing.T) {
|
|||
|
||||
return cipherText.String(), "", nil
|
||||
}
|
||||
opts.CommitSigningCallback = signCommitContent
|
||||
rebaseOpts.CommitSigningCallback = signCommitContent
|
||||
|
||||
commitOpts := commitOpts{
|
||||
commitOpts := commitOptions{
|
||||
CommitSigningCallback: signCommitContent,
|
||||
}
|
||||
|
||||
|
@ -201,7 +275,7 @@ func TestRebaseGpgSigned(t *testing.T) {
|
|||
}
|
||||
|
||||
// Rebase onto master
|
||||
rebase, err := performRebaseOnto(repo, "master", &opts)
|
||||
rebase, err := performRebaseOnto(repo, "master", &rebaseOpts)
|
||||
checkFatal(t, err)
|
||||
defer rebase.Free()
|
||||
|
||||
|
@ -245,7 +319,7 @@ func checkCommitSigned(t *testing.T, entity *openpgp.Entity, commit *Commit) err
|
|||
return err
|
||||
}
|
||||
|
||||
_, err = openpgp.CheckArmoredDetachedSignature(openpgp.EntityList{entity}, strings.NewReader(signedData), bytes.NewBufferString(signature))
|
||||
_, err = openpgp.CheckArmoredDetachedSignature(openpgp.EntityList{entity}, strings.NewReader(signedData), bytes.NewBufferString(signature), nil)
|
||||
if err != nil {
|
||||
t.Logf("Commit is not signed correctly\n%s", commit.ContentToSign())
|
||||
return err
|
||||
|
@ -255,7 +329,7 @@ func checkCommitSigned(t *testing.T, entity *openpgp.Entity, commit *Commit) err
|
|||
}
|
||||
|
||||
// Utils
|
||||
func setupRepoForRebase(repo *Repository, masterCommit, branchName string, opts commitOpts) error {
|
||||
func setupRepoForRebase(repo *Repository, masterCommit, branchName string, commitOpts commitOptions) error {
|
||||
// Create a new branch from master
|
||||
err := createBranch(repo, branchName)
|
||||
if err != nil {
|
||||
|
@ -263,7 +337,7 @@ func setupRepoForRebase(repo *Repository, masterCommit, branchName string, opts
|
|||
}
|
||||
|
||||
// Create a commit in master
|
||||
_, err = commitSomething(repo, masterCommit, masterCommit, opts)
|
||||
_, err = commitSomething(repo, masterCommit, masterCommit, commitOpts)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
@ -282,7 +356,7 @@ func setupRepoForRebase(repo *Repository, masterCommit, branchName string, opts
|
|||
return nil
|
||||
}
|
||||
|
||||
func performRebaseOnto(repo *Repository, branch string, opts *RebaseOptions) (*Rebase, error) {
|
||||
func performRebaseOnto(repo *Repository, branch string, rebaseOpts *RebaseOptions) (*Rebase, error) {
|
||||
master, err := repo.LookupBranch(branch, BranchLocal)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
|
@ -296,7 +370,7 @@ func performRebaseOnto(repo *Repository, branch string, opts *RebaseOptions) (*R
|
|||
defer onto.Free()
|
||||
|
||||
// Init rebase
|
||||
rebase, err := repo.InitRebase(nil, nil, onto, opts)
|
||||
rebase, err := repo.InitRebase(nil, nil, onto, rebaseOpts)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
@ -397,7 +471,7 @@ func headTree(repo *Repository) (*Tree, error) {
|
|||
return tree, nil
|
||||
}
|
||||
|
||||
func commitSomething(repo *Repository, something, content string, commitOpts commitOpts) (*Oid, error) {
|
||||
func commitSomething(repo *Repository, something, content string, commitOpts commitOptions) (*Oid, error) {
|
||||
headCommit, err := headCommit(repo)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
|
@ -470,10 +544,10 @@ func commitSomething(repo *Repository, something, content string, commitOpts com
|
|||
}
|
||||
}
|
||||
|
||||
opts := &CheckoutOpts{
|
||||
checkoutOpts := &CheckoutOptions{
|
||||
Strategy: CheckoutRemoveUntracked | CheckoutForce,
|
||||
}
|
||||
err = repo.CheckoutIndex(index, opts)
|
||||
err = repo.CheckoutIndex(index, checkoutOpts)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
|
4
refdb.go
4
refdb.go
|
@ -13,11 +13,13 @@ import (
|
|||
)
|
||||
|
||||
type Refdb struct {
|
||||
doNotCompare
|
||||
ptr *C.git_refdb
|
||||
r *Repository
|
||||
}
|
||||
|
||||
type RefdbBackend struct {
|
||||
doNotCompare
|
||||
ptr *C.git_refdb_backend
|
||||
}
|
||||
|
||||
|
@ -38,7 +40,7 @@ func (v *Repository) NewRefdb() (refdb *Refdb, err error) {
|
|||
}
|
||||
|
||||
func NewRefdbBackendFromC(ptr unsafe.Pointer) (backend *RefdbBackend) {
|
||||
backend = &RefdbBackend{(*C.git_refdb_backend)(ptr)}
|
||||
backend = &RefdbBackend{ptr: (*C.git_refdb_backend)(ptr)}
|
||||
return backend
|
||||
}
|
||||
|
||||
|
|
77
reference.go
77
reference.go
|
@ -17,11 +17,13 @@ const (
|
|||
)
|
||||
|
||||
type Reference struct {
|
||||
doNotCompare
|
||||
ptr *C.git_reference
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
type ReferenceCollection struct {
|
||||
doNotCompare
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
|
@ -293,12 +295,14 @@ func (v *Reference) Peel(t ObjectType) (*Object, error) {
|
|||
return allocObject(cobj, v.repo), nil
|
||||
}
|
||||
|
||||
// Owner returns a weak reference to the repository which owns this
|
||||
// reference.
|
||||
// Owner returns a weak reference to the repository which owns this reference.
|
||||
// This won't keep the underlying repository alive, but it should still be
|
||||
// Freed.
|
||||
func (v *Reference) Owner() *Repository {
|
||||
return &Repository{
|
||||
ptr: C.git_reference_owner(v.ptr),
|
||||
}
|
||||
repo := newRepositoryFromC(C.git_reference_owner(v.ptr))
|
||||
runtime.KeepAlive(v)
|
||||
repo.weak = true
|
||||
return repo
|
||||
}
|
||||
|
||||
// Cmp compares v to ref2. It returns 0 on equality, otherwise a
|
||||
|
@ -361,11 +365,13 @@ func (v *Reference) Free() {
|
|||
}
|
||||
|
||||
type ReferenceIterator struct {
|
||||
doNotCompare
|
||||
ptr *C.git_reference_iterator
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
type ReferenceNameIterator struct {
|
||||
doNotCompare
|
||||
*ReferenceIterator
|
||||
}
|
||||
|
||||
|
@ -420,11 +426,11 @@ func (repo *Repository) NewReferenceIteratorGlob(glob string) (*ReferenceIterato
|
|||
}
|
||||
|
||||
func (i *ReferenceIterator) Names() *ReferenceNameIterator {
|
||||
return &ReferenceNameIterator{i}
|
||||
return &ReferenceNameIterator{ReferenceIterator: i}
|
||||
}
|
||||
|
||||
// NextName retrieves the next reference name. If the iteration is over,
|
||||
// the returned error is git.ErrIterOver
|
||||
// the returned error code is git.ErrorCodeIterOver
|
||||
func (v *ReferenceNameIterator) Next() (string, error) {
|
||||
var ptr *C.char
|
||||
|
||||
|
@ -440,7 +446,7 @@ func (v *ReferenceNameIterator) Next() (string, error) {
|
|||
}
|
||||
|
||||
// Next retrieves the next reference. If the iterationis over, the
|
||||
// returned error is git.ErrIterOver
|
||||
// returned error code is git.ErrorCodeIterOver
|
||||
func (v *ReferenceIterator) Next() (*Reference, error) {
|
||||
var ptr *C.git_reference
|
||||
|
||||
|
@ -470,7 +476,7 @@ func (v *ReferenceIterator) Free() {
|
|||
C.git_reference_iterator_free(v.ptr)
|
||||
}
|
||||
|
||||
// ReferenceIsValidName ensures the reference name is well-formed.
|
||||
// ReferenceNameIsValid returns whether the reference name is well-formed.
|
||||
//
|
||||
// Valid reference names must follow one of two patterns:
|
||||
//
|
||||
|
@ -480,11 +486,56 @@ func (v *ReferenceIterator) Free() {
|
|||
// 2. Names prefixed with "refs/" can be almost anything. You must avoid
|
||||
// the characters '~', '^', ':', ' \ ', '?', '[', and '*', and the sequences
|
||||
// ".." and " @ {" which have special meaning to revparse.
|
||||
func ReferenceIsValidName(name string) bool {
|
||||
func ReferenceNameIsValid(name string) (bool, error) {
|
||||
cname := C.CString(name)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
if C.git_reference_is_valid_name(cname) == 1 {
|
||||
return true
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var valid C.int
|
||||
ret := C.git_reference_name_is_valid(&valid, cname)
|
||||
if ret < 0 {
|
||||
return false, MakeGitError(ret)
|
||||
}
|
||||
return false
|
||||
return valid == 1, nil
|
||||
}
|
||||
|
||||
const (
|
||||
// This should match GIT_REFNAME_MAX in src/refs.h
|
||||
_refnameMaxLength = C.size_t(1024)
|
||||
)
|
||||
|
||||
type ReferenceFormat uint
|
||||
|
||||
const (
|
||||
ReferenceFormatNormal ReferenceFormat = C.GIT_REFERENCE_FORMAT_NORMAL
|
||||
ReferenceFormatAllowOnelevel ReferenceFormat = C.GIT_REFERENCE_FORMAT_ALLOW_ONELEVEL
|
||||
ReferenceFormatRefspecPattern ReferenceFormat = C.GIT_REFERENCE_FORMAT_REFSPEC_PATTERN
|
||||
ReferenceFormatRefspecShorthand ReferenceFormat = C.GIT_REFERENCE_FORMAT_REFSPEC_SHORTHAND
|
||||
)
|
||||
|
||||
// ReferenceNormalizeName normalizes the reference name and checks validity.
|
||||
//
|
||||
// This will normalize the reference name by removing any leading slash '/'
|
||||
// characters and collapsing runs of adjacent slashes between name components
|
||||
// into a single slash.
|
||||
//
|
||||
// See git_reference_symbolic_create() for rules about valid names.
|
||||
func ReferenceNormalizeName(name string, flags ReferenceFormat) (string, error) {
|
||||
cname := C.CString(name)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
|
||||
buf := (*C.char)(C.malloc(_refnameMaxLength))
|
||||
defer C.free(unsafe.Pointer(buf))
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_reference_normalize_name(buf, _refnameMaxLength, cname, C.uint(flags))
|
||||
if ecode < 0 {
|
||||
return "", MakeGitError(ecode)
|
||||
}
|
||||
|
||||
return C.GoString(buf), nil
|
||||
}
|
||||
|
|
|
@ -106,7 +106,7 @@ func TestReferenceIterator(t *testing.T) {
|
|||
list = append(list, name)
|
||||
name, err = nameIter.Next()
|
||||
}
|
||||
if !IsErrorCode(err, ErrIterOver) {
|
||||
if !IsErrorCode(err, ErrorCodeIterOver) {
|
||||
t.Fatal("Iteration not over")
|
||||
}
|
||||
|
||||
|
@ -122,7 +122,7 @@ func TestReferenceIterator(t *testing.T) {
|
|||
count++
|
||||
_, err = iter.Next()
|
||||
}
|
||||
if !IsErrorCode(err, ErrIterOver) {
|
||||
if !IsErrorCode(err, ErrorCodeIterOver) {
|
||||
t.Fatal("Iteration not over")
|
||||
}
|
||||
|
||||
|
@ -214,16 +214,43 @@ func TestIsNote(t *testing.T) {
|
|||
}
|
||||
}
|
||||
|
||||
func TestReferenceIsValidName(t *testing.T) {
|
||||
func TestReferenceNameIsValid(t *testing.T) {
|
||||
t.Parallel()
|
||||
if !ReferenceIsValidName("HEAD") {
|
||||
valid, err := ReferenceNameIsValid("HEAD")
|
||||
checkFatal(t, err)
|
||||
if !valid {
|
||||
t.Errorf("HEAD should be a valid reference name")
|
||||
}
|
||||
if ReferenceIsValidName("HEAD1") {
|
||||
valid, err = ReferenceNameIsValid("HEAD1")
|
||||
checkFatal(t, err)
|
||||
if valid {
|
||||
t.Errorf("HEAD1 should not be a valid reference name")
|
||||
}
|
||||
}
|
||||
|
||||
func TestReferenceNormalizeName(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
ref, err := ReferenceNormalizeName("refs/heads//master", ReferenceFormatNormal)
|
||||
checkFatal(t, err)
|
||||
|
||||
if ref != "refs/heads/master" {
|
||||
t.Errorf("ReferenceNormalizeName(%q) = %q; want %q", "refs/heads//master", ref, "refs/heads/master")
|
||||
}
|
||||
|
||||
ref, err = ReferenceNormalizeName("master", ReferenceFormatAllowOnelevel|ReferenceFormatRefspecShorthand)
|
||||
checkFatal(t, err)
|
||||
|
||||
if ref != "master" {
|
||||
t.Errorf("ReferenceNormalizeName(%q) = %q; want %q", "master", ref, "master")
|
||||
}
|
||||
|
||||
ref, err = ReferenceNormalizeName("foo^", ReferenceFormatNormal)
|
||||
if !IsErrorCode(err, ErrorCodeInvalidSpec) {
|
||||
t.Errorf("foo^ should be invalid")
|
||||
}
|
||||
}
|
||||
|
||||
func compareStringList(t *testing.T, expected, actual []string) {
|
||||
for i, v := range expected {
|
||||
if actual[i] != v {
|
||||
|
|
|
@ -0,0 +1,149 @@
|
|||
package git
|
||||
|
||||
/*
|
||||
#include <git2.h>
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"runtime"
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
type Refspec struct {
|
||||
doNotCompare
|
||||
ptr *C.git_refspec
|
||||
}
|
||||
|
||||
// ParseRefspec parses a given refspec string
|
||||
func ParseRefspec(input string, isFetch bool) (*Refspec, error) {
|
||||
var ptr *C.git_refspec
|
||||
|
||||
cinput := C.CString(input)
|
||||
defer C.free(unsafe.Pointer(cinput))
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_refspec_parse(&ptr, cinput, cbool(isFetch))
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
spec := &Refspec{ptr: ptr}
|
||||
runtime.SetFinalizer(spec, (*Refspec).Free)
|
||||
return spec, nil
|
||||
}
|
||||
|
||||
// Free releases a refspec object which has been created by ParseRefspec
|
||||
func (s *Refspec) Free() {
|
||||
runtime.SetFinalizer(s, nil)
|
||||
C.git_refspec_free(s.ptr)
|
||||
}
|
||||
|
||||
// Direction returns the refspec's direction
|
||||
func (s *Refspec) Direction() ConnectDirection {
|
||||
direction := C.git_refspec_direction(s.ptr)
|
||||
return ConnectDirection(direction)
|
||||
}
|
||||
|
||||
// Src returns the refspec's source specifier
|
||||
func (s *Refspec) Src() string {
|
||||
var ret string
|
||||
cstr := C.git_refspec_src(s.ptr)
|
||||
|
||||
if cstr != nil {
|
||||
ret = C.GoString(cstr)
|
||||
}
|
||||
|
||||
runtime.KeepAlive(s)
|
||||
return ret
|
||||
}
|
||||
|
||||
// Dst returns the refspec's destination specifier
|
||||
func (s *Refspec) Dst() string {
|
||||
var ret string
|
||||
cstr := C.git_refspec_dst(s.ptr)
|
||||
|
||||
if cstr != nil {
|
||||
ret = C.GoString(cstr)
|
||||
}
|
||||
|
||||
runtime.KeepAlive(s)
|
||||
return ret
|
||||
}
|
||||
|
||||
// Force returns the refspec's force-update setting
|
||||
func (s *Refspec) Force() bool {
|
||||
force := C.git_refspec_force(s.ptr)
|
||||
return force != 0
|
||||
}
|
||||
|
||||
// String returns the refspec's string representation
|
||||
func (s *Refspec) String() string {
|
||||
var ret string
|
||||
cstr := C.git_refspec_string(s.ptr)
|
||||
|
||||
if cstr != nil {
|
||||
ret = C.GoString(cstr)
|
||||
}
|
||||
|
||||
runtime.KeepAlive(s)
|
||||
return ret
|
||||
}
|
||||
|
||||
// SrcMatches checks if a refspec's source descriptor matches a reference
|
||||
func (s *Refspec) SrcMatches(refname string) bool {
|
||||
cname := C.CString(refname)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
|
||||
matches := C.git_refspec_src_matches(s.ptr, cname)
|
||||
return matches != 0
|
||||
}
|
||||
|
||||
// SrcMatches checks if a refspec's destination descriptor matches a reference
|
||||
func (s *Refspec) DstMatches(refname string) bool {
|
||||
cname := C.CString(refname)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
|
||||
matches := C.git_refspec_dst_matches(s.ptr, cname)
|
||||
return matches != 0
|
||||
}
|
||||
|
||||
// Transform a reference to its target following the refspec's rules
|
||||
func (s *Refspec) Transform(refname string) (string, error) {
|
||||
buf := C.git_buf{}
|
||||
|
||||
cname := C.CString(refname)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_refspec_transform(&buf, s.ptr, cname)
|
||||
if ret < 0 {
|
||||
return "", MakeGitError(ret)
|
||||
}
|
||||
defer C.git_buf_dispose(&buf)
|
||||
|
||||
return C.GoString(buf.ptr), nil
|
||||
}
|
||||
|
||||
// Rtransform converts a target reference to its source reference following the
|
||||
// refspec's rules
|
||||
func (s *Refspec) Rtransform(refname string) (string, error) {
|
||||
buf := C.git_buf{}
|
||||
|
||||
cname := C.CString(refname)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_refspec_rtransform(&buf, s.ptr, cname)
|
||||
if ret < 0 {
|
||||
return "", MakeGitError(ret)
|
||||
}
|
||||
defer C.git_buf_dispose(&buf)
|
||||
|
||||
return C.GoString(buf.ptr), nil
|
||||
}
|
|
@ -0,0 +1,75 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestRefspec(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
const (
|
||||
input = "+refs/heads/*:refs/remotes/origin/*"
|
||||
mainLocal = "refs/heads/main"
|
||||
mainRemote = "refs/remotes/origin/main"
|
||||
)
|
||||
|
||||
refspec, err := ParseRefspec(input, true)
|
||||
checkFatal(t, err)
|
||||
|
||||
// Accessors
|
||||
|
||||
s := refspec.String()
|
||||
if s != input {
|
||||
t.Errorf("expected string %q, got %q", input, s)
|
||||
}
|
||||
|
||||
if d := refspec.Direction(); d != ConnectDirectionFetch {
|
||||
t.Errorf("expected fetch refspec, got direction %v", d)
|
||||
}
|
||||
|
||||
if pat, expected := refspec.Src(), "refs/heads/*"; pat != expected {
|
||||
t.Errorf("expected refspec src %q, got %q", expected, pat)
|
||||
}
|
||||
|
||||
if pat, expected := refspec.Dst(), "refs/remotes/origin/*"; pat != expected {
|
||||
t.Errorf("expected refspec dst %q, got %q", expected, pat)
|
||||
}
|
||||
|
||||
if !refspec.Force() {
|
||||
t.Error("expected refspec force flag")
|
||||
}
|
||||
|
||||
// SrcMatches
|
||||
|
||||
if !refspec.SrcMatches(mainLocal) {
|
||||
t.Errorf("refspec source did not match %q", mainLocal)
|
||||
}
|
||||
|
||||
if refspec.SrcMatches("refs/tags/v1.0") {
|
||||
t.Error("refspec source matched under refs/tags")
|
||||
}
|
||||
|
||||
// DstMatches
|
||||
|
||||
if !refspec.DstMatches(mainRemote) {
|
||||
t.Errorf("refspec destination did not match %q", mainRemote)
|
||||
}
|
||||
|
||||
if refspec.DstMatches("refs/tags/v1.0") {
|
||||
t.Error("refspec destination matched under refs/tags")
|
||||
}
|
||||
|
||||
// Transforms
|
||||
|
||||
fromLocal, err := refspec.Transform(mainLocal)
|
||||
checkFatal(t, err)
|
||||
if fromLocal != mainRemote {
|
||||
t.Errorf("transform by refspec returned %s; expected %s", fromLocal, mainRemote)
|
||||
}
|
||||
|
||||
fromRemote, err := refspec.Rtransform(mainRemote)
|
||||
checkFatal(t, err)
|
||||
if fromRemote != mainLocal {
|
||||
t.Errorf("rtransform by refspec returned %s; expected %s", fromRemote, mainLocal)
|
||||
}
|
||||
}
|
701
remote.go
701
remote.go
|
@ -6,18 +6,39 @@ package git
|
|||
#include <git2.h>
|
||||
#include <git2/sys/cred.h>
|
||||
|
||||
extern void _go_git_setup_callbacks(git_remote_callbacks *callbacks);
|
||||
|
||||
extern void _go_git_populate_remote_callbacks(git_remote_callbacks *callbacks);
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"crypto/x509"
|
||||
"errors"
|
||||
"fmt"
|
||||
"reflect"
|
||||
"runtime"
|
||||
"strings"
|
||||
"sync"
|
||||
"unsafe"
|
||||
|
||||
"golang.org/x/crypto/ssh"
|
||||
)
|
||||
|
||||
// RemoteCreateOptionsFlag is Remote creation options flags
|
||||
type RemoteCreateOptionsFlag uint
|
||||
|
||||
const (
|
||||
// Ignore the repository apply.insteadOf configuration
|
||||
RemoteCreateSkipInsteadof RemoteCreateOptionsFlag = C.GIT_REMOTE_CREATE_SKIP_INSTEADOF
|
||||
// Don't build a fetchspec from the name if none is set
|
||||
RemoteCreateSkipDefaultFetchspec RemoteCreateOptionsFlag = C.GIT_REMOTE_CREATE_SKIP_DEFAULT_FETCHSPEC
|
||||
)
|
||||
|
||||
// RemoteCreateOptions contains options for creating a remote
|
||||
type RemoteCreateOptions struct {
|
||||
Name string
|
||||
FetchSpec string
|
||||
Flags RemoteCreateOptionsFlag
|
||||
}
|
||||
|
||||
type TransferProgress struct {
|
||||
TotalObjects uint
|
||||
IndexedObjects uint
|
||||
|
@ -49,15 +70,15 @@ const (
|
|||
ConnectDirectionPush ConnectDirection = C.GIT_DIRECTION_PUSH
|
||||
)
|
||||
|
||||
type TransportMessageCallback func(str string) ErrorCode
|
||||
type CompletionCallback func(RemoteCompletion) ErrorCode
|
||||
type CredentialsCallback func(url string, username_from_url string, allowed_types CredType) (*Cred, error)
|
||||
type TransferProgressCallback func(stats TransferProgress) ErrorCode
|
||||
type UpdateTipsCallback func(refname string, a *Oid, b *Oid) ErrorCode
|
||||
type CertificateCheckCallback func(cert *Certificate, valid bool, hostname string) ErrorCode
|
||||
type PackbuilderProgressCallback func(stage int32, current, total uint32) ErrorCode
|
||||
type PushTransferProgressCallback func(current, total uint32, bytes uint) ErrorCode
|
||||
type PushUpdateReferenceCallback func(refname, status string) ErrorCode
|
||||
type TransportMessageCallback func(str string) error
|
||||
type CompletionCallback func(RemoteCompletion) error
|
||||
type CredentialsCallback func(url string, username_from_url string, allowed_types CredentialType) (*Credential, error)
|
||||
type TransferProgressCallback func(stats TransferProgress) error
|
||||
type UpdateTipsCallback func(refname string, a *Oid, b *Oid) error
|
||||
type CertificateCheckCallback func(cert *Certificate, valid bool, hostname string) error
|
||||
type PackbuilderProgressCallback func(stage int32, current, total uint32) error
|
||||
type PushTransferProgressCallback func(current, total uint32, bytes uint) error
|
||||
type PushUpdateReferenceCallback func(refname, status string) error
|
||||
|
||||
type RemoteCallbacks struct {
|
||||
SidebandProgressCallback TransportMessageCallback
|
||||
|
@ -71,6 +92,11 @@ type RemoteCallbacks struct {
|
|||
PushUpdateReferenceCallback
|
||||
}
|
||||
|
||||
type remoteCallbacksData struct {
|
||||
callbacks *RemoteCallbacks
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
type FetchPrune uint
|
||||
|
||||
const (
|
||||
|
@ -85,7 +111,6 @@ const (
|
|||
type DownloadTags uint
|
||||
|
||||
const (
|
||||
|
||||
// Use the setting from the configuration.
|
||||
DownloadTagsUnspecified DownloadTags = C.GIT_REMOTE_DOWNLOAD_TAGS_UNSPECIFIED
|
||||
// Ask the server for tags pointing to objects we're already
|
||||
|
@ -122,6 +147,17 @@ type FetchOptions struct {
|
|||
ProxyOptions ProxyOptions
|
||||
}
|
||||
|
||||
type RemoteConnectOptions struct {
|
||||
// Proxy options to use for this fetch operation
|
||||
ProxyOptions ProxyOptions
|
||||
}
|
||||
|
||||
func remoteConnectOptionsFromC(copts *C.git_remote_connect_options) *RemoteConnectOptions {
|
||||
return &RemoteConnectOptions{
|
||||
ProxyOptions: proxyOptionsFromC(&copts.proxy_opts),
|
||||
}
|
||||
}
|
||||
|
||||
type ProxyType uint
|
||||
|
||||
const (
|
||||
|
@ -146,10 +182,79 @@ type ProxyOptions struct {
|
|||
Url string
|
||||
}
|
||||
|
||||
func proxyOptionsFromC(copts *C.git_proxy_options) ProxyOptions {
|
||||
return ProxyOptions{
|
||||
Type: ProxyType(copts._type),
|
||||
Url: C.GoString(copts.url),
|
||||
}
|
||||
}
|
||||
|
||||
type Remote struct {
|
||||
doNotCompare
|
||||
ptr *C.git_remote
|
||||
callbacks RemoteCallbacks
|
||||
repo *Repository
|
||||
// weak indicates that a remote is a weak pointer and should not be
|
||||
// freed.
|
||||
weak bool
|
||||
}
|
||||
|
||||
type remotePointerList struct {
|
||||
sync.RWMutex
|
||||
// stores the Go pointers
|
||||
pointers map[*C.git_remote]*Remote
|
||||
}
|
||||
|
||||
func newRemotePointerList() *remotePointerList {
|
||||
return &remotePointerList{
|
||||
pointers: make(map[*C.git_remote]*Remote),
|
||||
}
|
||||
}
|
||||
|
||||
// track adds the given pointer to the list of pointers to track and
|
||||
// returns a pointer value which can be passed to C as an opaque
|
||||
// pointer.
|
||||
func (v *remotePointerList) track(remote *Remote) {
|
||||
v.Lock()
|
||||
v.pointers[remote.ptr] = remote
|
||||
v.Unlock()
|
||||
|
||||
runtime.SetFinalizer(remote, (*Remote).Free)
|
||||
}
|
||||
|
||||
// untrack stops tracking the git_remote pointer.
|
||||
func (v *remotePointerList) untrack(remote *Remote) {
|
||||
v.Lock()
|
||||
delete(v.pointers, remote.ptr)
|
||||
v.Unlock()
|
||||
}
|
||||
|
||||
// clear stops tracking all the git_remote pointers.
|
||||
func (v *remotePointerList) clear() {
|
||||
v.Lock()
|
||||
var remotes []*Remote
|
||||
for remotePtr, remote := range v.pointers {
|
||||
remotes = append(remotes, remote)
|
||||
delete(v.pointers, remotePtr)
|
||||
}
|
||||
v.Unlock()
|
||||
|
||||
for _, remote := range remotes {
|
||||
remote.free()
|
||||
}
|
||||
}
|
||||
|
||||
// get retrieves the pointer from the given *git_remote.
|
||||
func (v *remotePointerList) get(ptr *C.git_remote) (*Remote, bool) {
|
||||
v.RLock()
|
||||
defer v.RUnlock()
|
||||
|
||||
r, ok := v.pointers[ptr]
|
||||
if !ok {
|
||||
return nil, false
|
||||
}
|
||||
|
||||
return r, true
|
||||
}
|
||||
|
||||
type CertificateKind uint
|
||||
|
@ -162,27 +267,32 @@ const (
|
|||
// Certificate represents the two possible certificates which libgit2
|
||||
// knows it might find. If Kind is CertficateX509 then the X509 field
|
||||
// will be filled. If Kind is CertificateHostkey then the Hostkey
|
||||
// field will be fille.d
|
||||
// field will be filled.
|
||||
type Certificate struct {
|
||||
Kind CertificateKind
|
||||
X509 *x509.Certificate
|
||||
Hostkey HostkeyCertificate
|
||||
}
|
||||
|
||||
// HostkeyKind is a bitmask of the available hashes in HostkeyCertificate.
|
||||
type HostkeyKind uint
|
||||
|
||||
const (
|
||||
HostkeyMD5 HostkeyKind = C.GIT_CERT_SSH_MD5
|
||||
HostkeySHA1 HostkeyKind = C.GIT_CERT_SSH_SHA1
|
||||
HostkeyMD5 HostkeyKind = C.GIT_CERT_SSH_MD5
|
||||
HostkeySHA1 HostkeyKind = C.GIT_CERT_SSH_SHA1
|
||||
HostkeySHA256 HostkeyKind = C.GIT_CERT_SSH_SHA256
|
||||
HostkeyRaw HostkeyKind = C.GIT_CERT_SSH_RAW
|
||||
)
|
||||
|
||||
// Server host key information. If Kind is HostkeyMD5 the MD5 field
|
||||
// will be filled. If Kind is HostkeySHA1, then HashSHA1 will be
|
||||
// filled.
|
||||
// Server host key information. A bitmask containing the available fields.
|
||||
// Check for combinations of: HostkeyMD5, HostkeySHA1, HostkeySHA256, HostkeyRaw.
|
||||
type HostkeyCertificate struct {
|
||||
Kind HostkeyKind
|
||||
HashMD5 [16]byte
|
||||
HashSHA1 [20]byte
|
||||
Kind HostkeyKind
|
||||
HashMD5 [16]byte
|
||||
HashSHA1 [20]byte
|
||||
HashSHA256 [32]byte
|
||||
Hostkey []byte
|
||||
SSHPublicKey ssh.PublicKey
|
||||
}
|
||||
|
||||
type PushOptions struct {
|
||||
|
@ -193,6 +303,9 @@ type PushOptions struct {
|
|||
|
||||
// Headers are extra headers for the push operation.
|
||||
Headers []string
|
||||
|
||||
// Proxy options to use for this push operation
|
||||
ProxyOptions ProxyOptions
|
||||
}
|
||||
|
||||
type RemoteHead struct {
|
||||
|
@ -207,54 +320,80 @@ func newRemoteHeadFromC(ptr *C.git_remote_head) RemoteHead {
|
|||
}
|
||||
}
|
||||
|
||||
func untrackCalbacksPayload(callbacks *C.git_remote_callbacks) {
|
||||
if callbacks != nil && callbacks.payload != nil {
|
||||
pointerHandles.Untrack(callbacks.payload)
|
||||
}
|
||||
}
|
||||
|
||||
func populateRemoteCallbacks(ptr *C.git_remote_callbacks, callbacks *RemoteCallbacks) {
|
||||
C.git_remote_init_callbacks(ptr, C.GIT_REMOTE_CALLBACKS_VERSION)
|
||||
if callbacks == nil {
|
||||
func untrackCallbacksPayload(callbacks *C.git_remote_callbacks) {
|
||||
if callbacks == nil || callbacks.payload == nil {
|
||||
return
|
||||
}
|
||||
C._go_git_setup_callbacks(ptr)
|
||||
ptr.payload = pointerHandles.Track(callbacks)
|
||||
pointerHandles.Untrack(callbacks.payload)
|
||||
}
|
||||
|
||||
func populateRemoteCallbacks(ptr *C.git_remote_callbacks, callbacks *RemoteCallbacks, errorTarget *error) *C.git_remote_callbacks {
|
||||
C.git_remote_init_callbacks(ptr, C.GIT_REMOTE_CALLBACKS_VERSION)
|
||||
if callbacks == nil {
|
||||
return ptr
|
||||
}
|
||||
C._go_git_populate_remote_callbacks(ptr)
|
||||
data := &remoteCallbacksData{
|
||||
callbacks: callbacks,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
ptr.payload = pointerHandles.Track(data)
|
||||
return ptr
|
||||
}
|
||||
|
||||
//export sidebandProgressCallback
|
||||
func sidebandProgressCallback(_str *C.char, _len C.int, data unsafe.Pointer) int {
|
||||
callbacks := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
if callbacks.SidebandProgressCallback == nil {
|
||||
return 0
|
||||
func sidebandProgressCallback(errorMessage **C.char, _str *C.char, _len C.int, handle unsafe.Pointer) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.SidebandProgressCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
str := C.GoStringN(_str, _len)
|
||||
return int(callbacks.SidebandProgressCallback(str))
|
||||
err := data.callbacks.SidebandProgressCallback(C.GoStringN(_str, _len))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export completionCallback
|
||||
func completionCallback(completion_type C.git_remote_completion_type, data unsafe.Pointer) int {
|
||||
callbacks := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
if callbacks.CompletionCallback == nil {
|
||||
return 0
|
||||
func completionCallback(errorMessage **C.char, completionType C.git_remote_completion_type, handle unsafe.Pointer) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.CompletionCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
return int(callbacks.CompletionCallback(RemoteCompletion(completion_type)))
|
||||
err := data.callbacks.CompletionCallback(RemoteCompletion(completionType))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export credentialsCallback
|
||||
func credentialsCallback(_cred **C.git_cred, _url *C.char, _username_from_url *C.char, allowed_types uint, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
if callbacks.CredentialsCallback == nil {
|
||||
return C.GIT_PASSTHROUGH
|
||||
func credentialsCallback(
|
||||
errorMessage **C.char,
|
||||
_cred **C.git_credential,
|
||||
_url *C.char,
|
||||
_username_from_url *C.char,
|
||||
allowed_types uint,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.CredentialsCallback == nil {
|
||||
return C.int(ErrorCodePassthrough)
|
||||
}
|
||||
url := C.GoString(_url)
|
||||
username_from_url := C.GoString(_username_from_url)
|
||||
cred, err := callbacks.CredentialsCallback(url, username_from_url, (CredType)(allowed_types))
|
||||
cred, err := data.callbacks.CredentialsCallback(url, username_from_url, (CredentialType)(allowed_types))
|
||||
if err != nil {
|
||||
if gitError, ok := err.(*GitError); ok {
|
||||
return int(gitError.Code)
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return C.GIT_EUSER
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
if cred != nil {
|
||||
*_cred = cred.ptr
|
||||
|
@ -263,41 +402,67 @@ func credentialsCallback(_cred **C.git_cred, _url *C.char, _username_from_url *C
|
|||
cred.ptr = nil
|
||||
runtime.SetFinalizer(cred, nil)
|
||||
}
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export transferProgressCallback
|
||||
func transferProgressCallback(stats *C.git_transfer_progress, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
if callbacks.TransferProgressCallback == nil {
|
||||
return 0
|
||||
func transferProgressCallback(errorMessage **C.char, stats *C.git_transfer_progress, handle unsafe.Pointer) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.TransferProgressCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
return int(callbacks.TransferProgressCallback(newTransferProgressFromC(stats)))
|
||||
err := data.callbacks.TransferProgressCallback(newTransferProgressFromC(stats))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export updateTipsCallback
|
||||
func updateTipsCallback(_refname *C.char, _a *C.git_oid, _b *C.git_oid, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
if callbacks.UpdateTipsCallback == nil {
|
||||
return 0
|
||||
func updateTipsCallback(
|
||||
errorMessage **C.char,
|
||||
_refname *C.char,
|
||||
_a *C.git_oid,
|
||||
_b *C.git_oid,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.UpdateTipsCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
refname := C.GoString(_refname)
|
||||
a := newOidFromC(_a)
|
||||
b := newOidFromC(_b)
|
||||
return int(callbacks.UpdateTipsCallback(refname, a, b))
|
||||
err := data.callbacks.UpdateTipsCallback(refname, a, b)
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export certificateCheckCallback
|
||||
func certificateCheckCallback(_cert *C.git_cert, _valid C.int, _host *C.char, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
func certificateCheckCallback(
|
||||
errorMessage **C.char,
|
||||
_cert *C.git_cert,
|
||||
_valid C.int,
|
||||
_host *C.char,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
// if there's no callback set, we need to make sure we fail if the library didn't consider this cert valid
|
||||
if callbacks.CertificateCheckCallback == nil {
|
||||
if _valid == 1 {
|
||||
return 0
|
||||
} else {
|
||||
return C.GIT_ECERTIFICATE
|
||||
if data.callbacks.CertificateCheckCallback == nil {
|
||||
if _valid == 0 {
|
||||
return C.int(ErrorCodeCertificate)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
host := C.GoString(_host)
|
||||
valid := _valid != 0
|
||||
|
||||
|
@ -307,7 +472,17 @@ func certificateCheckCallback(_cert *C.git_cert, _valid C.int, _host *C.char, da
|
|||
ccert := (*C.git_cert_x509)(unsafe.Pointer(_cert))
|
||||
x509_certs, err := x509.ParseCertificates(C.GoBytes(ccert.data, C.int(ccert.len)))
|
||||
if err != nil {
|
||||
return C.GIT_EUSER
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
if len(x509_certs) < 1 {
|
||||
err := errors.New("empty certificate list")
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
// we assume there's only one, which should hold true for any web server we want to talk to
|
||||
|
@ -318,79 +493,161 @@ func certificateCheckCallback(_cert *C.git_cert, _valid C.int, _host *C.char, da
|
|||
cert.Hostkey.Kind = HostkeyKind(ccert._type)
|
||||
C.memcpy(unsafe.Pointer(&cert.Hostkey.HashMD5[0]), unsafe.Pointer(&ccert.hash_md5[0]), C.size_t(len(cert.Hostkey.HashMD5)))
|
||||
C.memcpy(unsafe.Pointer(&cert.Hostkey.HashSHA1[0]), unsafe.Pointer(&ccert.hash_sha1[0]), C.size_t(len(cert.Hostkey.HashSHA1)))
|
||||
C.memcpy(unsafe.Pointer(&cert.Hostkey.HashSHA256[0]), unsafe.Pointer(&ccert.hash_sha256[0]), C.size_t(len(cert.Hostkey.HashSHA256)))
|
||||
if (cert.Hostkey.Kind & HostkeyRaw) == HostkeyRaw {
|
||||
cert.Hostkey.Hostkey = C.GoBytes(unsafe.Pointer(ccert.hostkey), C.int(ccert.hostkey_len))
|
||||
var err error
|
||||
cert.Hostkey.SSHPublicKey, err = ssh.ParsePublicKey(cert.Hostkey.Hostkey)
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
}
|
||||
} else {
|
||||
cstr := C.CString("Unsupported certificate type")
|
||||
C.giterr_set_str(C.GITERR_NET, cstr)
|
||||
C.free(unsafe.Pointer(cstr))
|
||||
return -1 // we don't support anything else atm
|
||||
err := errors.New("unsupported certificate type")
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
return int(callbacks.CertificateCheckCallback(&cert, valid, host))
|
||||
err := data.callbacks.CertificateCheckCallback(&cert, valid, host)
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export packProgressCallback
|
||||
func packProgressCallback(stage C.int, current, total C.uint, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
|
||||
if callbacks.PackProgressCallback == nil {
|
||||
return 0
|
||||
func packProgressCallback(errorMessage **C.char, stage C.int, current, total C.uint, handle unsafe.Pointer) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.PackProgressCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
return int(callbacks.PackProgressCallback(int32(stage), uint32(current), uint32(total)))
|
||||
err := data.callbacks.PackProgressCallback(int32(stage), uint32(current), uint32(total))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export pushTransferProgressCallback
|
||||
func pushTransferProgressCallback(current, total C.uint, bytes C.size_t, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
if callbacks.PushTransferProgressCallback == nil {
|
||||
return 0
|
||||
func pushTransferProgressCallback(errorMessage **C.char, current, total C.uint, bytes C.size_t, handle unsafe.Pointer) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.PushTransferProgressCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
return int(callbacks.PushTransferProgressCallback(uint32(current), uint32(total), uint(bytes)))
|
||||
err := data.callbacks.PushTransferProgressCallback(uint32(current), uint32(total), uint(bytes))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export pushUpdateReferenceCallback
|
||||
func pushUpdateReferenceCallback(refname, status *C.char, data unsafe.Pointer) int {
|
||||
callbacks, _ := pointerHandles.Get(data).(*RemoteCallbacks)
|
||||
|
||||
if callbacks.PushUpdateReferenceCallback == nil {
|
||||
return 0
|
||||
func pushUpdateReferenceCallback(errorMessage **C.char, refname, status *C.char, handle unsafe.Pointer) C.int {
|
||||
data := pointerHandles.Get(handle).(*remoteCallbacksData)
|
||||
if data.callbacks.PushUpdateReferenceCallback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
return int(callbacks.PushUpdateReferenceCallback(C.GoString(refname), C.GoString(status)))
|
||||
err := data.callbacks.PushUpdateReferenceCallback(C.GoString(refname), C.GoString(status))
|
||||
if err != nil {
|
||||
if data.errorTarget != nil {
|
||||
*data.errorTarget = err
|
||||
}
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func populateProxyOptions(ptr *C.git_proxy_options, opts *ProxyOptions) {
|
||||
C.git_proxy_options_init(ptr, C.GIT_PROXY_OPTIONS_VERSION)
|
||||
func populateProxyOptions(copts *C.git_proxy_options, opts *ProxyOptions) *C.git_proxy_options {
|
||||
C.git_proxy_options_init(copts, C.GIT_PROXY_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
copts._type = C.git_proxy_t(opts.Type)
|
||||
copts.url = C.CString(opts.Url)
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeProxyOptions(copts *C.git_proxy_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
|
||||
ptr._type = C.git_proxy_t(opts.Type)
|
||||
ptr.url = C.CString(opts.Url)
|
||||
C.free(unsafe.Pointer(copts.url))
|
||||
}
|
||||
|
||||
func freeProxyOptions(ptr *C.git_proxy_options) {
|
||||
C.free(unsafe.Pointer(ptr.url))
|
||||
}
|
||||
|
||||
func RemoteIsValidName(name string) bool {
|
||||
// RemoteNameIsValid returns whether the remote name is well-formed.
|
||||
func RemoteNameIsValid(name string) (bool, error) {
|
||||
cname := C.CString(name)
|
||||
defer C.free(unsafe.Pointer(cname))
|
||||
if C.git_remote_is_valid_name(cname) == 1 {
|
||||
return true
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var valid C.int
|
||||
ret := C.git_remote_name_is_valid(&valid, cname)
|
||||
if ret < 0 {
|
||||
return false, MakeGitError(ret)
|
||||
}
|
||||
return false
|
||||
return valid == 1, nil
|
||||
}
|
||||
|
||||
func (r *Remote) Free() {
|
||||
// free releases the resources of the Remote.
|
||||
func (r *Remote) free() {
|
||||
runtime.SetFinalizer(r, nil)
|
||||
C.git_remote_free(r.ptr)
|
||||
r.ptr = nil
|
||||
r.repo = nil
|
||||
}
|
||||
|
||||
// Free releases the resources of the Remote.
|
||||
func (r *Remote) Free() {
|
||||
r.repo.Remotes.untrackRemote(r)
|
||||
if r.weak {
|
||||
return
|
||||
}
|
||||
r.free()
|
||||
}
|
||||
|
||||
type RemoteCollection struct {
|
||||
doNotCompare
|
||||
repo *Repository
|
||||
|
||||
sync.RWMutex
|
||||
remotes map[*C.git_remote]*Remote
|
||||
}
|
||||
|
||||
func (c *RemoteCollection) trackRemote(r *Remote) {
|
||||
c.Lock()
|
||||
c.remotes[r.ptr] = r
|
||||
c.Unlock()
|
||||
|
||||
remotePointers.track(r)
|
||||
}
|
||||
|
||||
func (c *RemoteCollection) untrackRemote(r *Remote) {
|
||||
c.Lock()
|
||||
delete(c.remotes, r.ptr)
|
||||
c.Unlock()
|
||||
|
||||
remotePointers.untrack(r)
|
||||
}
|
||||
|
||||
func (c *RemoteCollection) List() ([]string, error) {
|
||||
|
@ -404,7 +661,7 @@ func (c *RemoteCollection) List() ([]string, error) {
|
|||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
defer C.git_strarray_free(&r)
|
||||
defer C.git_strarray_dispose(&r)
|
||||
|
||||
remotes := makeStringsFromCStrings(r.strings, int(r.count))
|
||||
return remotes, nil
|
||||
|
@ -425,7 +682,29 @@ func (c *RemoteCollection) Create(name string, url string) (*Remote, error) {
|
|||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
runtime.SetFinalizer(remote, (*Remote).Free)
|
||||
c.trackRemote(remote)
|
||||
return remote, nil
|
||||
}
|
||||
|
||||
// CreateWithOptions Creates a repository object with extended options.
|
||||
func (c *RemoteCollection) CreateWithOptions(url string, option *RemoteCreateOptions) (*Remote, error) {
|
||||
remote := &Remote{repo: c.repo}
|
||||
|
||||
curl := C.CString(url)
|
||||
defer C.free(unsafe.Pointer(curl))
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
copts := populateRemoteCreateOptions(&C.git_remote_create_options{}, option, c.repo)
|
||||
defer freeRemoteCreateOptions(copts)
|
||||
|
||||
ret := C.git_remote_create_with_opts(&remote.ptr, curl, copts)
|
||||
runtime.KeepAlive(c.repo)
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
c.trackRemote(remote)
|
||||
return remote, nil
|
||||
}
|
||||
|
||||
|
@ -461,7 +740,7 @@ func (c *RemoteCollection) CreateWithFetchspec(name string, url string, fetch st
|
|||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
runtime.SetFinalizer(remote, (*Remote).Free)
|
||||
c.trackRemote(remote)
|
||||
return remote, nil
|
||||
}
|
||||
|
||||
|
@ -478,7 +757,7 @@ func (c *RemoteCollection) CreateAnonymous(url string) (*Remote, error) {
|
|||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
runtime.SetFinalizer(remote, (*Remote).Free)
|
||||
c.trackRemote(remote)
|
||||
return remote, nil
|
||||
}
|
||||
|
||||
|
@ -495,10 +774,24 @@ func (c *RemoteCollection) Lookup(name string) (*Remote, error) {
|
|||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
runtime.SetFinalizer(remote, (*Remote).Free)
|
||||
c.trackRemote(remote)
|
||||
return remote, nil
|
||||
}
|
||||
|
||||
func (c *RemoteCollection) Free() {
|
||||
var remotes []*Remote
|
||||
c.Lock()
|
||||
for remotePtr, remote := range c.remotes {
|
||||
remotes = append(remotes, remote)
|
||||
delete(c.remotes, remotePtr)
|
||||
}
|
||||
c.Unlock()
|
||||
|
||||
for _, remote := range remotes {
|
||||
remotePointers.untrack(remote)
|
||||
}
|
||||
}
|
||||
|
||||
func (o *Remote) Name() string {
|
||||
s := C.git_remote_name(o.ptr)
|
||||
runtime.KeepAlive(o)
|
||||
|
@ -638,7 +931,7 @@ func (o *Remote) FetchRefspecs() ([]string, error) {
|
|||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
defer C.git_strarray_free(&crefspecs)
|
||||
defer C.git_strarray_dispose(&crefspecs)
|
||||
|
||||
refspecs := makeStringsFromCStrings(crefspecs.strings, int(crefspecs.count))
|
||||
return refspecs, nil
|
||||
|
@ -671,7 +964,7 @@ func (o *Remote) PushRefspecs() ([]string, error) {
|
|||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
defer C.git_strarray_free(&crefspecs)
|
||||
defer C.git_strarray_dispose(&crefspecs)
|
||||
runtime.KeepAlive(o)
|
||||
|
||||
refspecs := makeStringsFromCStrings(crefspecs.strings, int(crefspecs.count))
|
||||
|
@ -684,35 +977,58 @@ func (o *Remote) RefspecCount() uint {
|
|||
return uint(count)
|
||||
}
|
||||
|
||||
func populateFetchOptions(options *C.git_fetch_options, opts *FetchOptions) {
|
||||
C.git_fetch_options_init(options, C.GIT_FETCH_OPTIONS_VERSION)
|
||||
func populateFetchOptions(copts *C.git_fetch_options, opts *FetchOptions, errorTarget *error) *C.git_fetch_options {
|
||||
C.git_fetch_options_init(copts, C.GIT_FETCH_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return
|
||||
return nil
|
||||
}
|
||||
populateRemoteCallbacks(&options.callbacks, &opts.RemoteCallbacks)
|
||||
options.prune = C.git_fetch_prune_t(opts.Prune)
|
||||
options.update_fetchhead = cbool(opts.UpdateFetchhead)
|
||||
options.download_tags = C.git_remote_autotag_option_t(opts.DownloadTags)
|
||||
populateRemoteCallbacks(&copts.callbacks, &opts.RemoteCallbacks, errorTarget)
|
||||
copts.prune = C.git_fetch_prune_t(opts.Prune)
|
||||
fmt.Println("populateFetchOptions() is broken. fixme!")
|
||||
// fix this line: ./remote.go:988:27: cannot use cbool(opts.UpdateFetchhead) (value of type _Ctype_int) as _Ctype_uint value in assignment
|
||||
// copts.update_fetchhead = cbool(opts.UpdateFetchhead)
|
||||
copts.download_tags = C.git_remote_autotag_option_t(opts.DownloadTags)
|
||||
|
||||
options.custom_headers = C.git_strarray{}
|
||||
options.custom_headers.count = C.size_t(len(opts.Headers))
|
||||
options.custom_headers.strings = makeCStringsFromStrings(opts.Headers)
|
||||
populateProxyOptions(&options.proxy_opts, &opts.ProxyOptions)
|
||||
copts.custom_headers = C.git_strarray{
|
||||
count: C.size_t(len(opts.Headers)),
|
||||
strings: makeCStringsFromStrings(opts.Headers),
|
||||
}
|
||||
populateProxyOptions(&copts.proxy_opts, &opts.ProxyOptions)
|
||||
return copts
|
||||
}
|
||||
|
||||
func populatePushOptions(options *C.git_push_options, opts *PushOptions) {
|
||||
C.git_push_options_init(options, C.GIT_PUSH_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
func freeFetchOptions(copts *C.git_fetch_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
freeStrarray(&copts.custom_headers)
|
||||
untrackCallbacksPayload(&copts.callbacks)
|
||||
freeProxyOptions(&copts.proxy_opts)
|
||||
}
|
||||
|
||||
options.pb_parallelism = C.uint(opts.PbParallelism)
|
||||
func populatePushOptions(copts *C.git_push_options, opts *PushOptions, errorTarget *error) *C.git_push_options {
|
||||
C.git_push_options_init(copts, C.GIT_PUSH_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
options.custom_headers = C.git_strarray{}
|
||||
options.custom_headers.count = C.size_t(len(opts.Headers))
|
||||
options.custom_headers.strings = makeCStringsFromStrings(opts.Headers)
|
||||
copts.pb_parallelism = C.uint(opts.PbParallelism)
|
||||
copts.custom_headers = C.git_strarray{
|
||||
count: C.size_t(len(opts.Headers)),
|
||||
strings: makeCStringsFromStrings(opts.Headers),
|
||||
}
|
||||
populateRemoteCallbacks(&copts.callbacks, &opts.RemoteCallbacks, errorTarget)
|
||||
populateProxyOptions(&copts.proxy_opts, &opts.ProxyOptions)
|
||||
return copts
|
||||
}
|
||||
|
||||
populateRemoteCallbacks(&options.callbacks, &opts.RemoteCallbacks)
|
||||
func freePushOptions(copts *C.git_push_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
untrackCallbacksPayload(&copts.callbacks)
|
||||
freeStrarray(&copts.custom_headers)
|
||||
freeProxyOptions(&copts.proxy_opts)
|
||||
}
|
||||
|
||||
// Fetch performs a fetch operation. refspecs specifies which refspecs
|
||||
|
@ -726,26 +1042,29 @@ func (o *Remote) Fetch(refspecs []string, opts *FetchOptions, msg string) error
|
|||
defer C.free(unsafe.Pointer(cmsg))
|
||||
}
|
||||
|
||||
crefspecs := C.git_strarray{}
|
||||
crefspecs.count = C.size_t(len(refspecs))
|
||||
crefspecs.strings = makeCStringsFromStrings(refspecs)
|
||||
var err error
|
||||
crefspecs := C.git_strarray{
|
||||
count: C.size_t(len(refspecs)),
|
||||
strings: makeCStringsFromStrings(refspecs),
|
||||
}
|
||||
defer freeStrarray(&crefspecs)
|
||||
|
||||
coptions := (*C.git_fetch_options)(C.calloc(1, C.size_t(unsafe.Sizeof(C.git_fetch_options{}))))
|
||||
defer C.free(unsafe.Pointer(coptions))
|
||||
|
||||
populateFetchOptions(coptions, opts)
|
||||
defer untrackCalbacksPayload(&coptions.callbacks)
|
||||
defer freeStrarray(&coptions.custom_headers)
|
||||
coptions := populateFetchOptions(&C.git_fetch_options{}, opts, &err)
|
||||
defer freeFetchOptions(coptions)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_remote_fetch(o.ptr, &crefspecs, coptions, cmsg)
|
||||
runtime.KeepAlive(o)
|
||||
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
@ -765,23 +1084,27 @@ func (o *Remote) ConnectPush(callbacks *RemoteCallbacks, proxyOpts *ProxyOptions
|
|||
//
|
||||
// 'headers' are extra HTTP headers to use in this connection.
|
||||
func (o *Remote) Connect(direction ConnectDirection, callbacks *RemoteCallbacks, proxyOpts *ProxyOptions, headers []string) error {
|
||||
var ccallbacks C.git_remote_callbacks
|
||||
populateRemoteCallbacks(&ccallbacks, callbacks)
|
||||
var err error
|
||||
ccallbacks := populateRemoteCallbacks(&C.git_remote_callbacks{}, callbacks, &err)
|
||||
defer untrackCallbacksPayload(ccallbacks)
|
||||
|
||||
var cproxy C.git_proxy_options
|
||||
populateProxyOptions(&cproxy, proxyOpts)
|
||||
defer freeProxyOptions(&cproxy)
|
||||
cproxy := populateProxyOptions(&C.git_proxy_options{}, proxyOpts)
|
||||
defer freeProxyOptions(cproxy)
|
||||
|
||||
cheaders := C.git_strarray{}
|
||||
cheaders.count = C.size_t(len(headers))
|
||||
cheaders.strings = makeCStringsFromStrings(headers)
|
||||
cheaders := C.git_strarray{
|
||||
count: C.size_t(len(headers)),
|
||||
strings: makeCStringsFromStrings(headers),
|
||||
}
|
||||
defer freeStrarray(&cheaders)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_remote_connect(o.ptr, C.git_direction(direction), &ccallbacks, &cproxy, &cheaders)
|
||||
ret := C.git_remote_connect(o.ptr, C.git_direction(direction), ccallbacks, cproxy, &cheaders)
|
||||
runtime.KeepAlive(o)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret != 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
@ -845,23 +1168,24 @@ func (o *Remote) Ls(filterRefs ...string) ([]RemoteHead, error) {
|
|||
}
|
||||
|
||||
func (o *Remote) Push(refspecs []string, opts *PushOptions) error {
|
||||
crefspecs := C.git_strarray{}
|
||||
crefspecs.count = C.size_t(len(refspecs))
|
||||
crefspecs.strings = makeCStringsFromStrings(refspecs)
|
||||
crefspecs := C.git_strarray{
|
||||
count: C.size_t(len(refspecs)),
|
||||
strings: makeCStringsFromStrings(refspecs),
|
||||
}
|
||||
defer freeStrarray(&crefspecs)
|
||||
|
||||
coptions := (*C.git_push_options)(C.calloc(1, C.size_t(unsafe.Sizeof(C.git_push_options{}))))
|
||||
defer C.free(unsafe.Pointer(coptions))
|
||||
|
||||
populatePushOptions(coptions, opts)
|
||||
defer untrackCalbacksPayload(&coptions.callbacks)
|
||||
defer freeStrarray(&coptions.custom_headers)
|
||||
var err error
|
||||
coptions := populatePushOptions(&C.git_push_options{}, opts, &err)
|
||||
defer freePushOptions(coptions)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_remote_push(o.ptr, &crefspecs, coptions)
|
||||
runtime.KeepAlive(o)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
@ -873,16 +1197,71 @@ func (o *Remote) PruneRefs() bool {
|
|||
}
|
||||
|
||||
func (o *Remote) Prune(callbacks *RemoteCallbacks) error {
|
||||
var ccallbacks C.git_remote_callbacks
|
||||
populateRemoteCallbacks(&ccallbacks, callbacks)
|
||||
var err error
|
||||
ccallbacks := populateRemoteCallbacks(&C.git_remote_callbacks{}, callbacks, &err)
|
||||
defer untrackCallbacksPayload(ccallbacks)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_remote_prune(o.ptr, &ccallbacks)
|
||||
ret := C.git_remote_prune(o.ptr, ccallbacks)
|
||||
runtime.KeepAlive(o)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// DefaultApplyOptions returns default options for remote create
|
||||
func DefaultRemoteCreateOptions() (*RemoteCreateOptions, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
opts := C.git_remote_create_options{}
|
||||
ecode := C.git_remote_create_options_init(&opts, C.GIT_REMOTE_CREATE_OPTIONS_VERSION)
|
||||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
|
||||
return &RemoteCreateOptions{
|
||||
Flags: RemoteCreateOptionsFlag(opts.flags),
|
||||
}, nil
|
||||
}
|
||||
|
||||
func populateRemoteCreateOptions(copts *C.git_remote_create_options, opts *RemoteCreateOptions, repo *Repository) *C.git_remote_create_options {
|
||||
C.git_remote_create_options_init(copts, C.GIT_REMOTE_CREATE_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
var cRepository *C.git_repository
|
||||
if repo != nil {
|
||||
cRepository = repo.ptr
|
||||
}
|
||||
copts.repository = cRepository
|
||||
copts.name = C.CString(opts.Name)
|
||||
copts.fetchspec = C.CString(opts.FetchSpec)
|
||||
copts.flags = C.uint(opts.Flags)
|
||||
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeRemoteCreateOptions(ptr *C.git_remote_create_options) {
|
||||
if ptr == nil {
|
||||
return
|
||||
}
|
||||
C.free(unsafe.Pointer(ptr.name))
|
||||
C.free(unsafe.Pointer(ptr.fetchspec))
|
||||
}
|
||||
|
||||
// createNewEmptyRemote used to get a new empty object of *Remote
|
||||
func createNewEmptyRemote() *Remote {
|
||||
return &Remote{
|
||||
callbacks: RemoteCallbacks{},
|
||||
repo: nil,
|
||||
weak: false,
|
||||
}
|
||||
}
|
||||
|
|
357
remote_test.go
357
remote_test.go
|
@ -1,8 +1,22 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"crypto/rand"
|
||||
"crypto/rsa"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"net"
|
||||
"os"
|
||||
"os/exec"
|
||||
"strings"
|
||||
"sync"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/google/shlex"
|
||||
"golang.org/x/crypto/ssh"
|
||||
)
|
||||
|
||||
func TestListRemotes(t *testing.T) {
|
||||
|
@ -10,9 +24,9 @@ func TestListRemotes(t *testing.T) {
|
|||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
_, err := repo.Remotes.Create("test", "git://foo/bar")
|
||||
|
||||
remote, err := repo.Remotes.Create("test", "git://foo/bar")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
expected := []string{
|
||||
"test",
|
||||
|
@ -24,13 +38,13 @@ func TestListRemotes(t *testing.T) {
|
|||
compareStringList(t, expected, actual)
|
||||
}
|
||||
|
||||
func assertHostname(cert *Certificate, valid bool, hostname string, t *testing.T) ErrorCode {
|
||||
func assertHostname(cert *Certificate, valid bool, hostname string, t *testing.T) error {
|
||||
if hostname != "github.com" {
|
||||
t.Fatal("Hostname does not match")
|
||||
return ErrUser
|
||||
t.Fatal("hostname does not match")
|
||||
return errors.New("hostname does not match")
|
||||
}
|
||||
|
||||
return 0
|
||||
return nil
|
||||
}
|
||||
|
||||
func TestCertificateCheck(t *testing.T) {
|
||||
|
@ -40,10 +54,11 @@ func TestCertificateCheck(t *testing.T) {
|
|||
|
||||
remote, err := repo.Remotes.Create("origin", "https://github.com/libgit2/TestGitRepository")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
options := FetchOptions{
|
||||
RemoteCallbacks: RemoteCallbacks{
|
||||
CertificateCheckCallback: func(cert *Certificate, valid bool, hostname string) ErrorCode {
|
||||
CertificateCheckCallback: func(cert *Certificate, valid bool, hostname string) error {
|
||||
return assertHostname(cert, valid, hostname, t)
|
||||
},
|
||||
},
|
||||
|
@ -60,6 +75,30 @@ func TestRemoteConnect(t *testing.T) {
|
|||
|
||||
remote, err := repo.Remotes.Create("origin", "https://github.com/libgit2/TestGitRepository")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.ConnectFetch(nil, nil, nil)
|
||||
checkFatal(t, err)
|
||||
}
|
||||
|
||||
func TestRemoteConnectOption(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
config, err := repo.Config()
|
||||
checkFatal(t, err)
|
||||
err = config.SetString("url.git@github.com:.insteadof", "https://github.com/")
|
||||
checkFatal(t, err)
|
||||
|
||||
option, err := DefaultRemoteCreateOptions()
|
||||
checkFatal(t, err)
|
||||
option.Name = "origin"
|
||||
option.Flags = RemoteCreateSkipInsteadof
|
||||
|
||||
remote, err := repo.Remotes.CreateWithOptions("https://github.com/libgit2/TestGitRepository", option)
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.ConnectFetch(nil, nil, nil)
|
||||
checkFatal(t, err)
|
||||
|
@ -72,6 +111,7 @@ func TestRemoteLs(t *testing.T) {
|
|||
|
||||
remote, err := repo.Remotes.Create("origin", "https://github.com/libgit2/TestGitRepository")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.ConnectFetch(nil, nil, nil)
|
||||
checkFatal(t, err)
|
||||
|
@ -91,6 +131,7 @@ func TestRemoteLsFiltering(t *testing.T) {
|
|||
|
||||
remote, err := repo.Remotes.Create("origin", "https://github.com/libgit2/TestGitRepository")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.ConnectFetch(nil, nil, nil)
|
||||
checkFatal(t, err)
|
||||
|
@ -123,11 +164,13 @@ func TestRemotePruneRefs(t *testing.T) {
|
|||
err = config.SetBool("remote.origin.prune", true)
|
||||
checkFatal(t, err)
|
||||
|
||||
_, err = repo.Remotes.Create("origin", "https://github.com/libgit2/TestGitRepository")
|
||||
remote, err := repo.Remotes.Create("origin", "https://github.com/libgit2/TestGitRepository")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
remote, err := repo.Remotes.Lookup("origin")
|
||||
remote, err = repo.Remotes.Lookup("origin")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
if !remote.PruneRefs() {
|
||||
t.Fatal("Expected remote to be configured to prune references")
|
||||
|
@ -146,6 +189,7 @@ func TestRemotePrune(t *testing.T) {
|
|||
|
||||
remoteRef, err := remoteRepo.CreateBranch("test-prune", commit, true)
|
||||
checkFatal(t, err)
|
||||
defer remoteRef.Free()
|
||||
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
@ -157,12 +201,14 @@ func TestRemotePrune(t *testing.T) {
|
|||
remoteUrl := fmt.Sprintf("file://%s", remoteRepo.Workdir())
|
||||
remote, err := repo.Remotes.Create("origin", remoteUrl)
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.Fetch([]string{"test-prune"}, nil, "")
|
||||
checkFatal(t, err)
|
||||
|
||||
_, err = repo.References.Create("refs/remotes/origin/test-prune", head, true, "remote reference")
|
||||
ref, err := repo.References.Create("refs/remotes/origin/test-prune", head, true, "remote reference")
|
||||
checkFatal(t, err)
|
||||
defer ref.Free()
|
||||
|
||||
err = remoteRef.Delete()
|
||||
checkFatal(t, err)
|
||||
|
@ -172,6 +218,7 @@ func TestRemotePrune(t *testing.T) {
|
|||
|
||||
rr, err := repo.Remotes.Lookup("origin")
|
||||
checkFatal(t, err)
|
||||
defer rr.Free()
|
||||
|
||||
err = rr.ConnectFetch(nil, nil, nil)
|
||||
checkFatal(t, err)
|
||||
|
@ -179,8 +226,296 @@ func TestRemotePrune(t *testing.T) {
|
|||
err = rr.Prune(nil)
|
||||
checkFatal(t, err)
|
||||
|
||||
_, err = repo.References.Lookup("refs/remotes/origin/test-prune")
|
||||
ref, err = repo.References.Lookup("refs/remotes/origin/test-prune")
|
||||
if err == nil {
|
||||
ref.Free()
|
||||
t.Fatal("Expected error getting a pruned reference")
|
||||
}
|
||||
}
|
||||
|
||||
func TestRemoteCredentialsCalled(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
remote, err := repo.Remotes.CreateAnonymous("https://github.com/libgit2/non-existent")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
errNonExistent := errors.New("non-existent repository")
|
||||
fetchOpts := FetchOptions{
|
||||
RemoteCallbacks: RemoteCallbacks{
|
||||
CredentialsCallback: func(url, username string, allowedTypes CredentialType) (*Credential, error) {
|
||||
return nil, errNonExistent
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
err = remote.Fetch(nil, &fetchOpts, "fetch")
|
||||
if err != errNonExistent {
|
||||
t.Fatalf("remote.Fetch() = %v, want %v", err, errNonExistent)
|
||||
}
|
||||
}
|
||||
|
||||
func newChannelPipe(t *testing.T, w io.Writer, wg *sync.WaitGroup) (*os.File, error) {
|
||||
pr, pw, err := os.Pipe()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
wg.Add(1)
|
||||
go func() {
|
||||
_, err := io.Copy(w, pr)
|
||||
if err != nil && err != io.EOF {
|
||||
t.Logf("Failed to copy: %v", err)
|
||||
}
|
||||
wg.Done()
|
||||
}()
|
||||
|
||||
return pw, nil
|
||||
}
|
||||
|
||||
func startSSHServer(t *testing.T, hostKey ssh.Signer, authorizedKeys []ssh.PublicKey) net.Listener {
|
||||
t.Helper()
|
||||
|
||||
marshaledAuthorizedKeys := make([][]byte, len(authorizedKeys))
|
||||
for i, authorizedKey := range authorizedKeys {
|
||||
marshaledAuthorizedKeys[i] = authorizedKey.Marshal()
|
||||
}
|
||||
|
||||
config := &ssh.ServerConfig{
|
||||
PublicKeyCallback: func(c ssh.ConnMetadata, pubKey ssh.PublicKey) (*ssh.Permissions, error) {
|
||||
marshaledPubKey := pubKey.Marshal()
|
||||
for _, marshaledAuthorizedKey := range marshaledAuthorizedKeys {
|
||||
if bytes.Equal(marshaledPubKey, marshaledAuthorizedKey) {
|
||||
return &ssh.Permissions{
|
||||
// Record the public key used for authentication.
|
||||
Extensions: map[string]string{
|
||||
"pubkey-fp": ssh.FingerprintSHA256(pubKey),
|
||||
},
|
||||
}, nil
|
||||
}
|
||||
}
|
||||
t.Logf("unknown public key for %q:\n\t%+v\n\t%+v\n", c.User(), pubKey.Marshal(), authorizedKeys)
|
||||
return nil, fmt.Errorf("unknown public key for %q", c.User())
|
||||
},
|
||||
}
|
||||
config.AddHostKey(hostKey)
|
||||
|
||||
listener, err := net.Listen("tcp", "localhost:0")
|
||||
if err != nil {
|
||||
t.Fatalf("Failed to listen for connection: %v", err)
|
||||
}
|
||||
|
||||
go func() {
|
||||
nConn, err := listener.Accept()
|
||||
if err != nil {
|
||||
if strings.Contains(err.Error(), "use of closed network connection") {
|
||||
return
|
||||
}
|
||||
t.Logf("Failed to accept incoming connection: %v", err)
|
||||
return
|
||||
}
|
||||
defer nConn.Close()
|
||||
|
||||
conn, chans, reqs, err := ssh.NewServerConn(nConn, config)
|
||||
if err != nil {
|
||||
t.Logf("failed to handshake: %+v, %+v", conn, err)
|
||||
return
|
||||
}
|
||||
|
||||
// The incoming Request channel must be serviced.
|
||||
go func() {
|
||||
for newRequest := range reqs {
|
||||
t.Logf("new request %v", newRequest)
|
||||
}
|
||||
}()
|
||||
|
||||
// Service only the first channel request
|
||||
newChannel := <-chans
|
||||
defer func() {
|
||||
for newChannel := range chans {
|
||||
t.Logf("new channel %v", newChannel)
|
||||
newChannel.Reject(ssh.UnknownChannelType, "server closing")
|
||||
}
|
||||
}()
|
||||
|
||||
// Channels have a type, depending on the application level
|
||||
// protocol intended. In the case of a shell, the type is
|
||||
// "session" and ServerShell may be used to present a simple
|
||||
// terminal interface.
|
||||
if newChannel.ChannelType() != "session" {
|
||||
newChannel.Reject(ssh.UnknownChannelType, "unknown channel type")
|
||||
return
|
||||
}
|
||||
channel, requests, err := newChannel.Accept()
|
||||
if err != nil {
|
||||
t.Logf("Could not accept channel: %v", err)
|
||||
return
|
||||
}
|
||||
defer channel.Close()
|
||||
|
||||
// Sessions have out-of-band requests such as "shell",
|
||||
// "pty-req" and "env". Here we handle only the
|
||||
// "exec" request.
|
||||
req := <-requests
|
||||
if req.Type != "exec" {
|
||||
req.Reply(false, nil)
|
||||
return
|
||||
}
|
||||
// RFC 4254 Section 6.5.
|
||||
var payload struct {
|
||||
Command string
|
||||
}
|
||||
if err := ssh.Unmarshal(req.Payload, &payload); err != nil {
|
||||
t.Logf("invalid payload on channel %v: %v", channel, err)
|
||||
req.Reply(false, nil)
|
||||
return
|
||||
}
|
||||
args, err := shlex.Split(payload.Command)
|
||||
if err != nil {
|
||||
t.Logf("invalid command on channel %v: %v", channel, err)
|
||||
req.Reply(false, nil)
|
||||
return
|
||||
}
|
||||
if len(args) < 2 || (args[0] != "git-upload-pack" && args[0] != "git-receive-pack") {
|
||||
t.Logf("invalid command (%v) on channel %v: %v", args, channel, err)
|
||||
req.Reply(false, nil)
|
||||
return
|
||||
}
|
||||
req.Reply(true, nil)
|
||||
|
||||
go func(in <-chan *ssh.Request) {
|
||||
for req := range in {
|
||||
t.Logf("draining request %v", req)
|
||||
}
|
||||
}(requests)
|
||||
|
||||
// The first parameter is the (absolute) path of the repository.
|
||||
args[1] = "./testdata" + args[1]
|
||||
|
||||
cmd := exec.Command(args[0], args[1:]...)
|
||||
cmd.Stdin = channel
|
||||
var wg sync.WaitGroup
|
||||
stdoutPipe, err := newChannelPipe(t, channel, &wg)
|
||||
if err != nil {
|
||||
t.Logf("Failed to create stdout pipe: %v", err)
|
||||
return
|
||||
}
|
||||
cmd.Stdout = stdoutPipe
|
||||
stderrPipe, err := newChannelPipe(t, channel.Stderr(), &wg)
|
||||
if err != nil {
|
||||
t.Logf("Failed to create stderr pipe: %v", err)
|
||||
return
|
||||
}
|
||||
cmd.Stderr = stderrPipe
|
||||
|
||||
go func() {
|
||||
wg.Wait()
|
||||
channel.CloseWrite()
|
||||
}()
|
||||
|
||||
err = cmd.Start()
|
||||
if err != nil {
|
||||
t.Logf("Failed to start %v: %v", args, err)
|
||||
return
|
||||
}
|
||||
|
||||
// Once the process has started, we need to close the write end of the
|
||||
// pipes from this process so that we can know when the child has done
|
||||
// writing to it.
|
||||
stdoutPipe.Close()
|
||||
stderrPipe.Close()
|
||||
|
||||
timer := time.AfterFunc(5*time.Second, func() {
|
||||
t.Log("process timed out, terminating")
|
||||
cmd.Process.Kill()
|
||||
})
|
||||
defer timer.Stop()
|
||||
|
||||
err = cmd.Wait()
|
||||
if err != nil {
|
||||
t.Logf("Failed to run %v: %v", args, err)
|
||||
return
|
||||
}
|
||||
}()
|
||||
return listener
|
||||
}
|
||||
|
||||
func TestRemoteSSH(t *testing.T) {
|
||||
t.Parallel()
|
||||
pubKeyUsername := "testuser"
|
||||
|
||||
hostPrivKey, err := rsa.GenerateKey(rand.Reader, 512)
|
||||
if err != nil {
|
||||
t.Fatalf("Failed to generate the host RSA private key: %v", err)
|
||||
}
|
||||
hostSigner, err := ssh.NewSignerFromKey(hostPrivKey)
|
||||
if err != nil {
|
||||
t.Fatalf("Failed to generate SSH hostSigner: %v", err)
|
||||
}
|
||||
|
||||
privKey, err := rsa.GenerateKey(rand.Reader, 512)
|
||||
if err != nil {
|
||||
t.Fatalf("Failed to generate the user RSA private key: %v", err)
|
||||
}
|
||||
signer, err := ssh.NewSignerFromKey(privKey)
|
||||
if err != nil {
|
||||
t.Fatalf("Failed to generate SSH signer: %v", err)
|
||||
}
|
||||
// This is in the format "xx:xx:xx:...", so we remove the colons so that it
|
||||
// matches the fmt.Sprintf() below.
|
||||
// Note that not all libssh2 implementations support the SHA256 fingerprint,
|
||||
// so we use MD5 here for testing.
|
||||
publicKeyFingerprint := strings.Replace(ssh.FingerprintLegacyMD5(hostSigner.PublicKey()), ":", "", -1)
|
||||
|
||||
listener := startSSHServer(t, hostSigner, []ssh.PublicKey{signer.PublicKey()})
|
||||
defer listener.Close()
|
||||
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
certificateCheckCallbackCalled := false
|
||||
fetchOpts := FetchOptions{
|
||||
RemoteCallbacks: RemoteCallbacks{
|
||||
CertificateCheckCallback: func(cert *Certificate, valid bool, hostname string) error {
|
||||
hostkeyFingerprint := fmt.Sprintf("%x", cert.Hostkey.HashMD5[:])
|
||||
if hostkeyFingerprint != publicKeyFingerprint {
|
||||
return fmt.Errorf("server hostkey %q, want %q", hostkeyFingerprint, publicKeyFingerprint)
|
||||
}
|
||||
certificateCheckCallbackCalled = true
|
||||
return nil
|
||||
},
|
||||
CredentialsCallback: func(url, username string, allowedTypes CredentialType) (*Credential, error) {
|
||||
if allowedTypes&(CredentialTypeSSHKey|CredentialTypeSSHCustom|CredentialTypeSSHMemory) != 0 {
|
||||
return NewCredentialSSHKeyFromSigner(pubKeyUsername, signer)
|
||||
}
|
||||
if (allowedTypes & CredentialTypeUsername) != 0 {
|
||||
return NewCredentialUsername(pubKeyUsername)
|
||||
}
|
||||
return nil, fmt.Errorf("unknown credential type %+v", allowedTypes)
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
remote, err := repo.Remotes.Create(
|
||||
"origin",
|
||||
fmt.Sprintf("ssh://%s/TestGitRepository", listener.Addr().String()),
|
||||
)
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.Fetch(nil, &fetchOpts, "")
|
||||
checkFatal(t, err)
|
||||
if !certificateCheckCallbackCalled {
|
||||
t.Fatalf("CertificateCheckCallback was not called")
|
||||
}
|
||||
|
||||
heads, err := remote.Ls()
|
||||
checkFatal(t, err)
|
||||
|
||||
if len(heads) == 0 {
|
||||
t.Error("Expected remote heads")
|
||||
}
|
||||
}
|
||||
|
|
197
repository.go
197
repository.go
|
@ -14,6 +14,7 @@ import (
|
|||
|
||||
// Repository
|
||||
type Repository struct {
|
||||
doNotCompare
|
||||
ptr *C.git_repository
|
||||
// Remotes represents the collection of remotes and can be
|
||||
// used to add, remove and configure remotes for this
|
||||
|
@ -35,12 +36,17 @@ type Repository struct {
|
|||
// Stashes represents the collection of stashes and can be used to
|
||||
// save, apply and iterate over stash states in this repository.
|
||||
Stashes StashCollection
|
||||
|
||||
// weak indicates that a repository is a weak pointer and should not be
|
||||
// freed.
|
||||
weak bool
|
||||
}
|
||||
|
||||
func newRepositoryFromC(ptr *C.git_repository) *Repository {
|
||||
repo := &Repository{ptr: ptr}
|
||||
|
||||
repo.Remotes.repo = repo
|
||||
repo.Remotes.remotes = make(map[*C.git_remote]*Remote)
|
||||
repo.Submodules.repo = repo
|
||||
repo.References.repo = repo
|
||||
repo.Notes.repo = repo
|
||||
|
@ -136,8 +142,14 @@ func (v *Repository) SetRefdb(refdb *Refdb) {
|
|||
}
|
||||
|
||||
func (v *Repository) Free() {
|
||||
ptr := v.ptr
|
||||
v.ptr = nil
|
||||
runtime.SetFinalizer(v, nil)
|
||||
C.git_repository_free(v.ptr)
|
||||
v.Remotes.Free()
|
||||
if v.weak {
|
||||
return
|
||||
}
|
||||
C.git_repository_free(ptr)
|
||||
}
|
||||
|
||||
func (v *Repository) Config() (*Config, error) {
|
||||
|
@ -156,6 +168,23 @@ func (v *Repository) Config() (*Config, error) {
|
|||
return config, nil
|
||||
}
|
||||
|
||||
// SetConfig sets the configuration file for this repository.
|
||||
//
|
||||
// This configuration file will be used for all configuration queries involving
|
||||
// this repository.
|
||||
func (v *Repository) SetConfig(c *Config) error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_repository_set_config(v.ptr, c.ptr)
|
||||
runtime.KeepAlive(v)
|
||||
runtime.KeepAlive(c)
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (v *Repository) Index() (*Index, error) {
|
||||
var ptr *C.git_index
|
||||
|
||||
|
@ -459,6 +488,102 @@ func (v *Repository) CreateCommit(
|
|||
return oid, nil
|
||||
}
|
||||
|
||||
// CreateCommitWithSignature creates a commit object from the given contents and
|
||||
// signature.
|
||||
func (v *Repository) CreateCommitWithSignature(
|
||||
commitContent, signature, signatureField string,
|
||||
) (*Oid, error) {
|
||||
cCommitContent := C.CString(commitContent)
|
||||
defer C.free(unsafe.Pointer(cCommitContent))
|
||||
var cSignature *C.char
|
||||
if signature != "" {
|
||||
cSignature = C.CString(string(signature))
|
||||
defer C.free(unsafe.Pointer(cSignature))
|
||||
}
|
||||
var cSignatureField *C.char
|
||||
if signatureField != "" {
|
||||
cSignatureField = C.CString(string(signatureField))
|
||||
defer C.free(unsafe.Pointer(cSignatureField))
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
oid := new(Oid)
|
||||
ret := C.git_commit_create_with_signature(oid.toC(), v.ptr, cCommitContent, cSignature, cSignatureField)
|
||||
|
||||
runtime.KeepAlive(v)
|
||||
runtime.KeepAlive(oid)
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return oid, nil
|
||||
}
|
||||
|
||||
// CreateCommitBuffer creates a commit and write it into a buffer.
|
||||
func (v *Repository) CreateCommitBuffer(
|
||||
author, committer *Signature,
|
||||
messageEncoding MessageEncoding,
|
||||
message string,
|
||||
tree *Tree,
|
||||
parents ...*Commit,
|
||||
) ([]byte, error) {
|
||||
cmsg := C.CString(message)
|
||||
defer C.free(unsafe.Pointer(cmsg))
|
||||
var cencoding *C.char
|
||||
// Since the UTF-8 encoding is the default, pass in nil whenever UTF-8 is
|
||||
// provided. That will cause the commit to not have an explicit header for
|
||||
// it.
|
||||
if messageEncoding != MessageEncodingUTF8 && messageEncoding != MessageEncoding("") {
|
||||
cencoding = C.CString(string(messageEncoding))
|
||||
defer C.free(unsafe.Pointer(cencoding))
|
||||
}
|
||||
|
||||
var cparents []*C.git_commit = nil
|
||||
var parentsarg **C.git_commit = nil
|
||||
|
||||
nparents := len(parents)
|
||||
if nparents > 0 {
|
||||
cparents = make([]*C.git_commit, nparents)
|
||||
for i, v := range parents {
|
||||
cparents[i] = v.cast_ptr
|
||||
}
|
||||
parentsarg = &cparents[0]
|
||||
}
|
||||
|
||||
authorSig, err := author.toC()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer C.git_signature_free(authorSig)
|
||||
|
||||
committerSig, err := committer.toC()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer C.git_signature_free(committerSig)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var buf C.git_buf
|
||||
defer C.git_buf_dispose(&buf)
|
||||
ret := C.git_commit_create_buffer(
|
||||
&buf, v.ptr,
|
||||
authorSig, committerSig,
|
||||
cencoding, cmsg, tree.cast_ptr, C.size_t(nparents), parentsarg)
|
||||
|
||||
runtime.KeepAlive(v)
|
||||
runtime.KeepAlive(buf)
|
||||
runtime.KeepAlive(parents)
|
||||
if ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return C.GoBytes(unsafe.Pointer(buf.ptr), C.int(buf.size)), nil
|
||||
}
|
||||
|
||||
func (v *Repository) CreateCommitFromIds(
|
||||
refname string, author, committer *Signature,
|
||||
message string, tree *Oid, parents ...*Oid) (*Oid, error) {
|
||||
|
@ -679,3 +804,73 @@ func (r *Repository) ClearGitIgnoreRules() error {
|
|||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Message retrieves git's prepared message.
|
||||
// Operations such as git revert/cherry-pick/merge with the -n option stop just
|
||||
// short of creating a commit with the changes and save their prepared message
|
||||
// in .git/MERGE_MSG so the next git-commit execution can present it to the
|
||||
// user for them to amend if they wish.
|
||||
//
|
||||
// Use this function to get the contents of this file. Don't forget to remove
|
||||
// the file after you create the commit.
|
||||
func (r *Repository) Message() (string, error) {
|
||||
buf := C.git_buf{}
|
||||
defer C.git_buf_dispose(&buf)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cErr := C.git_repository_message(&buf, r.ptr)
|
||||
runtime.KeepAlive(r)
|
||||
if cErr < 0 {
|
||||
return "", MakeGitError(cErr)
|
||||
}
|
||||
return C.GoString(buf.ptr), nil
|
||||
}
|
||||
|
||||
// RemoveMessage removes git's prepared message.
|
||||
func (r *Repository) RemoveMessage() error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cErr := C.git_repository_message_remove(r.ptr)
|
||||
runtime.KeepAlive(r)
|
||||
if cErr < 0 {
|
||||
return MakeGitError(cErr)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
type RepositoryItem int
|
||||
|
||||
const (
|
||||
RepositoryItemGitDir RepositoryItem = C.GIT_REPOSITORY_ITEM_GITDIR
|
||||
RepositoryItemWorkDir RepositoryItem = C.GIT_REPOSITORY_ITEM_WORKDIR
|
||||
RepositoryItemCommonDir RepositoryItem = C.GIT_REPOSITORY_ITEM_COMMONDIR
|
||||
RepositoryItemIndex RepositoryItem = C.GIT_REPOSITORY_ITEM_INDEX
|
||||
RepositoryItemObjects RepositoryItem = C.GIT_REPOSITORY_ITEM_OBJECTS
|
||||
RepositoryItemRefs RepositoryItem = C.GIT_REPOSITORY_ITEM_REFS
|
||||
RepositoryItemPackedRefs RepositoryItem = C.GIT_REPOSITORY_ITEM_PACKED_REFS
|
||||
RepositoryItemRemotes RepositoryItem = C.GIT_REPOSITORY_ITEM_REMOTES
|
||||
RepositoryItemConfig RepositoryItem = C.GIT_REPOSITORY_ITEM_CONFIG
|
||||
RepositoryItemInfo RepositoryItem = C.GIT_REPOSITORY_ITEM_INFO
|
||||
RepositoryItemHooks RepositoryItem = C.GIT_REPOSITORY_ITEM_HOOKS
|
||||
RepositoryItemLogs RepositoryItem = C.GIT_REPOSITORY_ITEM_LOGS
|
||||
RepositoryItemModules RepositoryItem = C.GIT_REPOSITORY_ITEM_MODULES
|
||||
RepositoryItemWorkTrees RepositoryItem = C.GIT_REPOSITORY_ITEM_WORKTREES
|
||||
)
|
||||
|
||||
func (r *Repository) ItemPath(item RepositoryItem) (string, error) {
|
||||
var c_buf C.git_buf
|
||||
defer C.git_buf_dispose(&c_buf)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_repository_item_path(&c_buf, r.ptr, C.git_repository_item_t(item))
|
||||
runtime.KeepAlive(r)
|
||||
if ret < 0 {
|
||||
return "", MakeGitError(ret)
|
||||
}
|
||||
return C.GoString(c_buf.ptr), nil
|
||||
}
|
||||
|
|
|
@ -5,6 +5,60 @@ import (
|
|||
"time"
|
||||
)
|
||||
|
||||
func TestCreateCommitBuffer(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
loc, err := time.LoadLocation("Europe/Berlin")
|
||||
checkFatal(t, err)
|
||||
sig := &Signature{
|
||||
Name: "Rand Om Hacker",
|
||||
Email: "random@hacker.com",
|
||||
When: time.Date(2013, 03, 06, 14, 30, 0, 0, loc),
|
||||
}
|
||||
|
||||
idx, err := repo.Index()
|
||||
checkFatal(t, err)
|
||||
err = idx.AddByPath("README")
|
||||
checkFatal(t, err)
|
||||
err = idx.Write()
|
||||
checkFatal(t, err)
|
||||
treeId, err := idx.WriteTree()
|
||||
checkFatal(t, err)
|
||||
|
||||
message := "This is a commit\n"
|
||||
tree, err := repo.LookupTree(treeId)
|
||||
checkFatal(t, err)
|
||||
|
||||
for encoding, expected := range map[MessageEncoding]string{
|
||||
MessageEncodingUTF8: `tree b7119b11e8ef7a1a5a34d3ac87f5b075228ac81e
|
||||
author Rand Om Hacker <random@hacker.com> 1362576600 +0100
|
||||
committer Rand Om Hacker <random@hacker.com> 1362576600 +0100
|
||||
|
||||
This is a commit
|
||||
`,
|
||||
MessageEncoding("ASCII"): `tree b7119b11e8ef7a1a5a34d3ac87f5b075228ac81e
|
||||
author Rand Om Hacker <random@hacker.com> 1362576600 +0100
|
||||
committer Rand Om Hacker <random@hacker.com> 1362576600 +0100
|
||||
encoding ASCII
|
||||
|
||||
This is a commit
|
||||
`,
|
||||
} {
|
||||
encoding := encoding
|
||||
expected := expected
|
||||
t.Run(string(encoding), func(t *testing.T) {
|
||||
buf, err := repo.CreateCommitBuffer(sig, sig, encoding, message, tree)
|
||||
checkFatal(t, err)
|
||||
|
||||
if expected != string(buf) {
|
||||
t.Errorf("mismatched commit buffer, expected %v, got %v", expected, string(buf))
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestCreateCommitFromIds(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
|
@ -40,3 +94,66 @@ func TestCreateCommitFromIds(t *testing.T) {
|
|||
t.Errorf("mismatched commit ids, expected %v, got %v", expectedCommitId.String(), commitId.String())
|
||||
}
|
||||
}
|
||||
|
||||
func TestRepositorySetConfig(t *testing.T) {
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
loc, err := time.LoadLocation("Europe/Berlin")
|
||||
checkFatal(t, err)
|
||||
sig := &Signature{
|
||||
Name: "Rand Om Hacker",
|
||||
Email: "random@hacker.com",
|
||||
When: time.Date(2013, 03, 06, 14, 30, 0, 0, loc),
|
||||
}
|
||||
|
||||
idx, err := repo.Index()
|
||||
checkFatal(t, err)
|
||||
err = idx.AddByPath("README")
|
||||
|
||||
treeId, err := idx.WriteTree()
|
||||
checkFatal(t, err)
|
||||
|
||||
message := "This is a commit\n"
|
||||
tree, err := repo.LookupTree(treeId)
|
||||
checkFatal(t, err)
|
||||
_, err = repo.CreateCommit("HEAD", sig, sig, message, tree)
|
||||
checkFatal(t, err)
|
||||
|
||||
repoConfig, err := repo.Config()
|
||||
checkFatal(t, err)
|
||||
|
||||
temp := Config{}
|
||||
localConfig, err := temp.OpenLevel(repoConfig, ConfigLevelLocal)
|
||||
checkFatal(t, err)
|
||||
repoConfig = nil
|
||||
|
||||
err = repo.SetConfig(localConfig)
|
||||
checkFatal(t, err)
|
||||
|
||||
configFieldName := "core.filemode"
|
||||
err = localConfig.SetBool(configFieldName, true)
|
||||
checkFatal(t, err)
|
||||
|
||||
localConfig = nil
|
||||
|
||||
repoConfig, err = repo.Config()
|
||||
checkFatal(t, err)
|
||||
|
||||
result, err := repoConfig.LookupBool(configFieldName)
|
||||
checkFatal(t, err)
|
||||
if result != true {
|
||||
t.Fatal("result must be true")
|
||||
}
|
||||
}
|
||||
|
||||
func TestRepositoryItemPath(t *testing.T) {
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
gitDir, err := repo.ItemPath(RepositoryItemGitDir)
|
||||
checkFatal(t, err)
|
||||
if gitDir == "" {
|
||||
t.Error("expected not empty gitDir")
|
||||
}
|
||||
}
|
||||
|
|
11
reset.go
11
reset.go
|
@ -14,11 +14,18 @@ const (
|
|||
ResetHard ResetType = C.GIT_RESET_HARD
|
||||
)
|
||||
|
||||
func (r *Repository) ResetToCommit(commit *Commit, resetType ResetType, opts *CheckoutOpts) error {
|
||||
func (r *Repository) ResetToCommit(commit *Commit, resetType ResetType, opts *CheckoutOptions) error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
ret := C.git_reset(r.ptr, commit.ptr, C.git_reset_t(resetType), opts.toC())
|
||||
|
||||
var err error
|
||||
cOpts := populateCheckoutOptions(&C.git_checkout_options{}, opts, &err)
|
||||
defer freeCheckoutOptions(cOpts)
|
||||
|
||||
ret := C.git_reset(r.ptr, commit.ptr, C.git_reset_t(resetType), cOpts)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
|
|
@ -37,7 +37,7 @@ func TestResetToCommit(t *testing.T) {
|
|||
commitToResetTo, err := repo.LookupCommit(commitId)
|
||||
checkFatal(t, err)
|
||||
|
||||
repo.ResetToCommit(commitToResetTo, ResetHard, &CheckoutOpts{})
|
||||
repo.ResetToCommit(commitToResetTo, ResetHard, &CheckoutOptions{})
|
||||
|
||||
// check that the file now reads "testing reset" like it did before
|
||||
bytes, err := ioutil.ReadFile(pathInRepo(repo, "README"))
|
||||
|
|
69
revert.go
69
revert.go
|
@ -10,46 +10,52 @@ import (
|
|||
|
||||
// RevertOptions contains options for performing a revert
|
||||
type RevertOptions struct {
|
||||
Mainline uint
|
||||
MergeOpts MergeOptions
|
||||
CheckoutOpts CheckoutOpts
|
||||
Mainline uint
|
||||
MergeOptions MergeOptions
|
||||
CheckoutOptions CheckoutOptions
|
||||
}
|
||||
|
||||
func (opts *RevertOptions) toC() *C.git_revert_options {
|
||||
return &C.git_revert_options{
|
||||
version: C.GIT_REVERT_OPTIONS_VERSION,
|
||||
mainline: C.uint(opts.Mainline),
|
||||
merge_opts: *opts.MergeOpts.toC(),
|
||||
checkout_opts: *opts.CheckoutOpts.toC(),
|
||||
func populateRevertOptions(copts *C.git_revert_options, opts *RevertOptions, errorTarget *error) *C.git_revert_options {
|
||||
C.git_revert_options_init(copts, C.GIT_REVERT_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
copts.mainline = C.uint(opts.Mainline)
|
||||
populateMergeOptions(&copts.merge_opts, &opts.MergeOptions)
|
||||
populateCheckoutOptions(&copts.checkout_opts, &opts.CheckoutOptions, errorTarget)
|
||||
return copts
|
||||
}
|
||||
|
||||
func revertOptionsFromC(opts *C.git_revert_options) RevertOptions {
|
||||
func revertOptionsFromC(copts *C.git_revert_options) RevertOptions {
|
||||
return RevertOptions{
|
||||
Mainline: uint(opts.mainline),
|
||||
MergeOpts: mergeOptionsFromC(&opts.merge_opts),
|
||||
CheckoutOpts: checkoutOptionsFromC(&opts.checkout_opts),
|
||||
Mainline: uint(copts.mainline),
|
||||
MergeOptions: mergeOptionsFromC(&copts.merge_opts),
|
||||
CheckoutOptions: checkoutOptionsFromC(&copts.checkout_opts),
|
||||
}
|
||||
}
|
||||
|
||||
func freeRevertOptions(opts *C.git_revert_options) {
|
||||
freeCheckoutOpts(&opts.checkout_opts)
|
||||
func freeRevertOptions(copts *C.git_revert_options) {
|
||||
if copts != nil {
|
||||
return
|
||||
}
|
||||
freeMergeOptions(&copts.merge_opts)
|
||||
freeCheckoutOptions(&copts.checkout_opts)
|
||||
}
|
||||
|
||||
// DefaultRevertOptions initialises a RevertOptions struct with default values
|
||||
func DefaultRevertOptions() (RevertOptions, error) {
|
||||
opts := C.git_revert_options{}
|
||||
copts := C.git_revert_options{}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ecode := C.git_revert_options_init(&opts, C.GIT_REVERT_OPTIONS_VERSION)
|
||||
ecode := C.git_revert_options_init(&copts, C.GIT_REVERT_OPTIONS_VERSION)
|
||||
if ecode < 0 {
|
||||
return RevertOptions{}, MakeGitError(ecode)
|
||||
}
|
||||
|
||||
defer freeRevertOptions(&opts)
|
||||
return revertOptionsFromC(&opts), nil
|
||||
defer freeRevertOptions(&copts)
|
||||
return revertOptionsFromC(&copts), nil
|
||||
}
|
||||
|
||||
// Revert the provided commit leaving the index updated with the results of the revert
|
||||
|
@ -57,19 +63,19 @@ func (r *Repository) Revert(commit *Commit, revertOptions *RevertOptions) error
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var cOpts *C.git_revert_options
|
||||
var err error
|
||||
cOpts := populateRevertOptions(&C.git_revert_options{}, revertOptions, &err)
|
||||
defer freeRevertOptions(cOpts)
|
||||
|
||||
if revertOptions != nil {
|
||||
cOpts = revertOptions.toC()
|
||||
defer freeRevertOptions(cOpts)
|
||||
}
|
||||
|
||||
ecode := C.git_revert(r.ptr, commit.cast_ptr, cOpts)
|
||||
ret := C.git_revert(r.ptr, commit.cast_ptr, cOpts)
|
||||
runtime.KeepAlive(r)
|
||||
runtime.KeepAlive(commit)
|
||||
|
||||
if ecode < 0 {
|
||||
return MakeGitError(ecode)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
|
@ -81,11 +87,8 @@ func (r *Repository) RevertCommit(revertCommit *Commit, ourCommit *Commit, mainl
|
|||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var cOpts *C.git_merge_options
|
||||
|
||||
if mergeOptions != nil {
|
||||
cOpts = mergeOptions.toC()
|
||||
}
|
||||
cOpts := populateMergeOptions(&C.git_merge_options{}, mergeOptions)
|
||||
defer freeMergeOptions(cOpts)
|
||||
|
||||
var index *C.git_index
|
||||
|
||||
|
|
|
@ -60,11 +60,11 @@ func TestRevertCommit(t *testing.T) {
|
|||
revertOptions, err := DefaultRevertOptions()
|
||||
checkFatal(t, err)
|
||||
|
||||
index, err := repo.RevertCommit(commit, commit, 0, &revertOptions.MergeOpts)
|
||||
index, err := repo.RevertCommit(commit, commit, 0, &revertOptions.MergeOptions)
|
||||
checkFatal(t, err)
|
||||
defer index.Free()
|
||||
|
||||
err = repo.CheckoutIndex(index, &revertOptions.CheckoutOpts)
|
||||
err = repo.CheckoutIndex(index, &revertOptions.CheckoutOptions)
|
||||
checkFatal(t, err)
|
||||
|
||||
actualReadmeContents := readReadme(t, repo)
|
||||
|
|
|
@ -20,6 +20,7 @@ const (
|
|||
)
|
||||
|
||||
type Revspec struct {
|
||||
doNotCompare
|
||||
to *Object
|
||||
from *Object
|
||||
flags RevparseFlag
|
||||
|
|
|
@ -46,23 +46,45 @@ if [ -z "${BUILD_SHARED_LIBS}" ]; then
|
|||
usage
|
||||
fi
|
||||
|
||||
if [ -n "${BUILD_LIBGIT_REF}" ]; then
|
||||
git -C "${VENDORED_PATH}" checkout "${BUILD_LIBGIT_REF}"
|
||||
trap "git submodule update --init" EXIT
|
||||
fi
|
||||
|
||||
BUILD_DEPRECATED_HARD="ON"
|
||||
if [ "${BUILD_SYSTEM}" = "ON" ]; then
|
||||
BUILD_INSTALL_PREFIX=${SYSTEM_INSTALL_PREFIX-"/usr"}
|
||||
# Most system-wide installations won't intentionally omit deprecated symbols.
|
||||
BUILD_DEPRECATED_HARD="OFF"
|
||||
else
|
||||
BUILD_INSTALL_PREFIX="${BUILD_PATH}/install"
|
||||
mkdir -p "${BUILD_PATH}/install/lib"
|
||||
fi
|
||||
|
||||
USE_BUNDLED_ZLIB="ON"
|
||||
if [ "${USE_CHROMIUM_ZLIB}" = "ON" ]; then
|
||||
USE_BUNDLED_ZLIB="Chromium"
|
||||
fi
|
||||
|
||||
mkdir -p "${BUILD_PATH}/build" &&
|
||||
cd "${BUILD_PATH}/build" &&
|
||||
cmake -DTHREADSAFE=ON \
|
||||
-DBUILD_CLAR=OFF \
|
||||
-DBUILD_SHARED_LIBS"=${BUILD_SHARED_LIBS}" \
|
||||
-DREGEX_BACKEND=builtin \
|
||||
-DUSE_BUNDLED_ZLIB="${USE_BUNDLED_ZLIB}" \
|
||||
-DUSE_HTTPS=OFF \
|
||||
-DUSE_SSH=OFF \
|
||||
-DCMAKE_C_FLAGS=-fPIC \
|
||||
-DCMAKE_BUILD_TYPE="RelWithDebInfo" \
|
||||
-DCMAKE_INSTALL_PREFIX="${BUILD_INSTALL_PREFIX}" \
|
||||
-DCMAKE_INSTALL_LIBDIR="lib" \
|
||||
"${VENDORED_PATH}" &&
|
||||
-DDEPRECATE_HARD="${BUILD_DEPRECATE_HARD}" \
|
||||
"${VENDORED_PATH}"
|
||||
|
||||
exec cmake --build . --target install
|
||||
if which make nproc >/dev/null && [ -f Makefile ]; then
|
||||
# Make the build parallel if make is available and cmake used Makefiles.
|
||||
exec make "-j$(nproc --all)" install
|
||||
else
|
||||
exec cmake --build . --target install
|
||||
fi
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
#!/bin/sh
|
||||
|
||||
#
|
||||
# Install libgit2 to git2go in dynamic mode on Travis
|
||||
# Install libgit2 to go libgit2 in dynamic mode on Travis
|
||||
#
|
||||
|
||||
set -ex
|
||||
|
|
16
settings.go
16
settings.go
|
@ -93,6 +93,22 @@ func EnableCaching(enabled bool) error {
|
|||
}
|
||||
}
|
||||
|
||||
func EnableStrictHashVerification(enabled bool) error {
|
||||
if enabled {
|
||||
return setSizet(C.GIT_OPT_ENABLE_STRICT_HASH_VERIFICATION, 1)
|
||||
} else {
|
||||
return setSizet(C.GIT_OPT_ENABLE_STRICT_HASH_VERIFICATION, 0)
|
||||
}
|
||||
}
|
||||
|
||||
func EnableFsyncGitDir(enabled bool) error {
|
||||
if enabled {
|
||||
return setSizet(C.GIT_OPT_ENABLE_FSYNC_GITDIR, 1)
|
||||
} else {
|
||||
return setSizet(C.GIT_OPT_ENABLE_FSYNC_GITDIR, 0)
|
||||
}
|
||||
}
|
||||
|
||||
func CachedMemory() (current int, allowed int, err error) {
|
||||
return getSizetSizet(C.GIT_OPT_GET_CACHED_MEMORY)
|
||||
}
|
||||
|
|
|
@ -57,6 +57,22 @@ func TestEnableCaching(t *testing.T) {
|
|||
checkFatal(t, err)
|
||||
}
|
||||
|
||||
func TestEnableStrictHashVerification(t *testing.T) {
|
||||
err := EnableStrictHashVerification(false)
|
||||
checkFatal(t, err)
|
||||
|
||||
err = EnableStrictHashVerification(true)
|
||||
checkFatal(t, err)
|
||||
}
|
||||
|
||||
func TestEnableFsyncGitDir(t *testing.T) {
|
||||
err := EnableFsyncGitDir(false)
|
||||
checkFatal(t, err)
|
||||
|
||||
err = EnableFsyncGitDir(true)
|
||||
checkFatal(t, err)
|
||||
}
|
||||
|
||||
func TestCachedMemory(t *testing.T) {
|
||||
current, allowed, err := CachedMemory()
|
||||
checkFatal(t, err)
|
||||
|
@ -80,4 +96,4 @@ func TestSetCacheMaxSize(t *testing.T) {
|
|||
// revert to default 256MB
|
||||
err = SetCacheMaxSize(256 * 1024 * 1024)
|
||||
checkFatal(t, err)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,250 @@
|
|||
package git
|
||||
|
||||
/*
|
||||
#include <git2.h>
|
||||
|
||||
#include <git2/sys/credential.h>
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"crypto/md5"
|
||||
"crypto/sha1"
|
||||
"crypto/sha256"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"net"
|
||||
"net/url"
|
||||
"runtime"
|
||||
"strings"
|
||||
"unsafe"
|
||||
|
||||
"golang.org/x/crypto/ssh"
|
||||
)
|
||||
|
||||
// RegisterManagedSSHTransport registers a Go-native implementation of an SSH
|
||||
// transport that doesn't rely on any system libraries (e.g. libssh2).
|
||||
//
|
||||
// If Shutdown or ReInit are called, make sure that the smart transports are
|
||||
// freed before it.
|
||||
func RegisterManagedSSHTransport(protocol string) (*RegisteredSmartTransport, error) {
|
||||
return NewRegisteredSmartTransport(protocol, false, sshSmartSubtransportFactory)
|
||||
}
|
||||
|
||||
func registerManagedSSH() error {
|
||||
globalRegisteredSmartTransports.Lock()
|
||||
defer globalRegisteredSmartTransports.Unlock()
|
||||
|
||||
for _, protocol := range []string{"ssh", "ssh+git", "git+ssh"} {
|
||||
if _, ok := globalRegisteredSmartTransports.transports[protocol]; ok {
|
||||
continue
|
||||
}
|
||||
managed, err := newRegisteredSmartTransport(protocol, false, sshSmartSubtransportFactory, true)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to register transport for %q: %v", protocol, err)
|
||||
}
|
||||
globalRegisteredSmartTransports.transports[protocol] = managed
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func sshSmartSubtransportFactory(remote *Remote, transport *Transport) (SmartSubtransport, error) {
|
||||
return &sshSmartSubtransport{
|
||||
transport: transport,
|
||||
}, nil
|
||||
}
|
||||
|
||||
type sshSmartSubtransport struct {
|
||||
transport *Transport
|
||||
|
||||
lastAction SmartServiceAction
|
||||
client *ssh.Client
|
||||
session *ssh.Session
|
||||
stdin io.WriteCloser
|
||||
stdout io.Reader
|
||||
currentStream *sshSmartSubtransportStream
|
||||
}
|
||||
|
||||
func (t *sshSmartSubtransport) Action(urlString string, action SmartServiceAction) (SmartSubtransportStream, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
u, err := url.Parse(urlString)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
// Escape \ and '.
|
||||
uPath := strings.Replace(u.Path, `\`, `\\`, -1)
|
||||
uPath = strings.Replace(uPath, `'`, `\'`, -1)
|
||||
|
||||
// TODO: Add percentage decode similar to libgit2.
|
||||
// Refer: https://github.com/libgit2/libgit2/blob/358a60e1b46000ea99ef10b4dd709e92f75ff74b/src/str.c#L455-L481
|
||||
|
||||
var cmd string
|
||||
switch action {
|
||||
case SmartServiceActionUploadpackLs, SmartServiceActionUploadpack:
|
||||
if t.currentStream != nil {
|
||||
if t.lastAction == SmartServiceActionUploadpackLs {
|
||||
return t.currentStream, nil
|
||||
}
|
||||
t.Close()
|
||||
}
|
||||
cmd = fmt.Sprintf("git-upload-pack '%s'", uPath)
|
||||
|
||||
case SmartServiceActionReceivepackLs, SmartServiceActionReceivepack:
|
||||
if t.currentStream != nil {
|
||||
if t.lastAction == SmartServiceActionReceivepackLs {
|
||||
return t.currentStream, nil
|
||||
}
|
||||
t.Close()
|
||||
}
|
||||
cmd = fmt.Sprintf("git-receive-pack '%s'", uPath)
|
||||
|
||||
default:
|
||||
return nil, fmt.Errorf("unexpected action: %v", action)
|
||||
}
|
||||
|
||||
cred, err := t.transport.SmartCredentials("", CredentialTypeSSHKey|CredentialTypeSSHMemory)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer cred.Free()
|
||||
|
||||
sshConfig, err := getSSHConfigFromCredential(cred)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
sshConfig.HostKeyCallback = func(hostname string, remote net.Addr, key ssh.PublicKey) error {
|
||||
marshaledKey := key.Marshal()
|
||||
cert := &Certificate{
|
||||
Kind: CertificateHostkey,
|
||||
Hostkey: HostkeyCertificate{
|
||||
Kind: HostkeySHA1 | HostkeyMD5 | HostkeySHA256 | HostkeyRaw,
|
||||
HashMD5: md5.Sum(marshaledKey),
|
||||
HashSHA1: sha1.Sum(marshaledKey),
|
||||
HashSHA256: sha256.Sum256(marshaledKey),
|
||||
Hostkey: marshaledKey,
|
||||
SSHPublicKey: key,
|
||||
},
|
||||
}
|
||||
|
||||
return t.transport.SmartCertificateCheck(cert, true, hostname)
|
||||
}
|
||||
|
||||
var addr string
|
||||
if u.Port() != "" {
|
||||
addr = fmt.Sprintf("%s:%s", u.Hostname(), u.Port())
|
||||
} else {
|
||||
addr = fmt.Sprintf("%s:22", u.Hostname())
|
||||
}
|
||||
|
||||
t.client, err = ssh.Dial("tcp", addr, sshConfig)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
t.session, err = t.client.NewSession()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
t.stdin, err = t.session.StdinPipe()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
t.stdout, err = t.session.StdoutPipe()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if err := t.session.Start(cmd); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
t.lastAction = action
|
||||
t.currentStream = &sshSmartSubtransportStream{
|
||||
owner: t,
|
||||
}
|
||||
|
||||
return t.currentStream, nil
|
||||
}
|
||||
|
||||
func (t *sshSmartSubtransport) Close() error {
|
||||
t.currentStream = nil
|
||||
if t.client != nil {
|
||||
t.stdin.Close()
|
||||
t.session.Wait()
|
||||
t.session.Close()
|
||||
t.client = nil
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (t *sshSmartSubtransport) Free() {
|
||||
}
|
||||
|
||||
type sshSmartSubtransportStream struct {
|
||||
owner *sshSmartSubtransport
|
||||
}
|
||||
|
||||
func (stream *sshSmartSubtransportStream) Read(buf []byte) (int, error) {
|
||||
return stream.owner.stdout.Read(buf)
|
||||
}
|
||||
|
||||
func (stream *sshSmartSubtransportStream) Write(buf []byte) (int, error) {
|
||||
return stream.owner.stdin.Write(buf)
|
||||
}
|
||||
|
||||
func (stream *sshSmartSubtransportStream) Free() {
|
||||
}
|
||||
|
||||
func getSSHConfigFromCredential(cred *Credential) (*ssh.ClientConfig, error) {
|
||||
switch cred.Type() {
|
||||
case CredentialTypeSSHCustom:
|
||||
credSSHCustom := (*C.git_credential_ssh_custom)(unsafe.Pointer(cred.ptr))
|
||||
data, ok := pointerHandles.Get(credSSHCustom.payload).(*credentialSSHCustomData)
|
||||
if !ok {
|
||||
return nil, errors.New("unsupported custom SSH credentials")
|
||||
}
|
||||
return &ssh.ClientConfig{
|
||||
User: C.GoString(credSSHCustom.username),
|
||||
Auth: []ssh.AuthMethod{ssh.PublicKeys(data.signer)},
|
||||
}, nil
|
||||
}
|
||||
|
||||
username, _, privatekey, passphrase, err := cred.GetSSHKey()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var pemBytes []byte
|
||||
if cred.Type() == CredentialTypeSSHMemory {
|
||||
pemBytes = []byte(privatekey)
|
||||
} else {
|
||||
pemBytes, err = ioutil.ReadFile(privatekey)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
var key ssh.Signer
|
||||
if passphrase != "" {
|
||||
key, err = ssh.ParsePrivateKeyWithPassphrase(pemBytes, []byte(passphrase))
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
} else {
|
||||
key, err = ssh.ParsePrivateKey(pemBytes)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
return &ssh.ClientConfig{
|
||||
User: username,
|
||||
Auth: []ssh.AuthMethod{ssh.PublicKeys(key)},
|
||||
}, nil
|
||||
}
|
139
stash.go
139
stash.go
|
@ -3,7 +3,7 @@ package git
|
|||
/*
|
||||
#include <git2.h>
|
||||
|
||||
extern void _go_git_setup_stash_apply_progress_callbacks(git_stash_apply_options *opts);
|
||||
extern void _go_git_populate_stash_apply_callbacks(git_stash_apply_options *opts);
|
||||
extern int _go_git_stash_foreach(git_repository *repo, void *payload);
|
||||
*/
|
||||
import "C"
|
||||
|
@ -35,6 +35,7 @@ const (
|
|||
// StashCollection represents the possible operations that can be
|
||||
// performed on the collection of stashes for a repository.
|
||||
type StashCollection struct {
|
||||
doNotCompare
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
|
@ -113,34 +114,34 @@ const (
|
|||
|
||||
// StashApplyProgressCallback is the apply operation notification callback.
|
||||
type StashApplyProgressCallback func(progress StashApplyProgress) error
|
||||
|
||||
type stashApplyProgressData struct {
|
||||
Callback StashApplyProgressCallback
|
||||
Error error
|
||||
type stashApplyProgressCallbackData struct {
|
||||
callback StashApplyProgressCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export stashApplyProgressCb
|
||||
func stashApplyProgressCb(progress C.git_stash_apply_progress_t, handle unsafe.Pointer) int {
|
||||
//export stashApplyProgressCallback
|
||||
func stashApplyProgressCallback(progress C.git_stash_apply_progress_t, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*stashApplyProgressData)
|
||||
data, ok := payload.(*stashApplyProgressCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve data for handle")
|
||||
}
|
||||
|
||||
if data != nil {
|
||||
err := data.Callback(StashApplyProgress(progress))
|
||||
if err != nil {
|
||||
data.Error = err
|
||||
return C.GIT_EUSER
|
||||
}
|
||||
if data == nil || data.callback == nil {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
return 0
|
||||
|
||||
err := data.callback(StashApplyProgress(progress))
|
||||
if err != nil {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// StashApplyOptions represents options to control the apply operation.
|
||||
type StashApplyOptions struct {
|
||||
Flags StashApplyFlag
|
||||
CheckoutOptions CheckoutOpts // options to use when writing files to the working directory
|
||||
CheckoutOptions CheckoutOptions // options to use when writing files to the working directory
|
||||
ProgressCallback StashApplyProgressCallback // optional callback to notify the consumer of application progress
|
||||
}
|
||||
|
||||
|
@ -161,51 +162,45 @@ func DefaultStashApplyOptions() (StashApplyOptions, error) {
|
|||
}, nil
|
||||
}
|
||||
|
||||
func (opts *StashApplyOptions) toC() (
|
||||
optsC *C.git_stash_apply_options, progressData *stashApplyProgressData) {
|
||||
|
||||
if opts != nil {
|
||||
progressData = &stashApplyProgressData{
|
||||
Callback: opts.ProgressCallback,
|
||||
}
|
||||
|
||||
optsC = &C.git_stash_apply_options{
|
||||
version: C.GIT_STASH_APPLY_OPTIONS_VERSION,
|
||||
flags: C.uint32_t(opts.Flags),
|
||||
}
|
||||
populateCheckoutOpts(&optsC.checkout_options, &opts.CheckoutOptions)
|
||||
if opts.ProgressCallback != nil {
|
||||
C._go_git_setup_stash_apply_progress_callbacks(optsC)
|
||||
optsC.progress_payload = pointerHandles.Track(progressData)
|
||||
}
|
||||
func populateStashApplyOptions(copts *C.git_stash_apply_options, opts *StashApplyOptions, errorTarget *error) *C.git_stash_apply_options {
|
||||
C.git_stash_apply_options_init(copts, C.GIT_STASH_APPLY_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
return
|
||||
copts.flags = C.uint32_t(opts.Flags)
|
||||
populateCheckoutOptions(&copts.checkout_options, &opts.CheckoutOptions, errorTarget)
|
||||
if opts.ProgressCallback != nil {
|
||||
progressData := &stashApplyProgressCallbackData{
|
||||
callback: opts.ProgressCallback,
|
||||
errorTarget: errorTarget,
|
||||
}
|
||||
C._go_git_populate_stash_apply_callbacks(copts)
|
||||
copts.progress_payload = pointerHandles.Track(progressData)
|
||||
}
|
||||
return copts
|
||||
}
|
||||
|
||||
// should be called after every call to toC() as deferred.
|
||||
func untrackStashApplyOptionsCallback(optsC *C.git_stash_apply_options) {
|
||||
if optsC != nil && optsC.progress_payload != nil {
|
||||
pointerHandles.Untrack(optsC.progress_payload)
|
||||
func freeStashApplyOptions(copts *C.git_stash_apply_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func freeStashApplyOptions(optsC *C.git_stash_apply_options) {
|
||||
if optsC != nil {
|
||||
freeCheckoutOpts(&optsC.checkout_options)
|
||||
if copts.progress_payload != nil {
|
||||
pointerHandles.Untrack(copts.progress_payload)
|
||||
}
|
||||
freeCheckoutOptions(&copts.checkout_options)
|
||||
}
|
||||
|
||||
// Apply applies a single stashed state from the stash list.
|
||||
//
|
||||
// If local changes in the working directory conflict with changes in the
|
||||
// stash then ErrConflict will be returned. In this case, the index
|
||||
// stash then ErrorCodeConflict will be returned. In this case, the index
|
||||
// will always remain unmodified and all files in the working directory will
|
||||
// remain unmodified. However, if you are restoring untracked files or
|
||||
// ignored files and there is a conflict when applying the modified files,
|
||||
// then those files will remain in the working directory.
|
||||
//
|
||||
// If passing the StashApplyReinstateIndex flag and there would be conflicts
|
||||
// when reinstating the index, the function will return ErrConflict
|
||||
// when reinstating the index, the function will return ErrorCodeConflict
|
||||
// and both the working directory and index will be left unmodified.
|
||||
//
|
||||
// Note that a minimum checkout strategy of 'CheckoutSafe' is implied.
|
||||
|
@ -213,15 +208,15 @@ func freeStashApplyOptions(optsC *C.git_stash_apply_options) {
|
|||
// 'index' is the position within the stash list. 0 points to the most
|
||||
// recent stashed state.
|
||||
//
|
||||
// Returns error code ErrNotFound if there's no stashed state for the given
|
||||
// index, error code ErrConflict if local changes in the working directory
|
||||
// Returns error code ErrorCodeNotFound if there's no stashed state for the given
|
||||
// index, error code ErrorCodeConflict if local changes in the working directory
|
||||
// conflict with changes in the stash, the user returned error from the
|
||||
// StashApplyProgressCallback, if any, or other error code.
|
||||
//
|
||||
// Error codes can be interogated with IsErrorCode(err, ErrNotFound).
|
||||
// Error codes can be interogated with IsErrorCode(err, ErrorCodeNotFound).
|
||||
func (c *StashCollection) Apply(index int, opts StashApplyOptions) error {
|
||||
optsC, progressData := opts.toC()
|
||||
defer untrackStashApplyOptionsCallback(optsC)
|
||||
var err error
|
||||
optsC := populateStashApplyOptions(&C.git_stash_apply_options{}, &opts, &err)
|
||||
defer freeStashApplyOptions(optsC)
|
||||
|
||||
runtime.LockOSThread()
|
||||
|
@ -229,8 +224,8 @@ func (c *StashCollection) Apply(index int, opts StashApplyOptions) error {
|
|||
|
||||
ret := C.git_stash_apply(c.repo.ptr, C.size_t(index), optsC)
|
||||
runtime.KeepAlive(c)
|
||||
if ret == C.GIT_EUSER {
|
||||
return progressData.Error
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
|
@ -245,26 +240,25 @@ func (c *StashCollection) Apply(index int, opts StashApplyOptions) error {
|
|||
// 'message' is the message used when creating the stash and 'id'
|
||||
// is the commit id of the stash.
|
||||
type StashCallback func(index int, message string, id *Oid) error
|
||||
|
||||
type stashCallbackData struct {
|
||||
Callback StashCallback
|
||||
Error error
|
||||
callback StashCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export stashForeachCb
|
||||
func stashForeachCb(index C.size_t, message *C.char, id *C.git_oid, handle unsafe.Pointer) int {
|
||||
//export stashForeachCallback
|
||||
func stashForeachCallback(index C.size_t, message *C.char, id *C.git_oid, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*stashCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve data for handle")
|
||||
}
|
||||
|
||||
err := data.Callback(int(index), C.GoString(message), newOidFromC(id))
|
||||
err := data.callback(int(index), C.GoString(message), newOidFromC(id))
|
||||
if err != nil {
|
||||
data.Error = err
|
||||
return C.GIT_EUSER
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// Foreach loops over all the stashed states and calls the callback
|
||||
|
@ -272,10 +266,11 @@ func stashForeachCb(index C.size_t, message *C.char, id *C.git_oid, handle unsaf
|
|||
//
|
||||
// If callback returns an error, this will stop looping.
|
||||
func (c *StashCollection) Foreach(callback StashCallback) error {
|
||||
var err error
|
||||
data := stashCallbackData{
|
||||
Callback: callback,
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
|
||||
handle := pointerHandles.Track(&data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
|
||||
|
@ -284,8 +279,8 @@ func (c *StashCollection) Foreach(callback StashCallback) error {
|
|||
|
||||
ret := C._go_git_stash_foreach(c.repo.ptr, handle)
|
||||
runtime.KeepAlive(c)
|
||||
if ret == C.GIT_EUSER {
|
||||
return data.Error
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
|
@ -298,7 +293,7 @@ func (c *StashCollection) Foreach(callback StashCallback) error {
|
|||
// 'index' is the position within the stash list. 0 points
|
||||
// to the most recent stashed state.
|
||||
//
|
||||
// Returns error code ErrNotFound if there's no stashed
|
||||
// Returns error code ErrorCodeNotFound if there's no stashed
|
||||
// state for the given index.
|
||||
func (c *StashCollection) Drop(index int) error {
|
||||
runtime.LockOSThread()
|
||||
|
@ -320,11 +315,11 @@ func (c *StashCollection) Drop(index int) error {
|
|||
//
|
||||
// 'opts' controls how stashes are applied.
|
||||
//
|
||||
// Returns error code ErrNotFound if there's no stashed
|
||||
// Returns error code ErrorCodeNotFound if there's no stashed
|
||||
// state for the given index.
|
||||
func (c *StashCollection) Pop(index int, opts StashApplyOptions) error {
|
||||
optsC, progressData := opts.toC()
|
||||
defer untrackStashApplyOptionsCallback(optsC)
|
||||
var err error
|
||||
optsC := populateStashApplyOptions(&C.git_stash_apply_options{}, &opts, &err)
|
||||
defer freeStashApplyOptions(optsC)
|
||||
|
||||
runtime.LockOSThread()
|
||||
|
@ -332,8 +327,8 @@ func (c *StashCollection) Pop(index int, opts StashApplyOptions) error {
|
|||
|
||||
ret := C.git_stash_pop(c.repo.ptr, C.size_t(index), optsC)
|
||||
runtime.KeepAlive(c)
|
||||
if ret == C.GIT_EUSER {
|
||||
return progressData.Error
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
|
|
|
@ -56,8 +56,8 @@ func TestStash(t *testing.T) {
|
|||
// Apply: no stash for the given index
|
||||
|
||||
err = repo.Stashes.Apply(1, opts)
|
||||
if !IsErrorCode(err, ErrNotFound) {
|
||||
t.Errorf("expecting GIT_ENOTFOUND error code %d, got %v", ErrNotFound, err)
|
||||
if !IsErrorCode(err, ErrorCodeNotFound) {
|
||||
t.Errorf("expecting GIT_ENOTFOUND error code %d, got %v", ErrorCodeNotFound, err)
|
||||
}
|
||||
|
||||
// Apply: callback stopped
|
||||
|
|
|
@ -55,6 +55,7 @@ func statusEntryFromC(statusEntry *C.git_status_entry) StatusEntry {
|
|||
}
|
||||
|
||||
type StatusList struct {
|
||||
doNotCompare
|
||||
ptr *C.git_status_list
|
||||
r *Repository
|
||||
}
|
||||
|
|
86
submodule.go
86
submodule.go
|
@ -6,6 +6,7 @@ package git
|
|||
extern int _go_git_visit_submodule(git_repository *repo, void *fct);
|
||||
*/
|
||||
import "C"
|
||||
|
||||
import (
|
||||
"runtime"
|
||||
"unsafe"
|
||||
|
@ -13,12 +14,13 @@ import (
|
|||
|
||||
// SubmoduleUpdateOptions
|
||||
type SubmoduleUpdateOptions struct {
|
||||
*CheckoutOpts
|
||||
*FetchOptions
|
||||
CheckoutOptions CheckoutOptions
|
||||
FetchOptions FetchOptions
|
||||
}
|
||||
|
||||
// Submodule
|
||||
type Submodule struct {
|
||||
doNotCompare
|
||||
ptr *C.git_submodule
|
||||
r *Repository
|
||||
}
|
||||
|
@ -80,6 +82,7 @@ const (
|
|||
)
|
||||
|
||||
type SubmoduleCollection struct {
|
||||
doNotCompare
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
|
@ -106,31 +109,52 @@ func (c *SubmoduleCollection) Lookup(name string) (*Submodule, error) {
|
|||
return newSubmoduleFromC(ptr, c.repo), nil
|
||||
}
|
||||
|
||||
type SubmoduleCbk func(sub *Submodule, name string) int
|
||||
|
||||
//export SubmoduleVisitor
|
||||
func SubmoduleVisitor(csub unsafe.Pointer, name *C.char, handle unsafe.Pointer) C.int {
|
||||
sub := &Submodule{(*C.git_submodule)(csub), nil}
|
||||
|
||||
if callback, ok := pointerHandles.Get(handle).(SubmoduleCbk); ok {
|
||||
return (C.int)(callback(sub, C.GoString(name)))
|
||||
} else {
|
||||
panic("invalid submodule visitor callback")
|
||||
}
|
||||
// SubmoduleCallback is a function that is called for every submodule found in SubmoduleCollection.Foreach.
|
||||
type SubmoduleCallback func(sub *Submodule, name string) error
|
||||
type submoduleCallbackData struct {
|
||||
callback SubmoduleCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
func (c *SubmoduleCollection) Foreach(cbk SubmoduleCbk) error {
|
||||
//export submoduleCallback
|
||||
func submoduleCallback(csub unsafe.Pointer, name *C.char, handle unsafe.Pointer) C.int {
|
||||
sub := &Submodule{ptr: (*C.git_submodule)(csub)}
|
||||
|
||||
data, ok := pointerHandles.Get(handle).(submoduleCallbackData)
|
||||
if !ok {
|
||||
panic("invalid submodule visitor callback")
|
||||
}
|
||||
|
||||
err := data.callback(sub, C.GoString(name))
|
||||
if err != nil {
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
func (c *SubmoduleCollection) Foreach(callback SubmoduleCallback) error {
|
||||
var err error
|
||||
data := submoduleCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
handle := pointerHandles.Track(cbk)
|
||||
handle := pointerHandles.Track(data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
|
||||
ret := C._go_git_visit_submodule(c.repo.ptr, handle)
|
||||
runtime.KeepAlive(c)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
@ -341,17 +365,18 @@ func (sub *Submodule) Open() (*Repository, error) {
|
|||
}
|
||||
|
||||
func (sub *Submodule) Update(init bool, opts *SubmoduleUpdateOptions) error {
|
||||
var copts C.git_submodule_update_options
|
||||
err := populateSubmoduleUpdateOptions(&copts, opts)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
var err error
|
||||
cOpts := populateSubmoduleUpdateOptions(&C.git_submodule_update_options{}, opts, &err)
|
||||
defer freeSubmoduleUpdateOptions(cOpts)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C.git_submodule_update(sub.ptr, cbool(init), &copts)
|
||||
ret := C.git_submodule_update(sub.ptr, cbool(init), cOpts)
|
||||
runtime.KeepAlive(sub)
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
@ -359,15 +384,22 @@ func (sub *Submodule) Update(init bool, opts *SubmoduleUpdateOptions) error {
|
|||
return nil
|
||||
}
|
||||
|
||||
func populateSubmoduleUpdateOptions(ptr *C.git_submodule_update_options, opts *SubmoduleUpdateOptions) error {
|
||||
C.git_submodule_update_options_init(ptr, C.GIT_SUBMODULE_UPDATE_OPTIONS_VERSION)
|
||||
|
||||
func populateSubmoduleUpdateOptions(copts *C.git_submodule_update_options, opts *SubmoduleUpdateOptions, errorTarget *error) *C.git_submodule_update_options {
|
||||
C.git_submodule_update_options_init(copts, C.GIT_SUBMODULE_UPDATE_OPTIONS_VERSION)
|
||||
if opts == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
populateCheckoutOpts(&ptr.checkout_opts, opts.CheckoutOpts)
|
||||
populateFetchOptions(&ptr.fetch_opts, opts.FetchOptions)
|
||||
populateCheckoutOptions(&copts.checkout_opts, &opts.CheckoutOptions, errorTarget)
|
||||
populateFetchOptions(&copts.fetch_opts, &opts.FetchOptions, errorTarget)
|
||||
|
||||
return nil
|
||||
return copts
|
||||
}
|
||||
|
||||
func freeSubmoduleUpdateOptions(copts *C.git_submodule_update_options) {
|
||||
if copts == nil {
|
||||
return
|
||||
}
|
||||
freeCheckoutOptions(&copts.checkout_opts)
|
||||
freeFetchOptions(&copts.fetch_opts)
|
||||
}
|
||||
|
|
|
@ -15,9 +15,9 @@ func TestSubmoduleForeach(t *testing.T) {
|
|||
checkFatal(t, err)
|
||||
|
||||
i := 0
|
||||
err = repo.Submodules.Foreach(func(sub *Submodule, name string) int {
|
||||
err = repo.Submodules.Foreach(func(sub *Submodule, name string) error {
|
||||
i++
|
||||
return 0
|
||||
return nil
|
||||
})
|
||||
checkFatal(t, err)
|
||||
|
||||
|
|
42
tag.go
42
tag.go
|
@ -13,6 +13,7 @@ import (
|
|||
|
||||
// Tag
|
||||
type Tag struct {
|
||||
doNotCompare
|
||||
Object
|
||||
cast_ptr *C.git_tag
|
||||
}
|
||||
|
@ -64,6 +65,7 @@ func (t *Tag) TargetType() ObjectType {
|
|||
}
|
||||
|
||||
type TagsCollection struct {
|
||||
doNotCompare
|
||||
repo *Repository
|
||||
}
|
||||
|
||||
|
@ -159,7 +161,7 @@ func (c *TagsCollection) List() ([]string, error) {
|
|||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
defer C.git_strarray_free(&strC)
|
||||
defer C.git_strarray_dispose(&strC)
|
||||
|
||||
tags := makeStringsFromCStrings(strC.strings, int(strC.count))
|
||||
return tags, nil
|
||||
|
@ -183,7 +185,7 @@ func (c *TagsCollection) ListWithMatch(pattern string) ([]string, error) {
|
|||
if ecode < 0 {
|
||||
return nil, MakeGitError(ecode)
|
||||
}
|
||||
defer C.git_strarray_free(&strC)
|
||||
defer C.git_strarray_dispose(&strC)
|
||||
|
||||
tags := makeStringsFromCStrings(strC.strings, int(strC.count))
|
||||
return tags, nil
|
||||
|
@ -197,48 +199,48 @@ func (c *TagsCollection) ListWithMatch(pattern string) ([]string, error) {
|
|||
// so repo.LookupTag() will return an error for these tags. Use
|
||||
// repo.References.Lookup() instead.
|
||||
type TagForeachCallback func(name string, id *Oid) error
|
||||
type tagForeachData struct {
|
||||
callback TagForeachCallback
|
||||
err error
|
||||
type tagForeachCallbackData struct {
|
||||
callback TagForeachCallback
|
||||
errorTarget *error
|
||||
}
|
||||
|
||||
//export gitTagForeachCb
|
||||
func gitTagForeachCb(name *C.char, id *C.git_oid, handle unsafe.Pointer) int {
|
||||
//export tagForeachCallback
|
||||
func tagForeachCallback(name *C.char, id *C.git_oid, handle unsafe.Pointer) C.int {
|
||||
payload := pointerHandles.Get(handle)
|
||||
data, ok := payload.(*tagForeachData)
|
||||
data, ok := payload.(*tagForeachCallbackData)
|
||||
if !ok {
|
||||
panic("could not retrieve tag foreach CB handle")
|
||||
}
|
||||
|
||||
err := data.callback(C.GoString(name), newOidFromC(id))
|
||||
if err != nil {
|
||||
data.err = err
|
||||
return C.GIT_EUSER
|
||||
*data.errorTarget = err
|
||||
return C.int(ErrorCodeUser)
|
||||
}
|
||||
|
||||
return 0
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
// Foreach calls the callback for each tag in the repository.
|
||||
func (c *TagsCollection) Foreach(callback TagForeachCallback) error {
|
||||
data := tagForeachData{
|
||||
callback: callback,
|
||||
err: nil,
|
||||
var err error
|
||||
data := tagForeachCallbackData{
|
||||
callback: callback,
|
||||
errorTarget: &err,
|
||||
}
|
||||
|
||||
handle := pointerHandles.Track(&data)
|
||||
defer pointerHandles.Untrack(handle)
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
err := C._go_git_tag_foreach(c.repo.ptr, handle)
|
||||
ret := C._go_git_tag_foreach(c.repo.ptr, handle)
|
||||
runtime.KeepAlive(c)
|
||||
if err == C.GIT_EUSER {
|
||||
return data.err
|
||||
if ret == C.int(ErrorCodeUser) && err != nil {
|
||||
return err
|
||||
}
|
||||
if err < 0 {
|
||||
return MakeGitError(err)
|
||||
if ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
ref: refs/heads/master
|
|
@ -0,0 +1,6 @@
|
|||
[core]
|
||||
repositoryformatversion = 0
|
||||
filemode = true
|
||||
bare = true
|
||||
[remote "origin"]
|
||||
url = https://github.com/libgit2/TestGitRepository
|
|
@ -0,0 +1 @@
|
|||
Unnamed repository; edit this file 'description' to name the repository.
|
|
@ -0,0 +1,6 @@
|
|||
# git ls-files --others --exclude-from=.git/info/exclude
|
||||
# Lines that start with '#' are comments.
|
||||
# For a project mostly in C, the following would be a good set of
|
||||
# exclude patterns (uncomment them if you want to use them):
|
||||
# *.[oa]
|
||||
# *~
|
|
@ -0,0 +1,8 @@
|
|||
0966a434eb1a025db6b71485ab63a3bfbea520b6 refs/heads/first-merge
|
||||
49322bb17d3acc9146f98c97d078513228bbf3c0 refs/heads/master
|
||||
42e4e7c5e507e113ebbb7801b16b52cf867b7ce1 refs/heads/no-parent
|
||||
d96c4e80345534eccee5ac7b07fc7603b56124cb refs/tags/annotated_tag
|
||||
c070ad8c08840c8116da865b2d65593a6bb9cd2a refs/tags/annotated_tag^{}
|
||||
55a1a760df4b86a02094a904dfa511deb5655905 refs/tags/blob
|
||||
8f50ba15d49353813cc6e20298002c0d17b0a9ee refs/tags/commit_tree
|
||||
6e0c7bdb9b4ed93212491ee778ca1c65047cab4e refs/tags/nearly-dangling
|
Binary file not shown.
|
@ -0,0 +1,2 @@
|
|||
P pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.pack
|
||||
|
BIN
testdata/TestGitRepository.git/objects/pack/pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.bitmap
vendored
Normal file
BIN
testdata/TestGitRepository.git/objects/pack/pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.bitmap
vendored
Normal file
Binary file not shown.
BIN
testdata/TestGitRepository.git/objects/pack/pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.idx
vendored
Normal file
BIN
testdata/TestGitRepository.git/objects/pack/pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.idx
vendored
Normal file
Binary file not shown.
BIN
testdata/TestGitRepository.git/objects/pack/pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.pack
vendored
Normal file
BIN
testdata/TestGitRepository.git/objects/pack/pack-ccace4e169a0858c13d9ae781a91d76fc33769b8.pack
vendored
Normal file
Binary file not shown.
|
@ -0,0 +1,9 @@
|
|||
# pack-refs with: peeled fully-peeled sorted
|
||||
0966a434eb1a025db6b71485ab63a3bfbea520b6 refs/heads/first-merge
|
||||
49322bb17d3acc9146f98c97d078513228bbf3c0 refs/heads/master
|
||||
42e4e7c5e507e113ebbb7801b16b52cf867b7ce1 refs/heads/no-parent
|
||||
d96c4e80345534eccee5ac7b07fc7603b56124cb refs/tags/annotated_tag
|
||||
^c070ad8c08840c8116da865b2d65593a6bb9cd2a
|
||||
55a1a760df4b86a02094a904dfa511deb5655905 refs/tags/blob
|
||||
8f50ba15d49353813cc6e20298002c0d17b0a9ee refs/tags/commit_tree
|
||||
6e0c7bdb9b4ed93212491ee778ca1c65047cab4e refs/tags/nearly-dangling
|
|
@ -0,0 +1 @@
|
|||
49322bb17d3acc9146f98c97d078513228bbf3c0
|
|
@ -0,0 +1,504 @@
|
|||
package git
|
||||
|
||||
/*
|
||||
#include <string.h>
|
||||
|
||||
#include <git2.h>
|
||||
#include <git2/sys/transport.h>
|
||||
|
||||
typedef struct {
|
||||
git_smart_subtransport parent;
|
||||
void *handle;
|
||||
} _go_managed_smart_subtransport;
|
||||
|
||||
typedef struct {
|
||||
git_smart_subtransport_stream parent;
|
||||
void *handle;
|
||||
} _go_managed_smart_subtransport_stream;
|
||||
|
||||
int _go_git_transport_register(const char *prefix, void *handle);
|
||||
int _go_git_transport_smart(git_transport **out, git_remote *owner, int stateless, _go_managed_smart_subtransport *subtransport_payload);
|
||||
void _go_git_setup_smart_subtransport_stream(_go_managed_smart_subtransport_stream *stream);
|
||||
*/
|
||||
import "C"
|
||||
import (
|
||||
"fmt"
|
||||
"io"
|
||||
"reflect"
|
||||
"runtime"
|
||||
"sync"
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
var (
|
||||
// globalRegisteredSmartTransports is a mapping of global, libgit2 go-managed
|
||||
// transports.
|
||||
globalRegisteredSmartTransports = struct {
|
||||
sync.Mutex
|
||||
transports map[string]*RegisteredSmartTransport
|
||||
}{
|
||||
transports: make(map[string]*RegisteredSmartTransport),
|
||||
}
|
||||
)
|
||||
|
||||
// unregisterManagedTransports unregisters all libgit2 go-managed transports.
|
||||
func unregisterManagedTransports() error {
|
||||
globalRegisteredSmartTransports.Lock()
|
||||
originalTransports := globalRegisteredSmartTransports.transports
|
||||
globalRegisteredSmartTransports.transports = make(map[string]*RegisteredSmartTransport)
|
||||
globalRegisteredSmartTransports.Unlock()
|
||||
|
||||
var err error
|
||||
for protocol, managed := range originalTransports {
|
||||
unregisterErr := managed.Free()
|
||||
if err == nil && unregisterErr != nil {
|
||||
err = fmt.Errorf("failed to unregister transport for %q: %v", protocol, unregisterErr)
|
||||
}
|
||||
}
|
||||
return err
|
||||
}
|
||||
|
||||
// SmartServiceAction is an action that the smart transport can ask a
|
||||
// subtransport to perform.
|
||||
type SmartServiceAction int
|
||||
|
||||
const (
|
||||
// SmartServiceActionUploadpackLs is used upon connecting to a remote, and is
|
||||
// used to perform reference discovery prior to performing a pull operation.
|
||||
SmartServiceActionUploadpackLs SmartServiceAction = C.GIT_SERVICE_UPLOADPACK_LS
|
||||
|
||||
// SmartServiceActionUploadpack is used when performing a pull operation.
|
||||
SmartServiceActionUploadpack SmartServiceAction = C.GIT_SERVICE_UPLOADPACK
|
||||
|
||||
// SmartServiceActionReceivepackLs is used upon connecting to a remote, and is
|
||||
// used to perform reference discovery prior to performing a push operation.
|
||||
SmartServiceActionReceivepackLs SmartServiceAction = C.GIT_SERVICE_RECEIVEPACK_LS
|
||||
|
||||
// SmartServiceActionReceivepack is used when performing a push operation.
|
||||
SmartServiceActionReceivepack SmartServiceAction = C.GIT_SERVICE_RECEIVEPACK
|
||||
)
|
||||
|
||||
// Transport encapsulates a way to communicate with a Remote.
|
||||
type Transport struct {
|
||||
doNotCompare
|
||||
ptr *C.git_transport
|
||||
}
|
||||
|
||||
// SmartRemoteConnectOptions gets a copy of the proxy options for this transport.
|
||||
func (t *Transport) SmartRemoteConnectOptions() (*RemoteConnectOptions, error) {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
var copts C.git_remote_connect_options
|
||||
if ret := C.git_transport_remote_connect_options(&copts, t.ptr); ret < 0 {
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return remoteConnectOptionsFromC(&copts), nil
|
||||
}
|
||||
|
||||
// SmartCredentials calls the credentials callback for this transport.
|
||||
func (t *Transport) SmartCredentials(user string, methods CredentialType) (*Credential, error) {
|
||||
cred := newCredential()
|
||||
var cstr *C.char
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
if user != "" {
|
||||
cstr = C.CString(user)
|
||||
defer C.free(unsafe.Pointer(cstr))
|
||||
}
|
||||
ret := C.git_transport_smart_credentials(&cred.ptr, t.ptr, cstr, C.int(methods))
|
||||
if ret != 0 {
|
||||
cred.Free()
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
return cred, nil
|
||||
}
|
||||
|
||||
// SmartCertificateCheck calls the certificate check for this transport.
|
||||
func (t *Transport) SmartCertificateCheck(cert *Certificate, valid bool, hostname string) error {
|
||||
var ccert *C.git_cert
|
||||
switch cert.Kind {
|
||||
case CertificateHostkey:
|
||||
chostkeyCert := C.git_cert_hostkey{
|
||||
parent: C.git_cert{
|
||||
cert_type: C.GIT_CERT_HOSTKEY_LIBSSH2,
|
||||
},
|
||||
_type: C.git_cert_ssh_t(cert.Kind),
|
||||
hostkey: (*C.char)(C.CBytes(cert.Hostkey.Hostkey)),
|
||||
hostkey_len: C.size_t(len(cert.Hostkey.Hostkey)),
|
||||
}
|
||||
defer C.free(unsafe.Pointer(chostkeyCert.hostkey))
|
||||
C.memcpy(unsafe.Pointer(&chostkeyCert.hash_md5[0]), unsafe.Pointer(&cert.Hostkey.HashMD5[0]), C.size_t(len(cert.Hostkey.HashMD5)))
|
||||
C.memcpy(unsafe.Pointer(&chostkeyCert.hash_sha1[0]), unsafe.Pointer(&cert.Hostkey.HashSHA1[0]), C.size_t(len(cert.Hostkey.HashSHA1)))
|
||||
C.memcpy(unsafe.Pointer(&chostkeyCert.hash_sha256[0]), unsafe.Pointer(&cert.Hostkey.HashSHA256[0]), C.size_t(len(cert.Hostkey.HashSHA256)))
|
||||
if cert.Hostkey.SSHPublicKey.Type() == "ssh-rsa" {
|
||||
chostkeyCert.raw_type = C.GIT_CERT_SSH_RAW_TYPE_RSA
|
||||
} else if cert.Hostkey.SSHPublicKey.Type() == "ssh-dss" {
|
||||
chostkeyCert.raw_type = C.GIT_CERT_SSH_RAW_TYPE_DSS
|
||||
} else {
|
||||
chostkeyCert.raw_type = C.GIT_CERT_SSH_RAW_TYPE_UNKNOWN
|
||||
}
|
||||
ccert = (*C.git_cert)(unsafe.Pointer(&chostkeyCert))
|
||||
|
||||
case CertificateX509:
|
||||
cx509Cert := C.git_cert_x509{
|
||||
parent: C.git_cert{
|
||||
cert_type: C.GIT_CERT_X509,
|
||||
},
|
||||
len: C.size_t(len(cert.X509.Raw)),
|
||||
data: C.CBytes(cert.X509.Raw),
|
||||
}
|
||||
defer C.free(cx509Cert.data)
|
||||
ccert = (*C.git_cert)(unsafe.Pointer(&cx509Cert))
|
||||
}
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
chostname := C.CString(hostname)
|
||||
defer C.free(unsafe.Pointer(chostname))
|
||||
|
||||
cvalid := C.int(0)
|
||||
if valid {
|
||||
cvalid = C.int(1)
|
||||
}
|
||||
|
||||
ret := C.git_transport_smart_certificate_check(t.ptr, ccert, cvalid, chostname)
|
||||
if ret != 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// SmartSubtransport is the interface for custom subtransports which carry data
|
||||
// for the smart transport.
|
||||
type SmartSubtransport interface {
|
||||
// Action creates a SmartSubtransportStream for the provided url and
|
||||
// requested action.
|
||||
Action(url string, action SmartServiceAction) (SmartSubtransportStream, error)
|
||||
|
||||
// Close closes the SmartSubtransport.
|
||||
//
|
||||
// Subtransports are guaranteed a call to Close between
|
||||
// calls to Action, except for the following two "natural" progressions
|
||||
// of actions against a constant URL.
|
||||
//
|
||||
// 1. UPLOADPACK_LS -> UPLOADPACK
|
||||
// 2. RECEIVEPACK_LS -> RECEIVEPACK
|
||||
Close() error
|
||||
|
||||
// Free releases the resources of the SmartSubtransport.
|
||||
Free()
|
||||
}
|
||||
|
||||
// SmartSubtransportStream is the interface for streams used by the smart
|
||||
// transport to read and write data from a subtransport.
|
||||
type SmartSubtransportStream interface {
|
||||
io.Reader
|
||||
io.Writer
|
||||
|
||||
// Free releases the resources of the SmartSubtransportStream.
|
||||
Free()
|
||||
}
|
||||
|
||||
// SmartSubtransportCallback is a function which creates a new subtransport for
|
||||
// the smart transport.
|
||||
type SmartSubtransportCallback func(remote *Remote, transport *Transport) (SmartSubtransport, error)
|
||||
|
||||
// RegisteredSmartTransport represents a transport that has been registered.
|
||||
type RegisteredSmartTransport struct {
|
||||
doNotCompare
|
||||
name string
|
||||
stateless bool
|
||||
callback SmartSubtransportCallback
|
||||
handle unsafe.Pointer
|
||||
}
|
||||
|
||||
// NewRegisteredSmartTransport adds a custom transport definition, to be used
|
||||
// in addition to the built-in set of transports that come with libgit2.
|
||||
func NewRegisteredSmartTransport(
|
||||
name string,
|
||||
stateless bool,
|
||||
callback SmartSubtransportCallback,
|
||||
) (*RegisteredSmartTransport, error) {
|
||||
return newRegisteredSmartTransport(name, stateless, callback, false)
|
||||
}
|
||||
|
||||
func newRegisteredSmartTransport(
|
||||
name string,
|
||||
stateless bool,
|
||||
callback SmartSubtransportCallback,
|
||||
global bool,
|
||||
) (*RegisteredSmartTransport, error) {
|
||||
if !global {
|
||||
// Check if we had already registered a smart transport for this protocol. If
|
||||
// we had, free it. The user is now responsible for this transport for the
|
||||
// lifetime of the library.
|
||||
globalRegisteredSmartTransports.Lock()
|
||||
if managed, ok := globalRegisteredSmartTransports.transports[name]; ok {
|
||||
delete(globalRegisteredSmartTransports.transports, name)
|
||||
globalRegisteredSmartTransports.Unlock()
|
||||
|
||||
err := managed.Free()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
} else {
|
||||
globalRegisteredSmartTransports.Unlock()
|
||||
}
|
||||
}
|
||||
|
||||
cstr := C.CString(name)
|
||||
defer C.free(unsafe.Pointer(cstr))
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
registeredSmartTransport := &RegisteredSmartTransport{
|
||||
name: name,
|
||||
stateless: stateless,
|
||||
callback: callback,
|
||||
}
|
||||
registeredSmartTransport.handle = pointerHandles.Track(registeredSmartTransport)
|
||||
|
||||
ret := C._go_git_transport_register(cstr, registeredSmartTransport.handle)
|
||||
if ret != 0 {
|
||||
pointerHandles.Untrack(registeredSmartTransport.handle)
|
||||
return nil, MakeGitError(ret)
|
||||
}
|
||||
|
||||
runtime.SetFinalizer(registeredSmartTransport, (*RegisteredSmartTransport).Free)
|
||||
return registeredSmartTransport, nil
|
||||
}
|
||||
|
||||
// Free releases all resources used by the RegisteredSmartTransport and
|
||||
// unregisters the custom transport definition referenced by it.
|
||||
func (t *RegisteredSmartTransport) Free() error {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
cstr := C.CString(t.name)
|
||||
defer C.free(unsafe.Pointer(cstr))
|
||||
|
||||
if ret := C.git_transport_unregister(cstr); ret < 0 {
|
||||
return MakeGitError(ret)
|
||||
}
|
||||
|
||||
pointerHandles.Untrack(t.handle)
|
||||
runtime.SetFinalizer(t, nil)
|
||||
t.handle = nil
|
||||
return nil
|
||||
}
|
||||
|
||||
//export smartTransportCallback
|
||||
func smartTransportCallback(
|
||||
errorMessage **C.char,
|
||||
out **C.git_transport,
|
||||
owner *C.git_remote,
|
||||
handle unsafe.Pointer,
|
||||
) C.int {
|
||||
registeredSmartTransport := pointerHandles.Get(handle).(*RegisteredSmartTransport)
|
||||
remote, ok := remotePointers.get(owner)
|
||||
if !ok {
|
||||
// create a new empty remote and set it
|
||||
// as a weak pointer, so that control stays in golang
|
||||
remote = createNewEmptyRemote()
|
||||
remote.weak = true
|
||||
}
|
||||
|
||||
managed := &managedSmartSubtransport{
|
||||
remote: remote,
|
||||
callback: registeredSmartTransport.callback,
|
||||
subtransport: (*C._go_managed_smart_subtransport)(C.calloc(1, C.size_t(unsafe.Sizeof(C._go_managed_smart_subtransport{})))),
|
||||
}
|
||||
managedHandle := pointerHandles.Track(managed)
|
||||
managed.handle = managedHandle
|
||||
managed.subtransport.handle = managedHandle
|
||||
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
ret := C._go_git_transport_smart(out, owner, cbool(registeredSmartTransport.stateless), managed.subtransport)
|
||||
if ret != 0 {
|
||||
pointerHandles.Untrack(managedHandle)
|
||||
}
|
||||
return ret
|
||||
}
|
||||
|
||||
//export smartTransportSubtransportCallback
|
||||
func smartTransportSubtransportCallback(
|
||||
errorMessage **C.char,
|
||||
wrapperPtr *C._go_managed_smart_subtransport,
|
||||
owner *C.git_transport,
|
||||
) C.int {
|
||||
subtransport := pointerHandles.Get(wrapperPtr.handle).(*managedSmartSubtransport)
|
||||
|
||||
underlyingSmartSubtransport, err := subtransport.callback(subtransport.remote, &Transport{ptr: owner})
|
||||
if err != nil {
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
subtransport.underlying = underlyingSmartSubtransport
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
type managedSmartSubtransport struct {
|
||||
owner *C.git_transport
|
||||
callback SmartSubtransportCallback
|
||||
remote *Remote
|
||||
subtransport *C._go_managed_smart_subtransport
|
||||
underlying SmartSubtransport
|
||||
handle unsafe.Pointer
|
||||
currentManagedStream *managedSmartSubtransportStream
|
||||
}
|
||||
|
||||
func getSmartSubtransportInterface(subtransport *C.git_smart_subtransport) *managedSmartSubtransport {
|
||||
wrapperPtr := (*C._go_managed_smart_subtransport)(unsafe.Pointer(subtransport))
|
||||
return pointerHandles.Get(wrapperPtr.handle).(*managedSmartSubtransport)
|
||||
}
|
||||
|
||||
//export smartSubtransportActionCallback
|
||||
func smartSubtransportActionCallback(
|
||||
errorMessage **C.char,
|
||||
out **C.git_smart_subtransport_stream,
|
||||
t *C.git_smart_subtransport,
|
||||
url *C.char,
|
||||
action C.git_smart_service_t,
|
||||
) C.int {
|
||||
subtransport := getSmartSubtransportInterface(t)
|
||||
|
||||
underlyingStream, err := subtransport.underlying.Action(C.GoString(url), SmartServiceAction(action))
|
||||
if err != nil {
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
// It's okay to do strict equality here: we expect both to be identical.
|
||||
if subtransport.currentManagedStream == nil || subtransport.currentManagedStream.underlying != underlyingStream {
|
||||
runtime.LockOSThread()
|
||||
defer runtime.UnlockOSThread()
|
||||
|
||||
stream := (*C._go_managed_smart_subtransport_stream)(C.calloc(1, C.size_t(unsafe.Sizeof(C._go_managed_smart_subtransport_stream{}))))
|
||||
managed := &managedSmartSubtransportStream{
|
||||
underlying: underlyingStream,
|
||||
streamPtr: stream,
|
||||
}
|
||||
managedHandle := pointerHandles.Track(managed)
|
||||
managed.handle = managedHandle
|
||||
stream.handle = managedHandle
|
||||
|
||||
C._go_git_setup_smart_subtransport_stream(stream)
|
||||
|
||||
subtransport.currentManagedStream = managed
|
||||
}
|
||||
|
||||
*out = &subtransport.currentManagedStream.streamPtr.parent
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export smartSubtransportCloseCallback
|
||||
func smartSubtransportCloseCallback(errorMessage **C.char, t *C.git_smart_subtransport) C.int {
|
||||
subtransport := getSmartSubtransportInterface(t)
|
||||
|
||||
subtransport.currentManagedStream = nil
|
||||
|
||||
if subtransport.underlying != nil {
|
||||
err := subtransport.underlying.Close()
|
||||
if err != nil {
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
}
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export smartSubtransportFreeCallback
|
||||
func smartSubtransportFreeCallback(t *C.git_smart_subtransport) {
|
||||
subtransport := getSmartSubtransportInterface(t)
|
||||
|
||||
if subtransport.underlying != nil {
|
||||
subtransport.underlying.Free()
|
||||
subtransport.underlying = nil
|
||||
}
|
||||
pointerHandles.Untrack(subtransport.handle)
|
||||
C.free(unsafe.Pointer(subtransport.subtransport))
|
||||
subtransport.handle = nil
|
||||
subtransport.subtransport = nil
|
||||
}
|
||||
|
||||
type managedSmartSubtransportStream struct {
|
||||
owner *C.git_smart_subtransport_stream
|
||||
streamPtr *C._go_managed_smart_subtransport_stream
|
||||
underlying SmartSubtransportStream
|
||||
handle unsafe.Pointer
|
||||
}
|
||||
|
||||
func getSmartSubtransportStreamInterface(subtransportStream *C.git_smart_subtransport_stream) *managedSmartSubtransportStream {
|
||||
managedSubtransportStream := (*C._go_managed_smart_subtransport_stream)(unsafe.Pointer(subtransportStream))
|
||||
return pointerHandles.Get(managedSubtransportStream.handle).(*managedSmartSubtransportStream)
|
||||
}
|
||||
|
||||
//export smartSubtransportStreamReadCallback
|
||||
func smartSubtransportStreamReadCallback(
|
||||
errorMessage **C.char,
|
||||
s *C.git_smart_subtransport_stream,
|
||||
buffer *C.char,
|
||||
bufSize C.size_t,
|
||||
bytesRead *C.size_t,
|
||||
) C.int {
|
||||
stream := getSmartSubtransportStreamInterface(s)
|
||||
|
||||
var p []byte
|
||||
header := (*reflect.SliceHeader)(unsafe.Pointer(&p))
|
||||
header.Cap = int(bufSize)
|
||||
header.Len = int(bufSize)
|
||||
header.Data = uintptr(unsafe.Pointer(buffer))
|
||||
|
||||
n, err := stream.underlying.Read(p)
|
||||
*bytesRead = C.size_t(n)
|
||||
if n == 0 && err != nil {
|
||||
if err == io.EOF {
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export smartSubtransportStreamWriteCallback
|
||||
func smartSubtransportStreamWriteCallback(
|
||||
errorMessage **C.char,
|
||||
s *C.git_smart_subtransport_stream,
|
||||
buffer *C.char,
|
||||
bufLen C.size_t,
|
||||
) C.int {
|
||||
stream := getSmartSubtransportStreamInterface(s)
|
||||
|
||||
var p []byte
|
||||
header := (*reflect.SliceHeader)(unsafe.Pointer(&p))
|
||||
header.Cap = int(bufLen)
|
||||
header.Len = int(bufLen)
|
||||
header.Data = uintptr(unsafe.Pointer(buffer))
|
||||
|
||||
if _, err := stream.underlying.Write(p); err != nil {
|
||||
return setCallbackError(errorMessage, err)
|
||||
}
|
||||
|
||||
return C.int(ErrorCodeOK)
|
||||
}
|
||||
|
||||
//export smartSubtransportStreamFreeCallback
|
||||
func smartSubtransportStreamFreeCallback(s *C.git_smart_subtransport_stream) {
|
||||
stream := getSmartSubtransportStreamInterface(s)
|
||||
|
||||
stream.underlying.Free()
|
||||
pointerHandles.Untrack(stream.handle)
|
||||
C.free(unsafe.Pointer(stream.streamPtr))
|
||||
stream.handle = nil
|
||||
stream.streamPtr = nil
|
||||
}
|
|
@ -0,0 +1,72 @@
|
|||
package git
|
||||
|
||||
import (
|
||||
"io"
|
||||
"reflect"
|
||||
"testing"
|
||||
)
|
||||
|
||||
type testSmartSubtransport struct {
|
||||
}
|
||||
|
||||
func (t *testSmartSubtransport) Action(url string, action SmartServiceAction) (SmartSubtransportStream, error) {
|
||||
return &testSmartSubtransportStream{}, nil
|
||||
}
|
||||
|
||||
func (t *testSmartSubtransport) Close() error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func (t *testSmartSubtransport) Free() {
|
||||
}
|
||||
|
||||
type testSmartSubtransportStream struct {
|
||||
}
|
||||
|
||||
func (s *testSmartSubtransportStream) Read(buf []byte) (int, error) {
|
||||
payload := "" +
|
||||
"001e# service=git-upload-pack\n" +
|
||||
"0000005d0000000000000000000000000000000000000000 HEAD\x00symref=HEAD:refs/heads/master agent=libgit\n" +
|
||||
"003f0000000000000000000000000000000000000000 refs/heads/master\n" +
|
||||
"0000"
|
||||
|
||||
return copy(buf, []byte(payload)), io.EOF
|
||||
}
|
||||
|
||||
func (s *testSmartSubtransportStream) Write(buf []byte) (int, error) {
|
||||
return 0, io.EOF
|
||||
}
|
||||
|
||||
func (s *testSmartSubtransportStream) Free() {
|
||||
}
|
||||
|
||||
func TestTransport(t *testing.T) {
|
||||
t.Parallel()
|
||||
repo := createTestRepo(t)
|
||||
defer cleanupTestRepo(t, repo)
|
||||
|
||||
callback := func(remote *Remote, transport *Transport) (SmartSubtransport, error) {
|
||||
return &testSmartSubtransport{}, nil
|
||||
}
|
||||
registeredSmartTransport, err := NewRegisteredSmartTransport("foo", true, callback)
|
||||
checkFatal(t, err)
|
||||
defer registeredSmartTransport.Free()
|
||||
|
||||
remote, err := repo.Remotes.Create("test", "foo://bar")
|
||||
checkFatal(t, err)
|
||||
defer remote.Free()
|
||||
|
||||
err = remote.ConnectFetch(nil, nil, nil)
|
||||
checkFatal(t, err)
|
||||
|
||||
remoteHeads, err := remote.Ls()
|
||||
checkFatal(t, err)
|
||||
|
||||
expectedRemoteHeads := []RemoteHead{
|
||||
{&Oid{}, "HEAD"},
|
||||
{&Oid{}, "refs/heads/master"},
|
||||
}
|
||||
if !reflect.DeepEqual(expectedRemoteHeads, remoteHeads) {
|
||||
t.Errorf("mismatched remote heads. expected %v, got %v", expectedRemoteHeads, remoteHeads)
|
||||
}
|
||||
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue