2015-04-13 12:31:41 -06:00
|
|
|
// Copyright 2015 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// +build ignore
|
|
|
|
|
|
|
|
// detect attempts to autodetect the correct
|
|
|
|
// values of the environment variables
|
|
|
|
// used by go_darwin_arm_exec.
|
2015-05-04 10:41:41 -06:00
|
|
|
// detect shells out to ideviceinfo, a third party program that can
|
|
|
|
// be obtained by following the instructions at
|
|
|
|
// https://github.com/libimobiledevice/libimobiledevice.
|
2015-04-13 12:31:41 -06:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
devID := detectDevID()
|
|
|
|
|
|
|
|
udid := detectUDID()
|
2017-02-01 01:37:47 -07:00
|
|
|
mps := detectMobileProvisionFiles(udid)
|
|
|
|
if len(mps) == 0 {
|
|
|
|
fail("did not find mobile provision matching device udid %s", udid)
|
|
|
|
}
|
2015-04-13 12:31:41 -06:00
|
|
|
|
2017-02-01 01:37:47 -07:00
|
|
|
fmt.Println("Available provisioning profiles below.")
|
|
|
|
fmt.Println("NOTE: Any existing app on the device with the app id specified by GOIOS_APP_ID")
|
|
|
|
fmt.Println("will be overwritten when running Go programs.")
|
|
|
|
for _, mp := range mps {
|
|
|
|
fmt.Println()
|
|
|
|
fmt.Printf("export GOIOS_DEV_ID=%s\n", devID)
|
|
|
|
f, err := ioutil.TempFile("", "go_ios_detect_")
|
|
|
|
check(err)
|
|
|
|
fname := f.Name()
|
|
|
|
defer os.Remove(fname)
|
|
|
|
|
|
|
|
out := output(parseMobileProvision(mp))
|
|
|
|
_, err = f.Write(out)
|
|
|
|
check(err)
|
|
|
|
check(f.Close())
|
|
|
|
|
|
|
|
appID, err := plistExtract(fname, "Entitlements:application-identifier")
|
|
|
|
check(err)
|
|
|
|
fmt.Printf("export GOIOS_APP_ID=%s\n", appID)
|
|
|
|
|
|
|
|
teamID, err := plistExtract(fname, "Entitlements:com.apple.developer.team-identifier")
|
|
|
|
check(err)
|
|
|
|
fmt.Printf("export GOIOS_TEAM_ID=%s\n", teamID)
|
|
|
|
}
|
2015-04-13 12:31:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
func detectDevID() string {
|
|
|
|
cmd := exec.Command("security", "find-identity", "-p", "codesigning", "-v")
|
|
|
|
lines := getLines(cmd)
|
|
|
|
|
|
|
|
for _, line := range lines {
|
|
|
|
if !bytes.Contains(line, []byte("iPhone Developer")) {
|
|
|
|
continue
|
|
|
|
}
|
2015-09-17 10:13:53 -06:00
|
|
|
if bytes.Contains(line, []byte("REVOKED")) {
|
|
|
|
continue
|
|
|
|
}
|
2015-04-13 12:31:41 -06:00
|
|
|
fields := bytes.Fields(line)
|
|
|
|
return string(fields[1])
|
|
|
|
}
|
|
|
|
fail("no code signing identity found")
|
|
|
|
panic("unreachable")
|
|
|
|
}
|
|
|
|
|
|
|
|
var udidPrefix = []byte("UniqueDeviceID: ")
|
|
|
|
|
|
|
|
func detectUDID() []byte {
|
|
|
|
cmd := exec.Command("ideviceinfo")
|
|
|
|
lines := getLines(cmd)
|
|
|
|
for _, line := range lines {
|
|
|
|
if bytes.HasPrefix(line, udidPrefix) {
|
|
|
|
return bytes.TrimPrefix(line, udidPrefix)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fail("udid not found; is the device connected?")
|
|
|
|
panic("unreachable")
|
|
|
|
}
|
|
|
|
|
2017-02-01 01:37:47 -07:00
|
|
|
func detectMobileProvisionFiles(udid []byte) []string {
|
2015-04-13 12:31:41 -06:00
|
|
|
cmd := exec.Command("mdfind", "-name", ".mobileprovision")
|
|
|
|
lines := getLines(cmd)
|
|
|
|
|
2017-02-01 01:37:47 -07:00
|
|
|
var files []string
|
2015-04-13 12:31:41 -06:00
|
|
|
for _, line := range lines {
|
|
|
|
if len(line) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
xmlLines := getLines(parseMobileProvision(string(line)))
|
|
|
|
for _, xmlLine := range xmlLines {
|
|
|
|
if bytes.Contains(xmlLine, udid) {
|
2017-02-01 01:37:47 -07:00
|
|
|
files = append(files, string(line))
|
2015-04-13 12:31:41 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-02-01 01:37:47 -07:00
|
|
|
return files
|
2015-04-13 12:31:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
func parseMobileProvision(fname string) *exec.Cmd {
|
|
|
|
return exec.Command("security", "cms", "-D", "-i", string(fname))
|
|
|
|
}
|
|
|
|
|
|
|
|
func plistExtract(fname string, path string) ([]byte, error) {
|
|
|
|
out, err := exec.Command("/usr/libexec/PlistBuddy", "-c", "Print "+path, fname).CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return bytes.TrimSpace(out), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getLines(cmd *exec.Cmd) [][]byte {
|
2017-02-01 00:57:46 -07:00
|
|
|
out := output(cmd)
|
2015-04-13 12:31:41 -06:00
|
|
|
return bytes.Split(out, []byte("\n"))
|
|
|
|
}
|
|
|
|
|
2017-02-01 00:57:46 -07:00
|
|
|
func output(cmd *exec.Cmd) []byte {
|
|
|
|
out, err := cmd.Output()
|
2015-04-13 12:31:41 -06:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(strings.Join(cmd.Args, "\n"))
|
|
|
|
fmt.Fprintln(os.Stderr, err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
|
|
|
func check(err error) {
|
|
|
|
if err != nil {
|
|
|
|
fail(err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func fail(msg string, v ...interface{}) {
|
|
|
|
fmt.Fprintf(os.Stderr, msg, v...)
|
|
|
|
fmt.Fprintln(os.Stderr)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|