package ooni import ( "io/ioutil" "os" "sync/atomic" "github.com/apex/log" "github.com/ooni/probe-cli/config" "github.com/ooni/probe-cli/internal/bindata" "github.com/ooni/probe-cli/internal/database" "github.com/ooni/probe-cli/internal/enginex" "github.com/ooni/probe-cli/utils" engine "github.com/ooni/probe-engine" "github.com/pkg/errors" "upper.io/db.v3/lib/sqlbuilder" ) // Context for OONI Probe type Context struct { Config *config.Config DB sqlbuilder.Database IsBatch bool Session *engine.Session Home string TempDir string dbPath string configPath string // We need to use a int64 in order to use the atomic.AddInt64/LoadInt64 // operations to ensure consistent reads of the variables. isTerminatedAtomicInt int64 } // MaybeLocationLookup will lookup the location of the user unless it's already cached func (c *Context) MaybeLocationLookup() error { return c.Session.MaybeLookupLocation() } // IsTerminated checks to see if the isTerminatedAtomicInt is set to a non zero // value and therefore we have received the signal to shutdown the running test func (c *Context) IsTerminated() bool { i := atomic.LoadInt64(&c.isTerminatedAtomicInt) return i != 0 } // Terminate interrupts the running context func (c *Context) Terminate() { atomic.AddInt64(&c.isTerminatedAtomicInt, 1) } // Init the OONI manager func (c *Context) Init(softwareName, softwareVersion string) error { var err error if err = MaybeInitializeHome(c.Home); err != nil { return err } if c.configPath != "" { log.Debugf("Reading config file from %s", c.configPath) c.Config, err = config.ReadConfig(c.configPath) } else { log.Debug("Reading default config file") c.Config, err = InitDefaultConfig(c.Home) } if err != nil { return err } if err = c.Config.MaybeMigrate(); err != nil { return errors.Wrap(err, "migrating config") } c.dbPath = utils.DBDir(c.Home, "main") log.Debugf("Connecting to database sqlite3://%s", c.dbPath) db, err := database.Connect(c.dbPath) if err != nil { return err } c.DB = db tempDir, err := ioutil.TempDir("", "ooni") if err != nil { return errors.Wrap(err, "creating TempDir") } c.TempDir = tempDir kvstore, err := engine.NewFileSystemKVStore( utils.EngineDir(c.Home), ) if err != nil { return errors.Wrap(err, "creating engine's kvstore") } sess, err := engine.NewSession(engine.SessionConfig{ KVStore: kvstore, Logger: enginex.Logger, SoftwareName: softwareName, SoftwareVersion: softwareVersion, AssetsDir: utils.AssetsDir(c.Home), TempDir: c.TempDir, }) if err != nil { return err } c.Session = sess return nil } // NewContext creates a new context instance. func NewContext(configPath string, homePath string) *Context { return &Context{ Home: homePath, Config: &config.Config{}, configPath: configPath, isTerminatedAtomicInt: 0, } } // MaybeInitializeHome does the setup for a new OONI Home func MaybeInitializeHome(home string) error { for _, d := range utils.RequiredDirs(home) { if _, e := os.Stat(d); e != nil { if err := os.MkdirAll(d, 0700); err != nil { return err } } } return nil } // InitDefaultConfig reads the config from common locations or creates it if // missing. func InitDefaultConfig(home string) (*config.Config, error) { var ( err error c *config.Config configPath = utils.ConfigPath(home) ) c, err = config.ReadConfig(configPath) if err != nil { if os.IsNotExist(err) { log.Debugf("writing default config to %s", configPath) var data []byte data, err = bindata.Asset("data/default-config.json") if err != nil { return nil, err } err = ioutil.WriteFile( configPath, data, 0644, ) if err != nil { return nil, err } return InitDefaultConfig(home) } return nil, err } return c, nil }