Files
i2p.plugins.tor-updater/supervise/supervise.go

431 lines
11 KiB
Go
Raw Normal View History

2022-01-16 21:23:00 -05:00
package tbsupervise
import (
2022-01-23 12:00:37 -05:00
"embed"
2022-01-16 22:04:35 -05:00
"fmt"
2022-01-23 12:00:37 -05:00
"io/fs"
"io/ioutil"
2022-01-16 22:04:35 -05:00
"log"
2022-01-22 01:13:30 -05:00
"net"
2022-01-16 21:23:00 -05:00
"os"
"os/exec"
"path"
2022-01-16 21:23:00 -05:00
"path/filepath"
"strings"
2022-01-16 21:23:00 -05:00
"github.com/mitchellh/go-ps"
cp "github.com/otiai10/copy"
2022-01-16 21:23:00 -05:00
tbget "i2pgit.org/idk/i2p.plugins.tor-manager/get"
)
var UNPACK_URL = tbget.UNPACK_PATH
var DEFAULT_TB_LANG = tbget.DefaultIETFLang
2022-01-16 21:23:00 -05:00
func OS() string {
return tbget.OS
}
func ARCH() string {
return tbget.ARCH
}
2022-01-16 21:23:00 -05:00
type Supervisor struct {
2022-02-06 16:57:30 -05:00
UnpackPath string
Lang string
torcmd *exec.Cmd
tbcmd *exec.Cmd
ibcmd *exec.Cmd
Profile *embed.FS
PassThroughArgs []string
}
func (s *Supervisor) PTAS() []string {
// loop over the arguments and make sure that we remove any --profile, -P args
// and blank them out.
var args []string
for index, arg := range s.PassThroughArgs {
if arg == "--profile" || arg == "-P" {
continue
}
if index > 0 {
if s.PassThroughArgs[index-1] == "--profile" || s.PassThroughArgs[index-1] == "-P" {
continue
}
}
args = append(args, arg)
}
return args
}
func (s *Supervisor) TBPath() string {
return filepath.Join(s.UnpackPath, "Browser", "start-tor-browser")
}
func (s *Supervisor) FirefoxPath() string {
switch OS() {
2022-01-23 14:45:12 -05:00
case "linux":
return filepath.Join(s.UnpackPath, "Browser", "firefox.real")
case "windows":
return filepath.Join(s.UnpackPath, "Browser", "firefox.exe")
default:
return filepath.Join(s.UnpackPath, "Browser", "firefox")
}
}
func (s *Supervisor) TBDirectory() string {
return filepath.Join(s.UnpackPath, "Browser")
}
func (s *Supervisor) TorPath() string {
return filepath.Join(s.UnpackPath, "Browser", "TorBrowser", "Tor", "tor")
}
func (s *Supervisor) TorDataPath() string {
return filepath.Join(s.UnpackPath, "Browser", "TorBrowser", "Data")
}
func (s *Supervisor) I2PProfilePath() string {
fp := filepath.Join(filepath.Dir(s.UnpackPath), ".i2p.firefox")
if !tbget.FileExists(fp) {
log.Printf("i2p data not found at %s, unpacking", fp)
if s.Profile != nil {
if err := s.UnpackI2PData(); err != nil {
log.Fatal(err)
}
}
}
return fp
}
func (s *Supervisor) I2PDataPath() string {
fp := s.I2PProfilePath()
up := filepath.Join(filepath.Dir(s.UnpackPath), "i2p.firefox")
if tbget.FileExists(up) {
return up
2022-01-23 12:00:37 -05:00
} else {
log.Printf("i2p workdir not found at %s, copying", up)
2022-01-23 12:00:37 -05:00
if s.Profile != nil {
if err := cp.Copy(fp, up); err != nil {
2022-01-23 12:00:37 -05:00
log.Fatal(err)
}
}
return up
2022-01-23 12:00:37 -05:00
}
}
func (s *Supervisor) UnpackI2PData() error {
return fs.WalkDir(s.Profile, ".", func(embedpath string, d fs.DirEntry, err error) error {
fp := filepath.Join(filepath.Dir(s.UnpackPath), ".i2p.firefox")
2022-01-23 12:00:37 -05:00
if err != nil {
log.Fatal(err)
2022-01-23 12:00:37 -05:00
}
fmt.Println(embedpath, filepath.Join(fp, strings.Replace(embedpath, "tor-browser/unpack/i2p.firefox", "", -1)))
if d.IsDir() {
os.MkdirAll(filepath.Join(fp, strings.Replace(embedpath, "tor-browser/unpack/i2p.firefox", "", -1)), 0755)
2022-01-23 12:00:37 -05:00
} else {
fullpath := path.Join(embedpath)
bytes, err := s.Profile.ReadFile(fullpath)
if err != nil {
2022-01-23 12:00:37 -05:00
return err
}
unpack := filepath.Join(fp, strings.Replace(embedpath, "tor-browser/unpack/i2p.firefox", "", -1))
if err := ioutil.WriteFile(unpack, bytes, 0644); err != nil {
return err
}
}
2022-01-23 12:00:37 -05:00
return nil
})
}
2022-02-06 16:57:30 -05:00
func (s *Supervisor) I2PAppDataPath() string {
fp := s.I2PProfilePath()
up := filepath.Join(filepath.Dir(s.UnpackPath), "i2p.firefox.config")
if tbget.FileExists(up) {
return up
} else {
log.Printf("i2p workdir not found at %s, copying", up)
if s.Profile != nil {
if err := cp.Copy(fp, up); err != nil {
log.Fatal(err)
}
}
return up
}
}
func (s *Supervisor) UnpackI2PAppData() error {
return fs.WalkDir(s.Profile, ".", func(embedpath string, d fs.DirEntry, err error) error {
fp := filepath.Join(filepath.Dir(s.UnpackPath), ".i2p.firefox.config")
if err != nil {
log.Fatal(err)
}
fmt.Println(embedpath, filepath.Join(fp, strings.Replace(embedpath, "tor-browser/unpack/i2p.firefox.config", "", -1)))
if d.IsDir() {
os.MkdirAll(filepath.Join(fp, strings.Replace(embedpath, "tor-browser/unpack/i2p.firefox.config", "", -1)), 0755)
} else {
fullpath := path.Join(embedpath)
bytes, err := s.Profile.ReadFile(fullpath)
if err != nil {
return err
}
unpack := filepath.Join(fp, strings.Replace(embedpath, "tor-browser/unpack/i2p.firefox.config", "", -1))
if err := ioutil.WriteFile(unpack, bytes, 0644); err != nil {
return err
}
}
return nil
})
}
2022-01-22 01:09:18 -05:00
func (s *Supervisor) tbbail() error {
if s.tbcmd != nil && s.tbcmd.Process != nil && s.tbcmd.ProcessState != nil {
if s.tbcmd.ProcessState.Exited() {
return nil
}
return fmt.Errorf("Already running")
}
return nil
}
func (s *Supervisor) RunTBWithLang() error {
tbget.ARCH = ARCH()
if s.Lang == "" {
s.Lang = DEFAULT_TB_LANG
2022-01-16 21:23:00 -05:00
}
if s.UnpackPath == "" {
s.UnpackPath = UNPACK_URL()
}
2022-01-22 01:09:18 -05:00
if s.tbbail() != nil {
return nil
}
log.Println("running tor browser with lang", s.Lang, s.UnpackPath, OS())
switch OS() {
2022-01-16 21:23:00 -05:00
case "linux":
if tbget.FileExists(s.UnpackPath) {
log.Println("running tor browser with lang", s.Lang, s.UnpackPath)
2022-02-06 16:57:30 -05:00
args := []string{}
args = append(args, s.PTAS()...)
s.tbcmd = exec.Command(s.TBPath(), args...)
2022-01-22 01:09:18 -05:00
s.tbcmd.Stdout = os.Stdout
s.tbcmd.Stderr = os.Stderr
return s.tbcmd.Run()
} else {
log.Println("tor browser not found at", s.TBPath())
return fmt.Errorf("tor browser not found at %s", s.TBPath())
}
case "darwin":
2022-01-22 01:09:18 -05:00
s.tbcmd = exec.Command("/usr/bin/env", "open", "-a", "\"Tor Browser.app\"")
s.tbcmd.Dir = s.TBDirectory()
return s.tbcmd.Run()
case "win":
log.Println("Running Windows EXE", s.TBDirectory(), "firefox.exe")
2022-02-06 16:57:30 -05:00
args := []string{}
args = append(args, s.PTAS()...)
s.tbcmd = exec.Command(filepath.Join(s.TBDirectory(), "firefox.exe"), args...)
2022-01-22 01:09:18 -05:00
s.tbcmd.Dir = s.TBDirectory()
return s.tbcmd.Run()
default:
}
return nil
}
2022-01-22 01:09:18 -05:00
func (s *Supervisor) ibbail() error {
if s.ibcmd != nil && s.ibcmd.Process != nil && s.ibcmd.ProcessState != nil {
if s.ibcmd.ProcessState.Exited() {
return nil
}
return fmt.Errorf("Already running")
}
return nil
}
func (s *Supervisor) RunI2PBWithLang() error {
tbget.ARCH = ARCH()
if s.Lang == "" {
s.Lang = DEFAULT_TB_LANG
}
if s.UnpackPath == "" {
s.UnpackPath = UNPACK_URL()
}
2022-01-22 01:09:18 -05:00
if s.ibbail() != nil {
return nil
}
log.Println("running i2p in tor browser with lang", s.Lang, s.UnpackPath, OS())
switch OS() {
case "linux":
if tbget.FileExists(s.UnpackPath) {
log.Println("running Tor browser with lang and I2P Profile", s.Lang, s.UnpackPath, s.FirefoxPath(), "--profile", s.I2PDataPath())
2022-02-06 16:57:30 -05:00
args := []string{"--profile", s.I2PDataPath()}
args = append(args, s.PTAS()...)
s.ibcmd = exec.Command(s.FirefoxPath(), args...)
2022-01-22 01:09:18 -05:00
s.ibcmd.Stdout = os.Stdout
s.ibcmd.Stderr = os.Stderr
return s.ibcmd.Run()
} else {
log.Println("tor browser not found at", s.FirefoxPath())
return fmt.Errorf("tor browser not found at %s", s.FirefoxPath())
}
case "darwin":
2022-01-22 01:09:18 -05:00
s.ibcmd = exec.Command("/usr/bin/env", "open", "-a", "\"Tor Browser.app\"")
s.ibcmd.Dir = s.TBDirectory()
return s.ibcmd.Run()
case "win":
log.Println("Running Windows EXE", filepath.Join(s.TBDirectory(), "firefox.exe"), "--profile", s.I2PDataPath())
2022-02-06 16:57:30 -05:00
args := []string{"--profile", "."}
args = append(args, s.PTAS()...)
s.ibcmd = exec.Command(filepath.Join(s.TBDirectory(), "firefox.exe"), args...)
s.ibcmd.Dir = s.I2PDataPath()
s.ibcmd.Stdout = os.Stdout
s.ibcmd.Stderr = os.Stderr
return s.ibcmd.Run()
default:
}
return nil
}
2022-02-06 16:57:30 -05:00
func (s *Supervisor) RunI2PBAppWithLang() error {
tbget.ARCH = ARCH()
if s.Lang == "" {
s.Lang = DEFAULT_TB_LANG
}
if s.UnpackPath == "" {
s.UnpackPath = UNPACK_URL()
}
if s.ibbail() != nil {
return nil
}
log.Println("running i2p in tor browser with lang", s.Lang, s.UnpackPath, OS())
switch OS() {
case "linux":
if tbget.FileExists(s.UnpackPath) {
log.Println("running Tor browser with lang and I2P Profile", s.Lang, s.UnpackPath, s.FirefoxPath(), "--profile", s.I2PAppDataPath())
args := []string{"--profile", s.I2PAppDataPath()}
args = append(args, s.PTAS()...)
s.ibcmd = exec.Command(s.FirefoxPath(), args...)
s.ibcmd.Stdout = os.Stdout
s.ibcmd.Stderr = os.Stderr
return s.ibcmd.Run()
} else {
log.Println("tor browser not found at", s.FirefoxPath())
return fmt.Errorf("tor browser not found at %s", s.FirefoxPath())
}
case "darwin":
s.ibcmd = exec.Command("/usr/bin/env", "open", "-a", "\"Tor Browser.app\"")
s.ibcmd.Dir = s.TBDirectory()
return s.ibcmd.Run()
case "win":
log.Println("Running Windows EXE", filepath.Join(s.TBDirectory(), "firefox.exe"), "--profile", s.I2PAppDataPath())
args := []string{"--profile", "."}
args = append(args, s.PTAS()...)
s.ibcmd = exec.Command(filepath.Join(s.TBDirectory(), "firefox.exe"), args...)
s.ibcmd.Dir = s.I2PAppDataPath()
s.ibcmd.Stdout = os.Stdout
s.ibcmd.Stderr = os.Stderr
return s.ibcmd.Run()
default:
}
return nil
}
2022-01-22 01:09:18 -05:00
func (s *Supervisor) torbail() error {
_, err := net.Listen("tcp", "127.0.0.1:9050")
2022-01-22 01:13:30 -05:00
if err != nil {
log.Println("Already Running on 9050", err)
2022-01-22 01:13:30 -05:00
return fmt.Errorf("Already running")
}
2022-01-22 01:09:18 -05:00
if s.torcmd != nil && s.torcmd.Process != nil && s.torcmd.ProcessState != nil {
if s.torcmd.ProcessState.Exited() {
log.Println("Tor exited, restarting")
2022-01-22 01:09:18 -05:00
return nil
}
log.Println("Already Running")
2022-01-22 01:09:18 -05:00
return fmt.Errorf("Already running")
}
log.Println("Starting Tor")
2022-01-22 01:09:18 -05:00
return nil
}
func (s *Supervisor) RunTorWithLang() error {
tbget.ARCH = ARCH()
if s.Lang == "" {
s.Lang = DEFAULT_TB_LANG
}
if s.UnpackPath == "" {
s.UnpackPath = UNPACK_URL()
}
2022-01-22 01:09:18 -05:00
if err := s.torbail(); err != nil {
return nil
}
log.Println("running tor with lang", s.Lang, s.UnpackPath)
switch OS() {
case "linux":
if tbget.FileExists(s.UnpackPath) {
log.Println("running tor with lang", s.Lang, s.UnpackPath)
2022-01-22 01:09:18 -05:00
s.torcmd = exec.Command(s.TorPath())
s.torcmd.Stdout = os.Stdout
s.torcmd.Stderr = os.Stderr
return s.torcmd.Run()
2022-01-16 22:04:35 -05:00
} else {
log.Println("tor not found at", s.TorPath())
return fmt.Errorf("tor not found at %s", s.TorPath())
2022-01-16 22:04:35 -05:00
}
2022-01-16 21:23:00 -05:00
case "darwin":
2022-01-22 01:09:18 -05:00
s.torcmd = exec.Command("/usr/bin/env", "open", "-a", "\"Tor Browser.app\"")
s.torcmd.Dir = s.TBDirectory()
return s.torcmd.Run()
case "win":
log.Println("Running Windows EXE", filepath.Join(s.TBDirectory(), "TorBrowser", "Tor", "tor.exe"))
s.torcmd = exec.Command(filepath.Join(s.TBDirectory(), "TorBrowser", "Tor", "tor.exe"))
s.torcmd.Dir = s.TBDirectory()
return s.torcmd.Run()
2022-01-16 21:23:00 -05:00
default:
}
return nil
}
2022-01-22 00:42:10 -05:00
func (s *Supervisor) StopTor() error {
2022-01-22 01:09:18 -05:00
return s.torcmd.Process.Kill()
2022-01-22 00:42:10 -05:00
}
func (s *Supervisor) TorIsAlive() (bool, bool) {
_, err := net.Listen("TCP", "127.0.0.1:9050")
if err != nil {
return true, false
}
if s.torcmd != nil && s.torcmd.Process != nil && s.torcmd.ProcessState != nil {
return !s.torcmd.ProcessState.Exited(), true
}
processes, err := ps.Processes()
if err != nil {
return false, true
}
for _, p := range processes {
if p.Executable() == s.TorPath() {
var err error
s.torcmd.Process, err = os.FindProcess(p.Pid())
if err == nil {
return true, true
}
}
}
return false, true
2022-01-22 22:04:30 -05:00
}
func NewSupervisor(tbPath, lang string) *Supervisor {
return &Supervisor{
UnpackPath: tbPath,
Lang: lang,
}
}