├── .gitignore ├── go.mod ├── config └── conf.go ├── generate └── generate.go ├── go.sum ├── main.go ├── README-zh-cn.md ├── README.md ├── core ├── file.go └── net.go └── LICENSE /.gitignore: -------------------------------------------------------------------------------- 1 | /.idea/ 2 | .DS_Store 3 | */.DS_Store 4 | -------------------------------------------------------------------------------- /go.mod: -------------------------------------------------------------------------------- 1 | module Magic-Stick-Creator 2 | 3 | go 1.14 4 | 5 | require ( 6 | github.com/mattn/go-runewidth v0.0.9 // indirect 7 | golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd // indirect 8 | gopkg.in/cheggaaa/pb.v1 v1.0.28 9 | howett.net/plist v0.0.0-20200225050739-77e249a2e2ba 10 | ) 11 | -------------------------------------------------------------------------------- /config/conf.go: -------------------------------------------------------------------------------- 1 | package config 2 | 3 | type Vertion struct { 4 | BID string 5 | SN string 6 | Vs string 7 | } 8 | 9 | //Vs:OS name 10 | var Path = "com.apple.recovery.boot" 11 | var Select = []Vertion{Default, Last, Catelina, Mojave, High_Sierra} 12 | var Last = Vertion{BID: "Mac-E43C1C25D4880AD6", SN: "00000000000000000", Vs: "Last Version(最新版)"} 13 | var Default = Vertion{BID: "Mac-E43C1C25D4880AD6", SN: "00000000000GDVQ00", Vs: "Default(默认)"} 14 | var Catelina = Vertion{BID: "Mac-00BE6ED71E35EB8", SN: "00000000000000000", Vs: "Catelina"} 15 | var Mojave = Vertion{BID: "Mac-7BA5B2DFE22DDD8C", SN: "00000000000KXPG00", Vs: "Mojave"} 16 | var High_Sierra = Vertion{BID: "Mac-7BA5B2D9E42DDD94", SN: "00000000000J80300", Vs: "High Sierra"} 17 | -------------------------------------------------------------------------------- /generate/generate.go: -------------------------------------------------------------------------------- 1 | package generate 2 | 3 | import ( 4 | "Magic-Stick-Creator/config" 5 | "Magic-Stick-Creator/core" 6 | "fmt" 7 | "math/rand" 8 | "strings" 9 | "time" 10 | ) 11 | func Genedate_id(x int)string{ 12 | valid_chars := []byte{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F'} 13 | id:=[]byte{} 14 | rand.Seed(time.Now().UnixNano()) 15 | for i := 0; i < x; i++ { 16 | id=append(id,valid_chars[rand.Intn(len(valid_chars))]) 17 | } 18 | return string(id) 19 | } 20 | func Get_image_info(ver config.Vertion)map[string]string{ 21 | url := "http://osrecovery.apple.com/InstallationPayload/RecoveryImage" 22 | method := "POST" 23 | version:="cid="+Genedate_id(16)+"\n"+"k="+Genedate_id(64)+"\n"+"sn="+ver.SN+"\n"+"fg="+Genedate_id(64)+"\n"+"os=default\n"+"bid="+ver.BID 24 | payload := strings.NewReader(version) 25 | return core.HttpHandle(url,method, map[string]string{"Cookie":Get_session()},2,payload).(map[string]string) 26 | } 27 | func Get_session()string{ 28 | return fmt.Sprintf("%s",core.HttpHandle("http://osrecovery.apple.com/","GET",map[string]string{},1,nil)) 29 | } -------------------------------------------------------------------------------- /go.sum: -------------------------------------------------------------------------------- 1 | github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= 2 | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= 3 | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= 4 | github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= 5 | github.com/mattn/go-runewidth v0.0.9 h1:Lm995f3rfxdpd6TSmuVCHVb/QhupuXlYr8sCI/QdE+0= 6 | github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= 7 | golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd h1:xhmwyvizuTgC2qz7ZlMluP20uW+C3Rm0FD/WLDX8884= 8 | golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= 9 | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= 10 | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= 11 | gopkg.in/cheggaaa/pb.v1 v1.0.28 h1:n1tBJnnK2r7g9OW2btFH91V92STTUevLXYFb8gy9EMk= 12 | gopkg.in/cheggaaa/pb.v1 v1.0.28/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= 13 | gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= 14 | howett.net/plist v0.0.0-20200225050739-77e249a2e2ba h1:HiEs/6jQFMHpFqsdPBAk3ieVcsSS8IV+D93f43UuDPo= 15 | howett.net/plist v0.0.0-20200225050739-77e249a2e2ba/go.mod h1:vMygbs4qMhSZSc4lCUl2OEE+rDiIIJAIdR4m7MiMcm0= 16 | -------------------------------------------------------------------------------- /main.go: -------------------------------------------------------------------------------- 1 | package main 2 | 3 | import ( 4 | "Magic-Stick-Creator/config" 5 | "Magic-Stick-Creator/core" 6 | "Magic-Stick-Creator/generate" 7 | "fmt" 8 | "os" 9 | "time" 10 | ) 11 | 12 | func main() { 13 | var action string 14 | var version int = 1 15 | if core.Exists("config.plist") { 16 | var mod int 17 | var ac string 18 | fmt.Printf("监测到存在plist,是否进行三码迁移(若plist中没有3码谨慎操作)[Y/n]") 19 | fmt.Scan(&ac) 20 | if ac == "y" || ac == "Y" { 21 | fmt.Println("请选择你的plist版本") 22 | fmt.Println("1.Clover") 23 | fmt.Println("2.OC") 24 | fmt.Scan(&mod) 25 | if mod <= 0 || mod > 2 { 26 | fmt.Println("选择错误") 27 | } else { 28 | core.ChangeSN(mod) 29 | } 30 | } 31 | fmt.Println("退出三码三码转移") 32 | } 33 | fmt.Println("请选择你需要下载镜像的版本") 34 | for i, v := range config.Select { 35 | fmt.Println(i, v.Vs) 36 | } 37 | fmt.Scan(&version) 38 | if version < 0 || version > len(config.Select)-1 { 39 | panic("版本选择错误") 40 | } else { 41 | fmt.Printf("需要先下载约500MB的Recovery镜像,是否开始? [Y/n]") 42 | fmt.Scan(&action) 43 | if action == "Y" || action == "y" { 44 | if core.Exists(config.Path) { 45 | os.Remove(config.Path) 46 | } 47 | os.Mkdir(config.Path, 0755) 48 | fmt.Println("选择了" + config.Select[version].Vs + "版本...") 49 | core.Download_image(generate.Get_image_info(config.Select[version])) 50 | Add_Details() 51 | fmt.Println("镜像下载完成") 52 | } else { 53 | fmt.Println("中止安装") 54 | return 55 | } 56 | } 57 | fmt.Println("将在3秒后自动退出") 58 | time.Sleep(3 * 1e9) 59 | } 60 | func Add_Details() { 61 | fileName := ".contentDetails" 62 | if core.Exists(fileName) { 63 | err := os.Remove(fileName) 64 | if err != nil { 65 | panic(err) 66 | } 67 | } 68 | dstFile, err := os.Create(config.Path + "/" + fileName) 69 | if err != nil { 70 | fmt.Println(err.Error()) 71 | return 72 | } 73 | defer dstFile.Close() 74 | s := "macOS Boot From Recovery" 75 | dstFile.WriteString(s) 76 | } 77 | -------------------------------------------------------------------------------- /README-zh-cn.md: -------------------------------------------------------------------------------- 1 | ![Mar-29-2020 00-13-05.gif](https://i.loli.net/2020/03/29/I1kQhAZyH9NYiPs.gif) 2 | 3 | [English](https://github.com/Max-Cheng/Magic-Stick-Creator/blob/master/README.md)/[简体中文](https://github.com/Max-Cheng/Magic-Stick-Creator/blob/master/README-zh-cn.md) 4 | 5 | [![GitHub license](https://img.shields.io/github/license/Max-Cheng/Magic-Stick-Creator)](https://github.com/Max-Cheng/Magic-Stick-Creator/blob/master/LICENSE) 6 | 7 | ## Magic-Stick-Creator 8 | 9 | 一个能够自动迁移三码/下载指定版本Recovery image的工具 10 | 11 | ## 快速使用 12 | 13 | 首先你需要准备一个***比较完美的OCEFI***、一个FAT32格式,容量大于1G的U盘. 14 | 15 | ### Windows 16 | 17 | 1. 将EFI拷贝至U盘根目录 18 | 2. 将[本工具](https://github.com/Max-Cheng/Magic-Stick-Creator/releases)拷贝至U盘根目录 19 | 3. 双击Magic-Stick-Creator.exe 20 | 4. 完成下载后即可使用 21 | 22 | ### Mac OS 23 | 24 | 1. 将EFI拷贝至U盘根目录 25 | 26 | 2. 将[本工具](https://github.com/Max-Cheng/Magic-Stick-Creator/releases)拷贝至U盘根目录 27 | 28 | 3. 打开终端 29 | 30 | ```bash 31 | cd /Volumes/你的U盘名称 32 | ./Magic-Stick-Creator 33 | ``` 34 | 35 | 4. 完成下载后即可使用 36 | 37 | ## 功能介绍 38 | 39 | ### 三码迁移 40 | 41 | 首先你需要将带有三码的plist移动至U盘根目录,再执行二进制文件(**注意**:不要使用没有三码的plist进行转移) 42 | 43 | ### 下载指定镜像 44 | 45 | 在根目录执行二进制文件后选择对应版本即可 46 | 47 | ### 自动生成三码 48 | 49 | Coming soon.... 50 | 51 | ## 关于反馈 52 | 53 | 你可以通过提交issues来提交Bug/功能,我会尽我所能去实现! 54 | 55 | ## ToDo List 56 | 57 | - [x] 三码读取 58 | - [x] 三码迁移 59 | - [ ] 自动生成三码 60 | - [ ] EFI在线获取(这方面可能要思考一段时间找出最优解,欢迎👏在issues提出你的想法) 61 | - [ ] 自动寻址 62 | 63 | ## Thanks For 64 | 65 | 排名不分前后 66 | 67 | [acidanthera](https://github.com/acidanthera):提供了思路 68 | 69 | [weachy](https://www.jianshu.com/u/82ec04331356):技术指导 70 | 71 | [DHowett](https://github.com/DHowett):plist文件解析 72 | 73 | ## 免责声明 74 | 75 | 本项目完全开源,免费,仅供技术学习和交流,**开发者团队并未授权任何组织、机构以及个人将其用于商业或者盈利性质的活动。也从未使用本项目进行任何盈利性活动。未来也不会将其用于开展营利性业务。个人或者组织,机构如果使用本项目产生的各类纠纷,法律问题,均由其本人承担。** 76 | 77 | 如果您开始使用本项目,即视为同意项目免责声明中的一切条款,条款更新不再另行通知.开发者仅接受和捐赠者之间不构成购买或雇佣关系的捐赠或者赞赏.如果您选择捐赠,那么我将视之为您完全自愿的,没有任何雇佣,购买关系的捐赠。 -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ![Mar-29-2020 00-13-05.gif](https://i.loli.net/2020/03/29/I1kQhAZyH9NYiPs.gif) 2 | 3 | [English](https://github.com/Max-Cheng/Magic-Stick-Creator/blob/master/README.md)/[简体中文](https://github.com/Max-Cheng/Magic-Stick-Creator/blob/master/README-zh-cn.md) 4 | 5 | [![GitHub license](https://img.shields.io/github/license/Max-Cheng/Magic-Stick-Creator)](https://github.com/Max-Cheng/Magic-Stick-Creator/blob/master/LICENSE) 6 | 7 | ## Magic-Stick-Creator 8 | 9 | A tool of download Mac Recovery Image for OpenCore User/Creater. 10 | 11 | ## Quick Start 12 | 13 | You need to repair a OpenCore EFI(**MUST CAN WORK**) 14 | 15 | ### Windows 16 | 17 | 1. Copy `EFI` to USB flash disk root 18 | 2. Copy [this tool](https://github.com/Max-Cheng/Magic-Stick-Creator/releases) to USB flash disk root 19 | 3. Click Magic-Stick-Creator.exe 20 | 4. You can use it after downloading 21 | 22 | ### Mac OS 23 | 24 | 1. Copy `EFI` to USB flash disk root 25 | 26 | 2. Copy [this tool](https://github.com/Max-Cheng/Magic-Stick-Creator/releases) to USB flash disk root 27 | 28 | 3. 打开终端 29 | 30 | ```bash 31 | cd /Volumes/your-volume-name 32 | ./Magic-Stick-Creator 33 | ``` 34 | 35 | 4. You can use it after downloading 36 | 37 | ## Feedback 38 | 39 | You can submit bugs/features by submitting issues, and I will do my best to implement them! 40 | 41 | ## ToDo List 42 | 43 | - [x] Read SN/SmUUID/MLB 44 | - [x] Transform SN/SmUUID/MLB 45 | - [ ] Generate SN/SmUUID/MLB 46 | - [ ] EFI download online 47 | - [ ] Automatic Update(Online EFI download?) 48 | 49 | ## Thanks For 50 | 51 | No ranking 52 | 53 | [acidanthera](https://github.com/acidanthera):Provides ideas. 54 | 55 | [weachy](https://www.jianshu.com/u/82ec04331356):Technical support 56 | 57 | [DHowett](https://github.com/DHowett):plist file parse 58 | 59 | ## Disclaimer 60 | 61 | This project is completely open source, free, and only for technical learning and communication. The **developer team does not authorize any organization, institution, or individual to use it for commercial or profitable activities. Never used this project for any profitable activities. It will not be used for profit-making business in the future. All kinds of disputes and legal issues arising from the use of this project by individuals or organizations or institutions shall be borne by themselves.** If you start using this project, you are deemed to agree to all the terms in the disclaimer of the project, and no further notice will be given to update the terms. The developer only accepts donations or appreciations that do not constitute a purchase or employment relationship with the donor. If you choose to donate , Then I will treat it as a completely voluntary donation without any employment or purchase relationship. 62 | -------------------------------------------------------------------------------- /core/file.go: -------------------------------------------------------------------------------- 1 | package core 2 | 3 | import ( 4 | "fmt" 5 | "howett.net/plist" 6 | "os" 7 | ) 8 | 9 | func Exists(path string) bool { 10 | _, err := os.Stat(path) //os.Stat获取文件信息 11 | if err != nil { 12 | if os.IsExist(err) { 13 | return true 14 | } 15 | return false 16 | } 17 | return true 18 | } 19 | func ChangeSN(mod int){ 20 | //1:Clover to OC. 21 | //2:OC to OC 22 | //Read User config.plist 23 | Target,err:=os.Open("clover.plist") 24 | if err != nil { 25 | panic(err) 26 | } 27 | var Targetdata map[string]interface{} 28 | decoder := plist.NewDecoder(Target) 29 | derr := decoder.Decode(&Targetdata) 30 | if derr != nil { 31 | fmt.Println(derr) 32 | } 33 | //Read EFI plist 34 | OC,err:=os.Open("EFI/OC/config.plist") 35 | if err != nil { 36 | panic(err) 37 | } 38 | var OCdata map[string]interface{} 39 | EFIdecoder := plist.NewDecoder(OC) 40 | EFIderr := EFIdecoder.Decode(&OCdata) 41 | if EFIderr != nil { 42 | fmt.Println(derr) 43 | } 44 | //Create New config.plist 45 | newfile,err:=os.Create("EFI/OC/config.plist") 46 | if err!=nil { 47 | panic(err) 48 | } 49 | //New Encoder 50 | a:=plist.NewEncoder(newfile) 51 | a.Indent(" ") 52 | switch mod { 53 | case 1: 54 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["MLB"]=Targetdata["SMBIOS"].(map[string]interface{})["BoardSerialNumber"] 55 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["ROM"]=Targetdata["RtVariables"].(map[string]interface{})["ROM"] 56 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["SystemSerialNumber"]=Targetdata["SMBIOS"].(map[string]interface{})["SerialNumber"] 57 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["SystemUUID"]=Targetdata["SystemParameters"].(map[string]interface{})["CustomUUID"] 58 | a.Encode(OCdata) 59 | break 60 | case 2: 61 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["MLB"]=Targetdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["MLB"] 62 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["ROM"]=Targetdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["ROM"] 63 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["SystemSerialNumber"]=Targetdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["SystemSerialNumber"] 64 | OCdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["SystemUUID"]=Targetdata["PlatformInfo"].(map[string]interface{})["Generic"].(map[string]interface{})["SystemUUID"] 65 | a.Encode(OCdata) 66 | break; 67 | default:fmt.Println("选择错误");break; 68 | } 69 | } 70 | -------------------------------------------------------------------------------- /core/net.go: -------------------------------------------------------------------------------- 1 | package core 2 | 3 | import ( 4 | "Magic-Stick-Creator/config" 5 | "fmt" 6 | "gopkg.in/cheggaaa/pb.v1" 7 | _ "gopkg.in/cheggaaa/pb.v1" 8 | "io" 9 | "io/ioutil" 10 | "net/http" 11 | "os" 12 | "path" 13 | "strconv" 14 | "strings" 15 | "time" 16 | ) 17 | func Download_image(UrlAndCookie map[string]string){ 18 | HttpHandle(UrlAndCookie["CU"],"GET",map[string]string{"Cookie":"AssetToken="+UrlAndCookie["CT"]},3,nil) 19 | HttpHandle(UrlAndCookie["AU"],"GET", map[string]string{"Cookie":"AssetToken="+UrlAndCookie["AT"]},3,nil) 20 | } 21 | 22 | func HttpHandle(url,method string,cookie map[string]string,mod int,Body io.Reader) interface{}{ 23 | //get_session mod:1,get_image_info:2,downloadmod:3 24 | client:=&http.Client{} 25 | reqest, err := http.NewRequest(method, url, Body) 26 | if err != nil { 27 | fmt.Println(err) 28 | } 29 | if len(cookie)!=0 { 30 | for i,v:=range cookie{ 31 | reqest.Header.Add(i,v) 32 | } 33 | } 34 | reqest.Header.Add("Host", "osrecovery.apple.com") 35 | reqest.Header.Add("Connection", "close") 36 | reqest.Header.Add("User-Agent", "InternetRecovery/1.0") 37 | reqest.Header.Add("Content-Type", "text/plain") 38 | reqest.Header.Add("Connection", "close") 39 | reqest.Header.Add("Accept", "*/*") 40 | reqest.Header.Add("Cache-Control", "no-cache") 41 | reqest.Header.Add("Postman-Token", "d375c5bb-5a92-4a8b-8f9c-41083df18781") 42 | reqest.Header.Add("Accept-Encoding", "gzip, deflate, br") 43 | reqest.Header.Add("Content-Length", "212") 44 | if err != nil { 45 | panic(err) 46 | } 47 | response, err := client.Do(reqest) 48 | if err != nil { 49 | panic(err) 50 | } 51 | defer response.Body.Close() 52 | switch mod { 53 | case 1: 54 | return response.Header.Get("Set-Cookie")[:83];break 55 | case 2: 56 | body, err := ioutil.ReadAll(response.Body) 57 | if err != nil { 58 | panic(err) 59 | } 60 | newbody:=(strings.Split(string(body),"\n")) 61 | result:= map[string]string{} 62 | for i:=0;i