Compare commits
No commits in common. "2c3b4b18f2aa8f4fea533c80e85ddb18060514e4" and "bce37e654153c1c0257494df3f063653616b4fc1" have entirely different histories.
2c3b4b18f2
...
bce37e6541
@ -12,7 +12,7 @@ import (
|
|||||||
func ShellJail(args []string) error {
|
func ShellJail(args []string) error {
|
||||||
// We cant shell more than one jail bc we replace gocage execution with jexec, so there wont be no return to gocage
|
// We cant shell more than one jail bc we replace gocage execution with jexec, so there wont be no return to gocage
|
||||||
if len(args) > 0 {
|
if len(args) > 0 {
|
||||||
cj, err := getJailFromArray(args[0], []string{"jail"}, gJails)
|
cj, err := getJailFromArray(args[0], gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting jail %s: %v\n", args[0], err)
|
fmt.Printf("Error getting jail %s: %v\n", args[0], err)
|
||||||
return err
|
return err
|
||||||
|
196
cmd/create.go
196
cmd/create.go
@ -1,196 +0,0 @@
|
|||||||
package cmd
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
"fmt"
|
|
||||||
//"log"
|
|
||||||
"time"
|
|
||||||
"strings"
|
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
)
|
|
||||||
|
|
||||||
// TODO : Add a flag to specify which parts of freebsd base we want : Slim jail only need base.txz, neither lib32 nor src.txz
|
|
||||||
func CreateJail(args []string) {
|
|
||||||
var err error
|
|
||||||
var jtype []string
|
|
||||||
|
|
||||||
if len(gCreateArgs.JailType) > 0 {
|
|
||||||
jtype = []string{gCreateArgs.JailType}
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, jname := range args {
|
|
||||||
// Check if jail exist and is distinctly named
|
|
||||||
_, err = getJailFromArray(jname, jtype, gJails)
|
|
||||||
if err != nil {
|
|
||||||
if strings.EqualFold(err.Error(), "Jail not found") {
|
|
||||||
|
|
||||||
} else {
|
|
||||||
fmt.Printf("ERROR: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
fmt.Printf("Jail exist: %s\n", jname)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Printf(" > create jail %s\n", jname)
|
|
||||||
|
|
||||||
var ds *Datastore
|
|
||||||
if len(gCreateArgs.Datastore) > 0 {
|
|
||||||
fmt.Printf("DEBUG: Use %s datastore\n", gCreateArgs.Datastore)
|
|
||||||
ds, err = getDatastoreFromArray(gCreateArgs.Datastore, gDatastores)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR Getting datastore: %s\n", gCreateArgs.Datastore, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
ds = &gDatastores[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
// Create and populate datasets
|
|
||||||
err = zfsCreateDataset(fmt.Sprintf("%s/jails/%s", ds.ZFSDataset, jname), fmt.Sprintf("%s/jails/%s", ds.Mountpoint, jname), "lz4")
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
err = zfsCreateDataset(fmt.Sprintf("%s/jails/%s/root", ds.ZFSDataset, jname), fmt.Sprintf("%s/jails/%s/root", ds.Mountpoint, jname), "lz4")
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Get base template if specified
|
|
||||||
if len(gCreateArgs.BaseTemplate) > 0 {
|
|
||||||
log.Debugf("Jail will be created from a base template\n")
|
|
||||||
/*bj, err := getJailFromArray(jname, []string{"template"}, gJails)
|
|
||||||
if err != nil {
|
|
||||||
if strings.EqualFold(err.Error(), "Jail not found") {
|
|
||||||
|
|
||||||
} else {
|
|
||||||
fmt.Printf("ERROR: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
fmt.Printf("Jail exist: %s\n", jname)
|
|
||||||
continue
|
|
||||||
}*/
|
|
||||||
} else {
|
|
||||||
// Normal jail with its own freebsd base
|
|
||||||
log.Debugf("Creating jail with its own freebsd base\n")
|
|
||||||
|
|
||||||
// First check if we got release on the same datastore
|
|
||||||
_, err := os.Stat(fmt.Sprintf("%s/releases/%s/root", ds.Mountpoint, gCreateArgs.Release))
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
fmt.Printf("ERROR: Release locally not available. Run \"gocage fetch\"\n")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// Create and populate jail filesystem
|
|
||||||
dstDset := fmt.Sprintf("%s/jails/%s", ds.ZFSDataset, jname)
|
|
||||||
fmt.Printf(" > Initialize dataset %s\n", dstDset)
|
|
||||||
sNow := time.Now().Format("20060102150405")
|
|
||||||
reldset := fmt.Sprintf("%s/releases/%s", ds.ZFSDataset, gCreateArgs.Release)
|
|
||||||
err = zfsSnapshot(reldset, sNow)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR Creating snapshot of %s: %s\n", reldset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
err = zfsCopy(fmt.Sprintf("%s@%s", reldset, sNow), dstDset)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR sending snapshot to %s: %s\n", dstDset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Remove snapshot of release, then snapshot of destination dataset
|
|
||||||
err = zfsDestroy(fmt.Sprintf("%s@%s", reldset, sNow))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR destroying snapshot %s: %s\n", reldset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
err = zfsDestroy(fmt.Sprintf("%s@%s", dstDset, sNow))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR destroying snapshot %s: %s\n", dstDset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
dstRootDset := fmt.Sprintf("%s/jails/%s/root", ds.ZFSDataset, jname)
|
|
||||||
fmt.Printf(" > Initialize dataset %s\n", dstRootDset)
|
|
||||||
relrootdset := fmt.Sprintf("%s/releases/%s/root", ds.ZFSDataset, gCreateArgs.Release)
|
|
||||||
err = zfsSnapshot(relrootdset, sNow)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR Creating snapshot of %s: %s\n", relrootdset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
err = zfsCopy(fmt.Sprintf("%s@%s", relrootdset, sNow), dstRootDset)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR sending snapshot to %s: %s\n", dstRootDset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Remove snapshot of release, then snapshot of destination dataset
|
|
||||||
err = zfsDestroy(fmt.Sprintf("%s@%s", relrootdset, sNow))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR destroying snapshot %s: %s\n", relrootdset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
err = zfsDestroy(fmt.Sprintf("%s@%s", dstRootDset, sNow))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR destroying snapshot %s: %s\n", dstRootDset, err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Printf("Jail filesystem successfuly initalized\n")
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// Copy defaults.json...
|
|
||||||
jailConfPath := fmt.Sprintf("%s/jails/%s/config.json", ds.Mountpoint, jname)
|
|
||||||
err = copyFile(fmt.Sprintf("%s/defaults.json", ds.Mountpoint),
|
|
||||||
jailConfPath)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR creating config.json: %s\n", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// ... and update it
|
|
||||||
// Get conf from config.json
|
|
||||||
jailConf, err := getJailConfig(jailConfPath)
|
|
||||||
if err != nil {
|
|
||||||
log.Println("ERROR reading jail config from %s", jailConfPath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. Build jail object from config
|
|
||||||
jailRootPath := fmt.Sprintf("%s/jails/%s/%s", ds.Mountpoint, jname, "root")
|
|
||||||
j := Jail{
|
|
||||||
Name: jailConf.Host_hostuuid,
|
|
||||||
Config: jailConf,
|
|
||||||
ConfigPath: jailConfPath,
|
|
||||||
Datastore: ds.Name,
|
|
||||||
RootPath: jailRootPath,
|
|
||||||
Running: false,
|
|
||||||
}
|
|
||||||
|
|
||||||
j.Config.Release = gCreateArgs.Release
|
|
||||||
j.Config.Host_hostname = jname
|
|
||||||
j.Config.Host_hostuuid = jname
|
|
||||||
|
|
||||||
j.WriteConfigToDisk(false)
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// Create fstab
|
|
||||||
fstabHandle, err := os.Create(fmt.Sprintf("%s/jails/%s/fstab", ds.Mountpoint, jname))
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR creating fstab: %s", err.Error())
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer fstabHandle.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO : Set JailType
|
|
||||||
}
|
|
||||||
}
|
|
@ -3,14 +3,13 @@ package cmd
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
//"log"
|
//"log"
|
||||||
"time"
|
|
||||||
//"errors"
|
//"errors"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
func DestroyJails(args []string) {
|
func DestroyJails(args []string) {
|
||||||
for _, a := range args {
|
for _, a := range args {
|
||||||
cj, err := getJailFromArray(a, []string{""}, gJails)
|
cj, err := getJailFromArray(a, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting jail: %s\n", err)
|
fmt.Printf("Error getting jail: %s\n", err)
|
||||||
return
|
return
|
||||||
@ -28,8 +27,6 @@ func DestroyJails(args []string) {
|
|||||||
}
|
}
|
||||||
fmt.Printf("Stopping jail %s\n", cj.Name)
|
fmt.Printf("Stopping jail %s\n", cj.Name)
|
||||||
StopJail([]string{fmt.Sprintf("%s/%s", cj.Datastore, cj.Name)})
|
StopJail([]string{fmt.Sprintf("%s/%s", cj.Datastore, cj.Name)})
|
||||||
// Give some time to the host OS to free all mounts accessing processes
|
|
||||||
time.Sleep(1 * time.Second)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get root and config datasets, then destroy
|
// Get root and config datasets, then destroy
|
||||||
@ -54,6 +51,7 @@ func DestroyJails(args []string) {
|
|||||||
fmt.Printf("Error deleting config dataset: %s\n", err)
|
fmt.Printf("Error deleting config dataset: %s\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fmt.Printf("Jail %s is no more!\n", cj.Name)
|
|
||||||
|
//TODO: Delete jail named directory
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
32
cmd/fetch.go
32
cmd/fetch.go
@ -23,9 +23,7 @@ const (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
// TODO : Make this a config/cmd line setting
|
FetchFiles = []string{"base.txz", "lib32.txz", "src.txz"}
|
||||||
//FetchFiles = []string{"base.txz", "lib32.txz", "src.txz"}
|
|
||||||
FetchFiles = []string{"base.txz"}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// TODO: Check if files already exist
|
// TODO: Check if files already exist
|
||||||
@ -68,9 +66,9 @@ func fetchRelease(release string, proto string, arch string, datastore string, f
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create download/XX.X dataset if necessary
|
// Create download/XX.X-RELEASE dataset if necessary
|
||||||
thisDownloadDsName := fmt.Sprintf("%s/%s", downloadDsName, release)
|
thisDownloadDsName := fmt.Sprintf("%s/%s-RELEASE", downloadDsName, release)
|
||||||
thisDownloadDsMountPoint := fmt.Sprintf("%s/%s", downloadDsMountPoint, release)
|
thisDownloadDsMountPoint := fmt.Sprintf("%s/%s-RELEASE", downloadDsMountPoint, release)
|
||||||
exist, err = doZfsDatasetExist(thisDownloadDsName)
|
exist, err = doZfsDatasetExist(thisDownloadDsName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Error accessing dataset %s: %v\n", thisDownloadDsName, err)
|
return fmt.Errorf("Error accessing dataset %s: %v\n", thisDownloadDsName, err)
|
||||||
@ -84,9 +82,9 @@ func fetchRelease(release string, proto string, arch string, datastore string, f
|
|||||||
|
|
||||||
var fetchUrl string
|
var fetchUrl string
|
||||||
if len(fetchFrom) > 0 {
|
if len(fetchFrom) > 0 {
|
||||||
fetchUrl = fmt.Sprintf("%s/%s", fetchFrom, release)
|
fetchUrl = fmt.Sprintf("%s/%s-RELEASE", fetchFrom, release)
|
||||||
} else {
|
} else {
|
||||||
fetchUrl = fmt.Sprintf("%s://%s/%s/%s/%s", proto, ReleaseServer, ReleaseRootDir, arch, release)
|
fetchUrl = fmt.Sprintf("%s://%s/%s/%s/%s-RELEASE", proto, ReleaseServer, ReleaseRootDir, arch, release)
|
||||||
}
|
}
|
||||||
log.Debugf("FetchURL = %s", fetchUrl)
|
log.Debugf("FetchURL = %s", fetchUrl)
|
||||||
|
|
||||||
@ -155,9 +153,9 @@ func extractRelease(release string, datastore string) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create releases/XX.X dataset if necessary
|
// Create releases/XX.X-RELEASE dataset if necessary
|
||||||
thisReleaseDsName := fmt.Sprintf("%s/%s", releaseDsName, release)
|
thisReleaseDsName := fmt.Sprintf("%s/%s-RELEASE", releaseDsName, release)
|
||||||
thisReleaseDsMountPoint := fmt.Sprintf("%s/%s", releaseDsMountPoint, release)
|
thisReleaseDsMountPoint := fmt.Sprintf("%s/%s-RELEASE", releaseDsMountPoint, release)
|
||||||
exist, err = doZfsDatasetExist(thisReleaseDsName)
|
exist, err = doZfsDatasetExist(thisReleaseDsName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error accessing dataset %s: %v\n", thisReleaseDsName, err)
|
fmt.Printf("Error accessing dataset %s: %v\n", thisReleaseDsName, err)
|
||||||
@ -171,7 +169,7 @@ func extractRelease(release string, datastore string) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create releases/XX.X/root dataset if necessary
|
// Create releases/XX.X-RELEASE/root dataset if necessary
|
||||||
thisReleaseRootDsName := fmt.Sprintf("%s/root", thisReleaseDsName)
|
thisReleaseRootDsName := fmt.Sprintf("%s/root", thisReleaseDsName)
|
||||||
thisReleaseRootDsMountPoint := fmt.Sprintf("%s/root", thisReleaseDsMountPoint)
|
thisReleaseRootDsMountPoint := fmt.Sprintf("%s/root", thisReleaseDsMountPoint)
|
||||||
exist, err = doZfsDatasetExist(thisReleaseRootDsName)
|
exist, err = doZfsDatasetExist(thisReleaseRootDsName)
|
||||||
@ -187,9 +185,9 @@ func extractRelease(release string, datastore string) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now extract download/$RELEASE/*.txz to releases/XX.X/root
|
// Now extract download/$RELEASE/*.txz to releases/XX.X-RELEASE/root
|
||||||
downloadDsMountPoint := fmt.Sprintf("%s/download", ds.Mountpoint)
|
downloadDsMountPoint := fmt.Sprintf("%s/download", ds.Mountpoint)
|
||||||
downloadDir := fmt.Sprintf("%s/%s", downloadDsMountPoint, release)
|
downloadDir := fmt.Sprintf("%s/%s-RELEASE", downloadDsMountPoint, release)
|
||||||
|
|
||||||
d, err := os.Open(downloadDir)
|
d, err := os.Open(downloadDir)
|
||||||
defer d.Close()
|
defer d.Close()
|
||||||
@ -203,13 +201,12 @@ func extractRelease(release string, datastore string) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extract every .txz files in FetchFiles
|
// Extract every .txz files
|
||||||
for _, fi := range files {
|
for _, fi := range files {
|
||||||
if false == fi.IsDir() {
|
if false == fi.IsDir() {
|
||||||
if strings.HasSuffix(fi.Name(), ".txz") {
|
if strings.HasSuffix(fi.Name(), ".txz") {
|
||||||
if isStringInArray(FetchFiles, fi.Name()) {
|
|
||||||
ar := fmt.Sprintf("%s/%s", downloadDir, fi.Name())
|
ar := fmt.Sprintf("%s/%s", downloadDir, fi.Name())
|
||||||
fmt.Printf("Extracting file %s to %s... ", ar, thisReleaseRootDsMountPoint)
|
fmt.Printf("Extracting file %s... ", ar)
|
||||||
// pure Go method, sorry this is so slow. Also I did not handle permissions in this
|
// pure Go method, sorry this is so slow. Also I did not handle permissions in this
|
||||||
/* f, err := os.Open(ar)
|
/* f, err := os.Open(ar)
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
@ -270,7 +267,6 @@ func extractRelease(release string, datastore string) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
func fetchFile(proto, baseUrl, fileName, storeDir string, checksum []byte) error {
|
func fetchFile(proto, baseUrl, fileName, storeDir string, checksum []byte) error {
|
||||||
// Check storeDir exist
|
// Check storeDir exist
|
||||||
|
@ -249,7 +249,7 @@ func listJailsFromDirectory(dir string, dsname string) ([]Jail, error) {
|
|||||||
jailConfPath := fmt.Sprintf("%s/%s/%s", dir, fi.Name(), "config.json")
|
jailConfPath := fmt.Sprintf("%s/%s/%s", dir, fi.Name(), "config.json")
|
||||||
jailConf, err := getJailConfig(jailConfPath)
|
jailConf, err := getJailConfig(jailConfPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("ERROR reading jail config from %s\n", jailConfPath)
|
log.Println("ERROR reading jail config for %s", jailConfPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Build jail object from config
|
// 2. Build jail object from config
|
||||||
|
@ -26,7 +26,7 @@ func MigrateJail(args []string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, jn := range jailNames {
|
for _, jn := range jailNames {
|
||||||
cj, err := getJailFromArray(jn, []string{""}, gJails)
|
cj, err := getJailFromArray(jn, gJails)
|
||||||
if cj == nil {
|
if cj == nil {
|
||||||
fmt.Printf("Error getting jail %s: Not found\n", jn)
|
fmt.Printf("Error getting jail %s: Not found\n", jn)
|
||||||
return
|
return
|
||||||
@ -177,7 +177,7 @@ func CleanMigrateMess(args []string) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, jn := range jailNames {
|
for _, jn := range jailNames {
|
||||||
cj, err := getJailFromArray(jn, []string{""}, gJails)
|
cj, err := getJailFromArray(jn, gJails)
|
||||||
if cj == nil {
|
if cj == nil {
|
||||||
return errors.New(fmt.Sprintf("Error getting jail %s: Not found\n", jn))
|
return errors.New(fmt.Sprintf("Error getting jail %s: Not found\n", jn))
|
||||||
}
|
}
|
||||||
|
@ -17,7 +17,7 @@ func GetJailProperties(args []string) {
|
|||||||
for i, a := range args {
|
for i, a := range args {
|
||||||
// Last arg is the jail name
|
// Last arg is the jail name
|
||||||
if i == len(args)-1 {
|
if i == len(args)-1 {
|
||||||
jail, err = getJailFromArray(a, []string{""}, gJails)
|
jail, err = getJailFromArray(a, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error: %s\n", err.Error())
|
fmt.Printf("Error: %s\n", err.Error())
|
||||||
return
|
return
|
||||||
|
41
cmd/root.go
41
cmd/root.go
@ -14,19 +14,12 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
gVersion = "0.36d"
|
gVersion = "0.36c"
|
||||||
|
|
||||||
// TODO : Get from $jail_zpool/defaults.json
|
// TODO : Get from $jail_zpool/defaults.json
|
||||||
MIN_DYN_DEVFS_RULESET = 1000
|
MIN_DYN_DEVFS_RULESET = 1000
|
||||||
)
|
)
|
||||||
|
|
||||||
type createArgs struct {
|
|
||||||
Release string
|
|
||||||
BaseTemplate string
|
|
||||||
Datastore string
|
|
||||||
JailType string
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
var (
|
||||||
gJailHost JailHost
|
gJailHost JailHost
|
||||||
gJails []Jail
|
gJails []Jail
|
||||||
@ -36,8 +29,6 @@ var (
|
|||||||
gForce bool
|
gForce bool
|
||||||
gDebug bool
|
gDebug bool
|
||||||
|
|
||||||
gCreateArgs createArgs
|
|
||||||
|
|
||||||
gConfigFile string
|
gConfigFile string
|
||||||
gDisplayJColumns string
|
gDisplayJColumns string
|
||||||
gDisplaySColumns string
|
gDisplaySColumns string
|
||||||
@ -89,20 +80,6 @@ It support iocage jails and can coexist with iocage.`,
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO
|
|
||||||
Initialize datastore(s) /iocage, /iocage/jails
|
|
||||||
Put defaults.json, update it with hostid,interfaces, and maybe other necessary fields
|
|
||||||
Initialize bridge
|
|
||||||
initCmd = &cobra.Command{
|
|
||||||
Use: "init",
|
|
||||||
Short: "Initialize GoCage",
|
|
||||||
//Long: `Let this show you how much fail I had to get this *cough* perfect`,
|
|
||||||
Run: func(cmd *cobra.Command, args []string) {
|
|
||||||
fv, _ := getFreeBSDVersion()
|
|
||||||
fmt.Printf("GoCage v.%s on FreeBSD %d.%d-%s\n", gVersion, fv.major, fv.minor, fv.flavor)
|
|
||||||
},
|
|
||||||
}*/
|
|
||||||
|
|
||||||
listCmd = &cobra.Command{
|
listCmd = &cobra.Command{
|
||||||
Use: "list",
|
Use: "list",
|
||||||
Short: "Print jails",
|
Short: "Print jails",
|
||||||
@ -340,15 +317,6 @@ You can specify multiple datastores.`,
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
createCmd = &cobra.Command{
|
|
||||||
Use: "create",
|
|
||||||
Short: "Create jail",
|
|
||||||
Run: func(cmd *cobra.Command, args []string) {
|
|
||||||
ListJails(args, false)
|
|
||||||
CreateJail(args)
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
testCmd = &cobra.Command{
|
testCmd = &cobra.Command{
|
||||||
Use: "test",
|
Use: "test",
|
||||||
Short: "temporary command to test some code snippet",
|
Short: "temporary command to test some code snippet",
|
||||||
@ -402,7 +370,7 @@ func init() {
|
|||||||
migrateCmd.Flags().BoolVarP(&gYesToAll, "yes", "y", false, "Answer yes to all questions")
|
migrateCmd.Flags().BoolVarP(&gYesToAll, "yes", "y", false, "Answer yes to all questions")
|
||||||
migrateCmd.MarkFlagRequired("datastore")
|
migrateCmd.MarkFlagRequired("datastore")
|
||||||
|
|
||||||
fetchCmd.Flags().StringVarP(&gFetchRelease, "release", "r", "", "Release to fetch (e.g.: \"13.1-RELEASE\"")
|
fetchCmd.Flags().StringVarP(&gFetchRelease, "release", "r", "", "Release to fetch (e.g.: \"13.1\"")
|
||||||
fetchCmd.Flags().StringVarP(&gFetchIntoDS, "datastore", "o", "", "Datastore release will be saved to")
|
fetchCmd.Flags().StringVarP(&gFetchIntoDS, "datastore", "o", "", "Datastore release will be saved to")
|
||||||
fetchCmd.Flags().StringVarP(&gFetchFrom, "from", "d", "", "Repository to download from. Should contain XY.Z-RELEASE. File protocol supported")
|
fetchCmd.Flags().StringVarP(&gFetchFrom, "from", "d", "", "Repository to download from. Should contain XY.Z-RELEASE. File protocol supported")
|
||||||
fetchCmd.MarkFlagRequired("release")
|
fetchCmd.MarkFlagRequired("release")
|
||||||
@ -411,10 +379,6 @@ func init() {
|
|||||||
upgradeCmd.Flags().StringVarP(&gUpgradeRelease, "release", "r", "", "Release to upgrade to (e.g.: \"13.1-RELEASE\"")
|
upgradeCmd.Flags().StringVarP(&gUpgradeRelease, "release", "r", "", "Release to upgrade to (e.g.: \"13.1-RELEASE\"")
|
||||||
upgradeCmd.MarkFlagRequired("release")
|
upgradeCmd.MarkFlagRequired("release")
|
||||||
|
|
||||||
createCmd.Flags().StringVarP(&gCreateArgs.Release, "release", "r", "", "Release for the jail (e.g.: \"13.1-RELEASE\"")
|
|
||||||
createCmd.Flags().StringVarP(&gCreateArgs.BaseTemplate, "basetpl", "b", "", "Base template. This will create a jail based on basetpl, so every up(date|grade) made to basetpl will immediately propagate to new jail\n")
|
|
||||||
createCmd.Flags().StringVarP(&gCreateArgs.Datastore, "datastore", "d", "", "Datastore to create the jail on. Defaults to first declared in config.")
|
|
||||||
|
|
||||||
// Now declare commands
|
// Now declare commands
|
||||||
rootCmd.AddCommand(versionCmd)
|
rootCmd.AddCommand(versionCmd)
|
||||||
rootCmd.AddCommand(listCmd)
|
rootCmd.AddCommand(listCmd)
|
||||||
@ -432,7 +396,6 @@ func init() {
|
|||||||
rootCmd.AddCommand(fetchCmd)
|
rootCmd.AddCommand(fetchCmd)
|
||||||
rootCmd.AddCommand(updateCmd)
|
rootCmd.AddCommand(updateCmd)
|
||||||
rootCmd.AddCommand(upgradeCmd)
|
rootCmd.AddCommand(upgradeCmd)
|
||||||
rootCmd.AddCommand(createCmd)
|
|
||||||
|
|
||||||
rootCmd.AddCommand(testCmd)
|
rootCmd.AddCommand(testCmd)
|
||||||
|
|
||||||
|
41
cmd/start.go
41
cmd/start.go
@ -330,10 +330,8 @@ func configureDhcpOrAcceptRtadv(jail *Jail, ipproto int, enable bool) error {
|
|||||||
|
|
||||||
for _, n := range nics {
|
for _, n := range nics {
|
||||||
// vnet0 is epair0b inside jail
|
// vnet0 is epair0b inside jail
|
||||||
//if strings.Contains(n, "vnet") {
|
if strings.Contains(n, "vnet") {
|
||||||
if strings.HasPrefix(n, "vnet") {
|
n = fmt.Sprintf("%sb", strings.Replace(n, "vnet", "epair", 1))
|
||||||
splitd := strings.Split(n, "|")
|
|
||||||
n = fmt.Sprintf("%sb", strings.Replace(splitd[0], "vnet", "epair", 1))
|
|
||||||
}
|
}
|
||||||
key := fmt.Sprintf("ifconfig_%s", n)
|
key := fmt.Sprintf("ifconfig_%s", n)
|
||||||
value := "SYNCDHCP"
|
value := "SYNCDHCP"
|
||||||
@ -344,12 +342,12 @@ func configureDhcpOrAcceptRtadv(jail *Jail, ipproto int, enable bool) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if enable == true {
|
if enable == true {
|
||||||
err := enableRcKeyValue(fmt.Sprintf("%s/etc/rc.conf", jail.RootPath), key, value)
|
err := enableRcKeyValue(jail.ConfigPath, key, value)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("ERROR setting %s=%s with sysrc for jail %s: %s\n", key, value, jail.Name, err)
|
return fmt.Errorf("ERROR setting %s=%s with sysrc for jail %s: %s\n", key, value, jail.Name, err)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
err := disableRcKey(fmt.Sprintf("%s/etc/rc.conf", jail.RootPath), key)
|
err := disableRcKey(jail.ConfigPath, key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("ERROR deleting %s with sysrc for jail %s: %v\n", key, jail.Name, err)
|
return fmt.Errorf("ERROR deleting %s with sysrc for jail %s: %v\n", key, jail.Name, err)
|
||||||
}
|
}
|
||||||
@ -510,7 +508,7 @@ func buildDevfsRuleSet(jail *Jail, m *sync.Mutex) (error, int) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Debugf("buildDevfsRuleSet: Build ruleset %d\n", ruleset)
|
log.Debug("buildDevfsRuleSet: Build ruleset %d\n", ruleset)
|
||||||
|
|
||||||
// Get default devfs_ruleset for the datastore
|
// Get default devfs_ruleset for the datastore
|
||||||
// UPDATE: We don't need this as every jail have a default Devfs_ruleset value
|
// UPDATE: We don't need this as every jail have a default Devfs_ruleset value
|
||||||
@ -895,11 +893,11 @@ func setupVnetInterfaceHostSide(jail *Jail) ([]string, error) {
|
|||||||
}
|
}
|
||||||
epairs = append(epairs, hsepair)
|
epairs = append(epairs, hsepair)
|
||||||
}
|
}
|
||||||
log.Debugf("setupVnetInterfaceHostSide: returning %v\n", epairs)
|
|
||||||
return epairs, nil
|
return epairs, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func setupVnetInterfaceJailSide(jail *Jail) error {
|
func setupVnetInterfaceJailSide(jail *Jail, hsepair string) error {
|
||||||
var jsmac []byte
|
var jsmac []byte
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
@ -948,7 +946,10 @@ func setupVnetInterfaceJailSide(jail *Jail) error {
|
|||||||
jsmac = val.Bytes()
|
jsmac = val.Bytes()
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd := fmt.Sprintf("/sbin/ifconfig %s vnet %s", jnic, jail.InternalName)
|
lasta := strings.LastIndex(hsepair, "a")
|
||||||
|
jsepair := hsepair[:lasta] + strings.Replace(hsepair[lasta:], "a", "b", 1)
|
||||||
|
|
||||||
|
cmd := fmt.Sprintf("/sbin/ifconfig %s vnet %s", jsepair, jail.InternalName)
|
||||||
_, err := executeCommand(cmd)
|
_, err := executeCommand(cmd)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Error linking interface to jail: %v\n", err)
|
return fmt.Errorf("Error linking interface to jail: %v\n", err)
|
||||||
@ -960,14 +961,14 @@ func setupVnetInterfaceJailSide(jail *Jail) error {
|
|||||||
return fmt.Errorf("Error getting bridge %s mtu: %v\n", bridge, err)
|
return fmt.Errorf("Error getting bridge %s mtu: %v\n", bridge, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd = fmt.Sprintf("/usr/sbin/jexec %d ifconfig %s mtu %d", jail.JID, jnic, mtu)
|
cmd = fmt.Sprintf("/usr/sbin/jexec %d ifconfig %s mtu %d", jail.JID, jsepair, mtu)
|
||||||
_, err = executeCommand(cmd)
|
_, err = executeCommand(cmd)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Error setting mtu: %v\n", err)
|
return fmt.Errorf("Error setting mtu: %v\n", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// rename epairXXb to epair0b (or opair1b, ...)
|
// rename epairXXb to epair0b (or opair1b, ...)
|
||||||
cmd = fmt.Sprintf("/usr/sbin/setfib %s jexec %d ifconfig %s name %s", jail.Config.Exec_fib, jail.JID, jnic, jnic)
|
cmd = fmt.Sprintf("/usr/sbin/setfib %s jexec %d ifconfig %s name %s", jail.Config.Exec_fib, jail.JID, jsepair, jnic)
|
||||||
_, err = executeCommand(cmd)
|
_, err = executeCommand(cmd)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Error linking interface to jail: %v\n", err)
|
return fmt.Errorf("Error linking interface to jail: %v\n", err)
|
||||||
@ -1009,8 +1010,6 @@ func setupVnetInterfaceJailSide(jail *Jail) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Debugf("setupVnetInterfaceJailSide: return with success\n")
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1166,7 +1165,7 @@ func StartJail(args []string) {
|
|||||||
|
|
||||||
for _, a := range args {
|
for _, a := range args {
|
||||||
// Check if jail exist and is distinctly named
|
// Check if jail exist and is distinctly named
|
||||||
cj, err = getJailFromArray(a, []string{"jail"}, gJails)
|
cj, err = getJailFromArray(a, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting jail: %s\n", err)
|
fmt.Printf("Error getting jail: %s\n", err)
|
||||||
continue
|
continue
|
||||||
@ -1416,20 +1415,21 @@ func StartJail(args []string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fmt.Printf(" > Setup VNet network:\n")
|
fmt.Printf(" > Setup VNet network:\n")
|
||||||
_, err = setupVnetInterfaceHostSide(cj);
|
hsepairs, err := setupVnetInterfaceHostSide(cj);
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error setting VNet interface host side: %v\n", err)
|
fmt.Printf("Error setting VNet interface host side: %v\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = setupVnetInterfaceJailSide(cj); err != nil {
|
for _, ep := range hsepairs {
|
||||||
|
if err = setupVnetInterfaceJailSide(cj, ep); err != nil {
|
||||||
fmt.Printf("Error setting VNet interface jail side: %v\n", err)
|
fmt.Printf("Error setting VNet interface jail side: %v\n", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
}
|
||||||
fmt.Printf(" > Setup VNet network: OK\n")
|
fmt.Printf(" > Setup VNet network: OK\n")
|
||||||
|
|
||||||
// Set default route, unless main network is dhcp
|
// TODO: Handle DHCP
|
||||||
if ! cj.isFirstNetDhcp() {
|
|
||||||
fmt.Printf(" > Setup default ipv4 gateway:\n")
|
fmt.Printf(" > Setup default ipv4 gateway:\n")
|
||||||
cmd := fmt.Sprintf("/usr/sbin/setfib %s /usr/sbin/jexec %d route add default %s", cj.Config.Exec_fib, cj.JID, cj.Config.Defaultrouter)
|
cmd := fmt.Sprintf("/usr/sbin/setfib %s /usr/sbin/jexec %d route add default %s", cj.Config.Exec_fib, cj.JID, cj.Config.Defaultrouter)
|
||||||
out, err := executeCommand(cmd)
|
out, err := executeCommand(cmd)
|
||||||
@ -1438,7 +1438,6 @@ func StartJail(args []string) {
|
|||||||
} else {
|
} else {
|
||||||
fmt.Printf(" > Setup default ipv4 gateway: OK\n")
|
fmt.Printf(" > Setup default ipv4 gateway: OK\n")
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if cj.Config.Ip6_addr != "none" {
|
if cj.Config.Ip6_addr != "none" {
|
||||||
fmt.Printf(" > Setup default ipv6 gateway:\n")
|
fmt.Printf(" > Setup default ipv6 gateway:\n")
|
||||||
@ -1478,7 +1477,7 @@ func StartJail(args []string) {
|
|||||||
fmt.Printf(" > Start services:\n")
|
fmt.Printf(" > Start services:\n")
|
||||||
cmd := fmt.Sprintf("/usr/sbin/setfib %s /usr/sbin/jexec %d %s", cj.Config.Exec_fib, cj.JID, cj.Config.Exec_start)
|
cmd := fmt.Sprintf("/usr/sbin/setfib %s /usr/sbin/jexec %d %s", cj.Config.Exec_fib, cj.JID, cj.Config.Exec_start)
|
||||||
err := executeCommandNonBlocking(cmd)
|
err := executeCommandNonBlocking(cmd)
|
||||||
if err != nil {
|
if err != nil && len(out) > 0 {
|
||||||
fmt.Printf("Error: %v\n", err)
|
fmt.Printf("Error: %v\n", err)
|
||||||
} else {
|
} else {
|
||||||
fmt.Printf(" > Start services: OK\n")
|
fmt.Printf(" > Start services: OK\n")
|
||||||
|
@ -270,7 +270,7 @@ func StopJail(args []string) {
|
|||||||
|
|
||||||
for _, a := range args {
|
for _, a := range args {
|
||||||
// Check if jail exist and is distinctly named
|
// Check if jail exist and is distinctly named
|
||||||
cj, err = getJailFromArray(a, []string{"jail"}, gJails)
|
cj, err = getJailFromArray(a, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting jail: %s\n", err)
|
fmt.Printf("Error getting jail: %s\n", err)
|
||||||
continue
|
continue
|
||||||
|
@ -53,7 +53,7 @@ func UpdateJail(args []string) {
|
|||||||
|
|
||||||
for _, a := range args {
|
for _, a := range args {
|
||||||
// Check if jail exist and is distinctly named
|
// Check if jail exist and is distinctly named
|
||||||
cj, err = getJailFromArray(a, []string{""}, gJails)
|
cj, err = getJailFromArray(a, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting jail: %s\n", err)
|
fmt.Printf("Error getting jail: %s\n", err)
|
||||||
continue
|
continue
|
||||||
|
@ -91,7 +91,7 @@ func UpgradeJail(args []string) {
|
|||||||
|
|
||||||
for _, a := range args {
|
for _, a := range args {
|
||||||
// Check if jail exist and is distinctly named
|
// Check if jail exist and is distinctly named
|
||||||
cj, err = getJailFromArray(a, []string{""}, gJails)
|
cj, err = getJailFromArray(a, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("Error getting jail: %s\n", err)
|
fmt.Printf("Error getting jail: %s\n", err)
|
||||||
continue
|
continue
|
||||||
|
234
cmd/utils.go
234
cmd/utils.go
@ -24,141 +24,6 @@ const (
|
|||||||
ifconfigipv4re = `inet[[:space:]](` + ipv4re + `)`
|
ifconfigipv4re = `inet[[:space:]](` + ipv4re + `)`
|
||||||
// Maximum thread qty for start/stop
|
// Maximum thread qty for start/stop
|
||||||
gMaxThreads = 4
|
gMaxThreads = 4
|
||||||
|
|
||||||
gDefaultsJson = ` {
|
|
||||||
"CONFIG_VERSION": "27",
|
|
||||||
"allow_chflags": 0,
|
|
||||||
"allow_mlock": 0,
|
|
||||||
"allow_mount": 0,
|
|
||||||
"allow_mount_devfs": 0,
|
|
||||||
"allow_mount_fusefs": 0,
|
|
||||||
"allow_mount_nullfs": 0,
|
|
||||||
"allow_mount_procfs": 0,
|
|
||||||
"allow_mount_tmpfs": 0,
|
|
||||||
"allow_mount_zfs": 0,
|
|
||||||
"allow_quotas": 0,
|
|
||||||
"allow_raw_sockets": 0,
|
|
||||||
"allow_set_hostname": 1,
|
|
||||||
"allow_socket_af": 0,
|
|
||||||
"allow_sysvipc": 0,
|
|
||||||
"allow_tun": 0,
|
|
||||||
"allow_vmm": 0,
|
|
||||||
"assign_localhost": 0,
|
|
||||||
"available": "readonly",
|
|
||||||
"basejail": 0,
|
|
||||||
"boot": 0,
|
|
||||||
"bpf": 0,
|
|
||||||
"children_max": "0",
|
|
||||||
"comment": "none",
|
|
||||||
"compression": "lz4",
|
|
||||||
"compressratio": "readonly",
|
|
||||||
"coredumpsize": "off",
|
|
||||||
"count": "1",
|
|
||||||
"cpuset": "off",
|
|
||||||
"cputime": "off",
|
|
||||||
"datasize": "off",
|
|
||||||
"dedup": "off",
|
|
||||||
"defaultrouter": "auto",
|
|
||||||
"defaultrouter6": "auto",
|
|
||||||
"depends": "none",
|
|
||||||
"devfs_ruleset": "4",
|
|
||||||
"dhcp": 0,
|
|
||||||
"enforce_statfs": "2",
|
|
||||||
"exec_clean": 1,
|
|
||||||
"exec_created": "/usr/bin/true",
|
|
||||||
"exec_fib": "0",
|
|
||||||
"exec_jail_user": "root",
|
|
||||||
"exec_poststart": "/usr/bin/true",
|
|
||||||
"exec_poststop": "/usr/bin/true",
|
|
||||||
"exec_prestart": "/usr/bin/true",
|
|
||||||
"exec_prestop": "/usr/bin/true",
|
|
||||||
"exec_start": "/bin/sh /etc/rc",
|
|
||||||
"exec_stop": "/bin/sh /etc/rc.shutdown",
|
|
||||||
"exec_system_jail_user": "0",
|
|
||||||
"exec_system_user": "root",
|
|
||||||
"exec_timeout": "60",
|
|
||||||
"host_domainname": "none",
|
|
||||||
"host_time": 1,
|
|
||||||
"hostid": "36353536-3135-5a43-4a34-313130315a56",
|
|
||||||
"hostid_strict_check": 0,
|
|
||||||
"interfaces": "vnet0:bridge0",
|
|
||||||
"ip4": "new",
|
|
||||||
"ip4_addr": "none",
|
|
||||||
"ip4_saddrsel": 1,
|
|
||||||
"ip6": "new",
|
|
||||||
"ip6_addr": "none",
|
|
||||||
"ip6_saddrsel": 1,
|
|
||||||
"ip_hostname": 0,
|
|
||||||
"jail_zfs": 0,
|
|
||||||
"jail_zfs_mountpoint": "none",
|
|
||||||
"last_started": "none",
|
|
||||||
"localhost_ip": "none",
|
|
||||||
"login_flags": "-f root",
|
|
||||||
"mac_prefix": "2c44fd",
|
|
||||||
"maxproc": "off",
|
|
||||||
"memorylocked": "off",
|
|
||||||
"memoryuse": "off",
|
|
||||||
"min_dyn_devfs_ruleset": "1000",
|
|
||||||
"mount_devfs": 1,
|
|
||||||
"mount_fdescfs": 1,
|
|
||||||
"mount_linprocfs": 0,
|
|
||||||
"mount_procfs": 0,
|
|
||||||
"mountpoint": "readonly",
|
|
||||||
"msgqqueued": "off",
|
|
||||||
"msgqsize": "off",
|
|
||||||
"nat": 0,
|
|
||||||
"nat_backend": "ipfw",
|
|
||||||
"nat_forwards": "none",
|
|
||||||
"nat_interface": "none",
|
|
||||||
"nat_prefix": "172.16",
|
|
||||||
"nmsgq": "off",
|
|
||||||
"notes": "none",
|
|
||||||
"nsem": "off",
|
|
||||||
"nsemop": "off",
|
|
||||||
"nshm": "off",
|
|
||||||
"nthr": "off",
|
|
||||||
"openfiles": "off",
|
|
||||||
"origin": "readonly",
|
|
||||||
"owner": "root",
|
|
||||||
"pcpu": "off",
|
|
||||||
"plugin_name": "none",
|
|
||||||
"plugin_repository": "none",
|
|
||||||
"priority": "99",
|
|
||||||
"pseudoterminals": "off",
|
|
||||||
"quota": "none",
|
|
||||||
"readbps": "off",
|
|
||||||
"readiops": "off",
|
|
||||||
"reservation": "none",
|
|
||||||
"resolver": "/etc/resolv.conf",
|
|
||||||
"rlimits": "off",
|
|
||||||
"rtsold": 0,
|
|
||||||
"securelevel": "2",
|
|
||||||
"shmsize": "off",
|
|
||||||
"stacksize": "off",
|
|
||||||
"stop_timeout": "30",
|
|
||||||
"swapuse": "off",
|
|
||||||
"sync_state": "none",
|
|
||||||
"sync_target": "none",
|
|
||||||
"sync_tgt_zpool": "none",
|
|
||||||
"sysvmsg": "new",
|
|
||||||
"sysvsem": "new",
|
|
||||||
"sysvshm": "new",
|
|
||||||
"template": 0,
|
|
||||||
"type": "jail",
|
|
||||||
"used": "readonly",
|
|
||||||
"vmemoryuse": "off",
|
|
||||||
"vnet": 0,
|
|
||||||
"vnet0_mac": "none",
|
|
||||||
"vnet1_mac": "none",
|
|
||||||
"vnet2_mac": "none",
|
|
||||||
"vnet3_mac": "none",
|
|
||||||
"vnet_default_interface": "auto",
|
|
||||||
"vnet_interfaces": "none",
|
|
||||||
"wallclock": "off",
|
|
||||||
"writebps": "off",
|
|
||||||
"writeiops": "off"
|
|
||||||
}
|
|
||||||
`
|
|
||||||
)
|
)
|
||||||
|
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
@ -354,8 +219,6 @@ func executeCommand(cmdline string) (string, error) {
|
|||||||
word = word + string(c)
|
word = word + string(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Debugf("executeCommand: %s\n", strings.Join(cmd, " "))
|
|
||||||
|
|
||||||
if len(cmd) > 1 {
|
if len(cmd) > 1 {
|
||||||
out, err = exec.Command(cmd[0], cmd[1:]...).CombinedOutput()
|
out, err = exec.Command(cmd[0], cmd[1:]...).CombinedOutput()
|
||||||
} else {
|
} else {
|
||||||
@ -682,11 +545,8 @@ func zfsSnapshot(dataset string, snapname string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy snapshot to a new dataset
|
|
||||||
// TODO : Intercept death of sending process, then kill receiving
|
|
||||||
func zfsCopy(src string, dest string) error {
|
func zfsCopy(src string, dest string) error {
|
||||||
// First, declare sending process & pipe
|
// First, declare sending process & pipe
|
||||||
log.Debugf("Execute: zfs send %s\n", src)
|
|
||||||
cmd_send := exec.Command("zfs", "send", src)
|
cmd_send := exec.Command("zfs", "send", src)
|
||||||
stdout_send, err := cmd_send.StdoutPipe()
|
stdout_send, err := cmd_send.StdoutPipe()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -695,7 +555,6 @@ func zfsCopy(src string, dest string) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// then declare receiving process & pipe
|
// then declare receiving process & pipe
|
||||||
log.Debugf("Execute: zfs receive %s\n", dest)
|
|
||||||
cmd_recv := exec.Command("zfs", "receive", dest)
|
cmd_recv := exec.Command("zfs", "receive", dest)
|
||||||
stdin_recv, err := cmd_recv.StdinPipe()
|
stdin_recv, err := cmd_recv.StdinPipe()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -709,19 +568,16 @@ func zfsCopy(src string, dest string) error {
|
|||||||
// then start processes and wait for finish
|
// then start processes and wait for finish
|
||||||
if err := cmd_recv.Start(); err != nil {
|
if err := cmd_recv.Start(); err != nil {
|
||||||
//fmt.Printf("Error: %v\n", err)
|
//fmt.Printf("Error: %v\n", err)
|
||||||
log.Debugf("zfs receive %s started: %v", dest, err)
|
|
||||||
return errors.New(fmt.Sprintf("Error starting receive process: %v\n", err))
|
return errors.New(fmt.Sprintf("Error starting receive process: %v\n", err))
|
||||||
}
|
}
|
||||||
//fmt.Printf("DEBUG: Start \"zfs send %s\"\n", dsconf)
|
//fmt.Printf("DEBUG: Start \"zfs send %s\"\n", dsconf)
|
||||||
if err := cmd_send.Start(); err != nil {
|
if err := cmd_send.Start(); err != nil {
|
||||||
//fmt.Printf("Error: %v\n", err)
|
//fmt.Printf("Error: %v\n", err)
|
||||||
log.Debugf("zfs send %s started: %v", src, err)
|
|
||||||
return errors.New(fmt.Sprintf("Error starting send process: %v\n", err))
|
return errors.New(fmt.Sprintf("Error starting send process: %v\n", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
//fmt.Printf("DEBUG: Wait for zfs send to finish\n")
|
//fmt.Printf("DEBUG: Wait for zfs send to finish\n")
|
||||||
if err := cmd_send.Wait(); err != nil {
|
if err := cmd_send.Wait(); err != nil {
|
||||||
log.Debugf("zfs send %s stopped with %v", err)
|
|
||||||
//fmt.Printf("Error: zfs send halted with %v\n", err)
|
//fmt.Printf("Error: zfs send halted with %v\n", err)
|
||||||
return errors.New(fmt.Sprintf("send halted with: %v\n", err))
|
return errors.New(fmt.Sprintf("send halted with: %v\n", err))
|
||||||
}
|
}
|
||||||
@ -812,7 +668,6 @@ func zfsGetDatasetByMountpoint(mountpoint string) (string, error) {
|
|||||||
|
|
||||||
// Delete a ZFS Dataset by name
|
// Delete a ZFS Dataset by name
|
||||||
func zfsDestroy(dataset string) error {
|
func zfsDestroy(dataset string) error {
|
||||||
log.Debugf("execute \"zfs destroy -r %s\"\n", dataset)
|
|
||||||
cmd := fmt.Sprintf("zfs destroy -r %s", dataset)
|
cmd := fmt.Sprintf("zfs destroy -r %s", dataset)
|
||||||
out, err := executeCommand(cmd)
|
out, err := executeCommand(cmd)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -821,30 +676,6 @@ func zfsDestroy(dataset string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy file */
|
|
||||||
func copyFile(src, dst string) error {
|
|
||||||
srcfinfo, err := os.Stat(src)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("Cannot find source file: %s", err.Error())
|
|
||||||
}
|
|
||||||
if !srcfinfo.Mode().IsRegular() {
|
|
||||||
return fmt.Errorf("%s is not a regular file", src)
|
|
||||||
}
|
|
||||||
|
|
||||||
srcHandle, err := os.Open(src)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("Cannot open source file: %s", err.Error())
|
|
||||||
}
|
|
||||||
defer srcHandle.Close()
|
|
||||||
dstHandle, err := os.Create(dst)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("Cannot create destination file: %s", err.Error())
|
|
||||||
}
|
|
||||||
defer dstHandle.Close()
|
|
||||||
_, err = io.Copy(dstHandle, srcHandle)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
*
|
*
|
||||||
* rc.conf management
|
* rc.conf management
|
||||||
@ -1021,29 +852,14 @@ func isStringInArray(strarr []string, searched string) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (j Jail) isFirstNetDhcp() bool {
|
|
||||||
for _, n := range strings.Split(j.Config.Ip4_addr, ",") {
|
|
||||||
splitd := strings.Split(n, "|")
|
|
||||||
if len(splitd) > 1 && strings.EqualFold(splitd[1], "dhcp") {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
/********************************************************************************
|
/********************************************************************************
|
||||||
* Get a specific jail reference, to update properties after a range loop
|
* Get a specific jail reference, to update properties after a range loop
|
||||||
* Name can be short or long form ("myjail" vs "mystore/myjail")
|
* Name can be short or long form ("myjail" vs "mystore/myjail")
|
||||||
* An empty jailtype means "all types"
|
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
func getJailFromArray(name string, jailtypes []string, jarray []Jail) (*Jail, error) {
|
func getJailFromArray(name string, jarray []Jail) (*Jail, error) {
|
||||||
var ds, jail string
|
var ds, jail string
|
||||||
var jails []Jail
|
var jails []Jail
|
||||||
|
|
||||||
if (len(jailtypes) == 1 && len(jailtypes[0]) == 0) || len(jailtypes) == 0 {
|
|
||||||
jailtypes = []string{"jail", "basetpl"}
|
|
||||||
}
|
|
||||||
|
|
||||||
if strings.Contains(name, "/") {
|
if strings.Contains(name, "/") {
|
||||||
split := strings.Split(name, "/")
|
split := strings.Split(name, "/")
|
||||||
if len(split) != 2 {
|
if len(split) != 2 {
|
||||||
@ -1056,8 +872,8 @@ func getJailFromArray(name string, jailtypes []string, jarray []Jail) (*Jail, er
|
|||||||
}
|
}
|
||||||
|
|
||||||
for i, j := range jarray {
|
for i, j := range jarray {
|
||||||
|
//if jail == j.Name {
|
||||||
if strings.HasPrefix(j.Name, jail) {
|
if strings.HasPrefix(j.Name, jail) {
|
||||||
if isStringInArray(jailtypes, j.Config.Jailtype) {
|
|
||||||
if len(ds) > 0 {
|
if len(ds) > 0 {
|
||||||
if strings.EqualFold(ds, j.Datastore) {
|
if strings.EqualFold(ds, j.Datastore) {
|
||||||
return &jarray[i], nil
|
return &jarray[i], nil
|
||||||
@ -1069,7 +885,6 @@ func getJailFromArray(name string, jailtypes []string, jarray []Jail) (*Jail, er
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if len(jails) > 0 {
|
if len(jails) > 0 {
|
||||||
if len(jails) > 1 {
|
if len(jails) > 1 {
|
||||||
return &Jail{}, errors.New("More than one jail matching, please use datastore/jail format or full name")
|
return &Jail{}, errors.New("More than one jail matching, please use datastore/jail format or full name")
|
||||||
@ -1098,7 +913,7 @@ func setJailConfigUpdated(jail *Jail) error {
|
|||||||
return errors.New(fmt.Sprintf("No config path for jail %s", jail.Name))
|
return errors.New(fmt.Sprintf("No config path for jail %s", jail.Name))
|
||||||
}
|
}
|
||||||
|
|
||||||
j, err := getJailFromArray(jail.Name, []string{""}, gJails)
|
j, err := getJailFromArray(jail.Name, gJails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -1172,47 +987,6 @@ func writeConfigToDisk(j *Jail, changeauto bool) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (j Jail) WriteConfigToDisk(changeauto bool) {
|
|
||||||
// we will manipulate properties so get a copy
|
|
||||||
jc := j.Config
|
|
||||||
|
|
||||||
if changeauto == false {
|
|
||||||
// Overwrite "auto" properties
|
|
||||||
ondiskjc, err := getJailConfig(j.ConfigPath)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
// TODO : List all fields, then call getStructFieldValue to compare value with "auto"
|
|
||||||
// If "auto" then keep it that way before writing ondiskjc to disk
|
|
||||||
var properties []string
|
|
||||||
properties = getStructFieldNames(ondiskjc, properties, "")
|
|
||||||
|
|
||||||
for _, p := range properties {
|
|
||||||
v, _, err := getStructFieldValue(ondiskjc, p)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
if v.String() == "auto" {
|
|
||||||
err = setStructFieldValue(&jc, p, "auto")
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR sanitizing config: %s\n", err.Error())
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
marshaled, err := json.MarshalIndent(jc, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("ERROR marshaling config: %s\n", err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
if os.WriteFile(j.ConfigPath, []byte(marshaled), 0644); err != nil {
|
|
||||||
fmt.Printf("Error writing config file %s: %v\n", j.ConfigPath, err)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/******************************************************************************
|
/******************************************************************************
|
||||||
* Return the quantity of jails with the name passed as parameter
|
* Return the quantity of jails with the name passed as parameter
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
@ -1228,7 +1002,7 @@ func countOfJailsWithThisName(name string) int {
|
|||||||
|
|
||||||
|
|
||||||
func isNameDistinctive(name string, jails []Jail) bool {
|
func isNameDistinctive(name string, jails []Jail) bool {
|
||||||
_, err := getJailFromArray(name, []string{""}, jails)
|
_, err := getJailFromArray(name, jails)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false
|
return false
|
||||||
} else {
|
} else {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user