Skip to content

Commit

Permalink
Promote the standalone mode
Browse files Browse the repository at this point in the history
This change:

- Adapts the sfconfig run to only setup the context. The command is renamed
  `sfconfig dev prepare`.
- Updates the developer documentation to reflect the change.

Change-Id: I9a680f57f1d8421f056cb7b84f959bd99e1f9a7b
  • Loading branch information
morucci committed Nov 16, 2023
1 parent 97086b0 commit 68ba2ce
Show file tree
Hide file tree
Showing 8 changed files with 134 additions and 238 deletions.
153 changes: 29 additions & 124 deletions cli/run.go → cli/sfconfig/cmd/dev/run.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,44 +15,54 @@ import (

"gopkg.in/yaml.v3"
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/log/zap"

sfv1 "github.com/softwarefactory-project/sf-operator/api/v1"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/gerrit"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/nodepool"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/sfprometheus"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/utils"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/config"
controllers "github.com/softwarefactory-project/sf-operator/controllers"
"k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/resource"
"github.com/spf13/cobra"
)

func Run(erase bool) {
var DevCmd = &cobra.Command{
Use: "dev",
Short: "developer utilities",
Run: func(cmd *cobra.Command, args []string) {},
}

var DevPrepareCmd = &cobra.Command{
Use: "prepare",
Short: "prepare dev environment",
Run: func(cmd *cobra.Command, args []string) { Run() },
}

func init() {
DevCmd.AddCommand(DevPrepareCmd)
}

func Run() {
ctrl.SetLogger(zap.New(zap.UseFlagOptions(&zap.Options{Development: true})))
sfconfig := config.GetSFConfigOrDie()
fmt.Println("sfconfig started with: ", sfconfig)
cli, err := utils.CreateKubernetesClient("")
if err != nil {
cli = EnsureCluster(err)
panic(err)
}
env := utils.ENV{
Ctx: context.TODO(),
Ns: "sf",
Cli: cli,
}
if erase {
fmt.Println("Erasing...")
// TODO: remove the sfconfig resource and the pv
} else {
// TODO: only do gerrit when provision demo is on?
gerrit.EnsureGerrit(&env, sfconfig.FQDN)
sfprometheus.EnsurePrometheus(&env, sfconfig.FQDN, false)
EnsureDemoConfig(&env, &sfconfig)
nodepool.CreateNamespaceForNodepool(&env, "", "nodepool", "")
EnsureDeployment(&env, &sfconfig)
}
// TODO: only do gerrit when provision demo is on?
EnsureNamespacePermissions(&env)
EnsureCertManager(&env)
EnsurePrometheusOperator(&env)
gerrit.EnsureGerrit(&env, sfconfig.FQDN)
sfprometheus.EnsurePrometheus(&env, sfconfig.FQDN, false)
EnsureDemoConfig(&env, &sfconfig)
nodepool.CreateNamespaceForNodepool(&env, "", "nodepool", "")
EnsureCRD()
}

// EnsureDemoConfig prepares a demo config
Expand Down Expand Up @@ -121,109 +131,14 @@ func EnsureRepo(sfconfig *config.SFConfig, apiKey string, name string) {
utils.RunCmd("git", "-C", path, "reset", "--hard", "origin/master")
}

// EnsureCluster recorvers from client creation error
func EnsureCluster(err error) client.Client {
// TODO: perform openstack server reboot?
panic(fmt.Errorf("cluster error: %s", err))
}

// EnsureDeployment ensures a deployment is running.
func EnsureDeployment(env *utils.ENV, sfconfig *config.SFConfig) {
fmt.Println("[+] Checking SF resource...")
sf, err := utils.GetSF(env, "my-sf")
if sf.Status.Ready {
// running the operator should be a no-op
RunOperator()

fmt.Println("Software Factory is already ready!")
// TODO: connect to the Zuul API and ensure it is running
fmt.Println("Check https://zuul." + sf.Spec.FQDN)
os.Exit(0)

} else if err != nil {
if errors.IsNotFound(err) {
// The resource does not exist
EnsureNamespacePermissions(env)
EnsureCR(env, sfconfig)
EnsureCertManager(env)
EnsurePrometheusOperator(env)
RunOperator()

} else if utils.IsCRDMissing(err) {
// The resource definition does not exist
EnsureNamespacePermissions(env)
EnsureCRD()
EnsureCR(env, sfconfig)
EnsureCertManager(env)
EnsurePrometheusOperator(env)
RunOperator()

} else {
// TODO: check what is the actual error and suggest counter measure, for example:
// if microshift host is up but service is done, apply the ansible-microshift-role
// if kubectl is not connecting ask for reboot or rebuild
fmt.Printf("Error %v\n", errors.IsInvalid(err))
fmt.Println(err)
}

} else {
// Software Factory resource exists, but it is not ready
if IsOperatorRunning() {
// TODO: check operator status
// TODO: check cluster status and/or suggest sf resource delete/recreate
} else {
EnsureCertManager(env)
EnsurePrometheusOperator(env)
RunOperator()
}
}

// TODO: suggest sfconfig --erase if the command does not succeed.
fmt.Println("[+] Couldn't deploy your software factory, sorry!")
}

func EnsureNamespacePermissions(env *utils.ENV) {
// TODO: implement natively
// TODO: ensure setup-namespaces role use this to avoid duplication
utils.RunCmd("kubectl", "label", "--overwrite", "ns", env.Ns, "pod-security.kubernetes.io/enforce=privileged")
utils.RunCmd("kubectl", "label", "--overwrite", "ns", env.Ns, "pod-security.kubernetes.io/enforce-version=v1.24")
utils.RunCmd("oc", "adm", "policy", "add-scc-to-user", "privileged", "-z", "default")
}

func EnsureCR(env *utils.ENV, sfconfig *config.SFConfig) {
fmt.Println("[+] Installing CR...")
var cr sfv1.SoftwareFactory
cr.SetName("my-sf")
cr.SetNamespace(env.Ns)
cr.Spec.FQDN = sfconfig.FQDN
cr.Spec.ConfigLocation = sfv1.ConfigLocationSpec{
BaseURL: "http://gerrit-httpd/",
Name: "config",
ZuulConnectionName: "gerrit",
}
cr.Spec.Zuul.GerritConns = []sfv1.GerritConnection{
{
Name: "gerrit",
Username: "zuul",
Hostname: "gerrit-sshd",
Puburl: "https://gerrit." + sfconfig.FQDN,
},
}

cr.Spec.StorageClassName = "topolvm-provisioner"
logserverVolumeSize, _ := resource.ParseQuantity("2Gi")
cr.Spec.Logserver.Storage.Size = logserverVolumeSize
var err error
for i := 0; i < 10; i++ {
err = env.Cli.Create(env.Ctx, &cr)
if err == nil {
return
}
// Sometime the api needs a bit of time to register the CRD
time.Sleep(2 * time.Second)
}
panic(fmt.Errorf("could not install CR: %s", err))
}

func EnsureCRD() {
// TODO: implement natively and avoir re-entry
fmt.Println("[+] Installing CRD...")
Expand Down Expand Up @@ -253,13 +168,3 @@ func EnsurePrometheusOperator(env *utils.ENV) {
panic(fmt.Errorf("could not install prometheus-operator: %s", err))
}
}

func RunOperator() {
fmt.Println("[+] Running the operator...")
controllers.Main("sf", ":8081", ":8080", false, true)
}

func IsOperatorRunning() bool {
// TODO: implement
return false
}
15 changes: 5 additions & 10 deletions cli/sfconfig/cmd/root.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,8 @@ import (
"fmt"
"os"

"github.com/softwarefactory-project/sf-operator/cli"
bootstrap "github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/bootstrap-tenant-config-repo"
cli "github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/dev"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/gerrit"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/nodepool"
"github.com/softwarefactory-project/sf-operator/cli/sfconfig/cmd/operator"
Expand All @@ -40,13 +40,8 @@ var cfgFile string
// rootCmd represents the base command when called without any subcommands
var rootCmd = &cobra.Command{
Use: "sfconfig",
Short: "sfconfig cli tool",
Long: `sfconfig command line tool
This tool is used to deploy and run tests for sf-operator`,
Run: func(cmd *cobra.Command, args []string) {
erase, _ := cmd.Flags().GetBool("erase")
cli.Run(erase)
},
Short: "Toolbox for sf-operator",
Run: func(cmd *cobra.Command, args []string) {},
}

// Execute adds all child commands to the root command and sets flags appropriately.
Expand All @@ -72,9 +67,9 @@ func init() {
rootCmd.AddCommand(sfprometheus.PrometheusCmd)
rootCmd.AddCommand(nodepool.ProvidersSecretsCmd)
rootCmd.AddCommand(zuul.ZuulCmd)

rootCmd.AddCommand(bootstrap.BootstrapTenantConfigRepoCmd)
rootCmd.Flags().BoolP("erase", "", false, "Erase data")
rootCmd.AddCommand(cli.DevCmd)

}

// initConfig reads in config file and ENV variables if set.
Expand Down
15 changes: 4 additions & 11 deletions cli/sfconfig/cmd/runTests.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,16 +21,9 @@ var extravars []string
// runTestsCmd represents the runTests command
var runTestsCmd = &cobra.Command{
Use: "runTests",
Short: "Run playbook/main.yaml",
Long: `Run playbook/main.yaml playbook, it used for CI job
and can be used locally
Run test_only tag
./tools/sfconfig runTests --test-only
`,
Short: "Run the test suite",
Run: func(cmd *cobra.Command, args []string) {
testOnly, _ := cmd.Flags().GetBool("test-only")
testOnly, _ := cmd.Flags().GetBool("standalone")
upgrade, _ := cmd.Flags().GetBool("upgrade")
verbose, _ := cmd.Flags().GetBool("v")
debug, _ := cmd.Flags().GetBool("vvv")
Expand Down Expand Up @@ -59,7 +52,7 @@ Run test_only tag
} else {
playbookYAML = "playbooks/main.yaml"
if testOnly {
ansiblePlaybookOptions.Tags = "test_only"
ansiblePlaybookOptions.Tags = "standalone"
ansiblePlaybookOptions.AddExtraVar("mode", "standalone")
} else {
ansiblePlaybookOptions.AddExtraVar("mode", "olm")
Expand Down Expand Up @@ -101,7 +94,7 @@ func varListToMap(varsList []string) (map[string]interface{}, error) {
func init() {
rootCmd.AddCommand(runTestsCmd)
runTestsCmd.Flags().StringSliceVarP(&extravars, "extra-var", "e", []string{}, "Set extra variables, the format of each variable must be <key>=<value>")
runTestsCmd.Flags().BoolP("test-only", "t", false, "run test_only")
runTestsCmd.Flags().BoolP("standalone", "s", false, "run standalone test")
runTestsCmd.Flags().BoolP("upgrade", "u", false, "run upgrade test")
runTestsCmd.Flags().Bool("v", false, "run ansible in verbose mode")
runTestsCmd.Flags().Bool("vvv", false, "run ansible in debug mode")
Expand Down
Loading

0 comments on commit 68ba2ce

Please sign in to comment.