Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adds process for zero touch registration and functionality to update processes and configs #64

Merged
merged 59 commits into from
Dec 3, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
59 commits
Select commit Hold shift + click to select a range
b0304ff
adding state artifact fetcher to fetch the state from harbor
Mehul-Kumar-27 Sep 23, 2024
8a5ba11
completing the url fetcher
Mehul-Kumar-27 Sep 25, 2024
c291e6a
adding schedulers and process to satellite
Mehul-Kumar-27 Sep 29, 2024
fe299f8
adding simple notifier to fetch state process
Mehul-Kumar-27 Sep 29, 2024
b6b5dea
added description to the scheduler
Mehul-Kumar-27 Sep 29, 2024
91222d4
coderabbit fixes and changes to fetcher and schedulers
Mehul-Kumar-27 Sep 29, 2024
09419ce
adding new format of the state file
Mehul-Kumar-27 Oct 2, 2024
c60f7a4
adding config to process new state artifact file
Mehul-Kumar-27 Oct 8, 2024
18b5633
Merge branch 'main' into satellite-state-fetcher
Mehul-Kumar-27 Oct 8, 2024
dec1ba0
coderabbit review
Mehul-Kumar-27 Oct 8, 2024
ef0d82a
added ./zot to gitignore
Mehul-Kumar-27 Oct 8, 2024
74fc4b9
fixing the replication process
Mehul-Kumar-27 Oct 10, 2024
3d0e209
fixing the replication and deletion process
Mehul-Kumar-27 Oct 12, 2024
6e54a14
fixing paning while removing the null tags
Mehul-Kumar-27 Oct 13, 2024
8904e99
using repository name instead of the image name while uploading the i…
Mehul-Kumar-27 Oct 16, 2024
d14af7e
adding container runtime config
Mehul-Kumar-27 Oct 20, 2024
7254c1b
containerd function and changing the harbor satellite to a cobra cli …
Mehul-Kumar-27 Oct 21, 2024
d111454
generating config file for containerd
Mehul-Kumar-27 Oct 21, 2024
222c666
adding better logging
Mehul-Kumar-27 Oct 21, 2024
d2180bf
fix
Mehul-Kumar-27 Oct 21, 2024
97f8b01
adding config generation for containerd
Mehul-Kumar-27 Oct 28, 2024
949c60c
fixing host gen file
Mehul-Kumar-27 Oct 28, 2024
1b6b45c
generating the config for the containerd fixes
Mehul-Kumar-27 Oct 29, 2024
8624668
fixes
Mehul-Kumar-27 Oct 29, 2024
771a6e3
coderabbit fixes
Mehul-Kumar-27 Oct 29, 2024
a08405d
fixes
Mehul-Kumar-27 Oct 29, 2024
0dc1160
adding config command for crio
Mehul-Kumar-27 Nov 4, 2024
f754085
moving from toml config to json config
Mehul-Kumar-27 Nov 4, 2024
abb1a3e
making config.json work with the replicator
Mehul-Kumar-27 Nov 4, 2024
a641df3
avoid printing confedential information in log
Mehul-Kumar-27 Nov 4, 2024
2f11137
coderabbit fixes
Mehul-Kumar-27 Nov 4, 2024
17f8ffa
changing satellite config from toml to json
Mehul-Kumar-27 Nov 12, 2024
c194bb0
fixing startup
Mehul-Kumar-27 Nov 12, 2024
86430aa
fixing panic error in generating container runtime config
Mehul-Kumar-27 Nov 12, 2024
2daa2eb
changing config structuring
Mehul-Kumar-27 Nov 18, 2024
6574d77
adding event driven architecture changes
Mehul-Kumar-27 Nov 18, 2024
24bd517
adds config fetch process architecture, registration process and stat…
Mehul-Kumar-27 Nov 18, 2024
872a8fd
adds docker compose and satellite registers different processes
Mehul-Kumar-27 Nov 18, 2024
a656219
minor fixes
Mehul-Kumar-27 Nov 19, 2024
1e8dc43
handelling config error
Mehul-Kumar-27 Nov 19, 2024
b950c1c
code rabbit fixes
Mehul-Kumar-27 Nov 19, 2024
2aa1a46
Merge remote-tracking branch 'origin/container' into architecture
Mehul-Kumar-27 Nov 19, 2024
5ca08c7
dagger version
Mehul-Kumar-27 Nov 19, 2024
e578bd4
Merge branch 'satellite-state-fetcher' into container
Mehul-Kumar-27 Nov 19, 2024
47bdb9c
Merge branch 'container' into architecture
Mehul-Kumar-27 Nov 19, 2024
2604c00
enhancing the config
Mehul-Kumar-27 Nov 20, 2024
7062d10
Merge branch 'main' into container
Mehul-Kumar-27 Nov 21, 2024
9c6d293
Merge branch 'main' into satellite-state-fetcher
Mehul-Kumar-27 Nov 21, 2024
9ccd683
Merge branch 'main' into architecture
Mehul-Kumar-27 Nov 21, 2024
80f59a3
otel version fix
Mehul-Kumar-27 Nov 21, 2024
9782e7e
replication fix
Mehul-Kumar-27 Nov 25, 2024
fc24959
Merge branch 'satellite-state-fetcher' into container
Mehul-Kumar-27 Nov 25, 2024
b9aa9c8
replication fix
Mehul-Kumar-27 Nov 25, 2024
a6e0e10
adds unwanted binary files to gitignore
Mehul-Kumar-27 Nov 28, 2024
20578ff
configurable cron schedule, local registry auth and fixing memory leaks
Mehul-Kumar-27 Nov 30, 2024
a969873
testing merge
Mehul-Kumar-27 Nov 30, 2024
a3a4233
fixing merge conflicts
Mehul-Kumar-27 Nov 30, 2024
af01828
panic fix
Mehul-Kumar-27 Dec 3, 2024
13cd2ee
pr reviews
Mehul-Kumar-27 Dec 3, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 0 additions & 8 deletions .env

This file was deleted.

3 changes: 2 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -31,4 +31,5 @@ secrets.txt
__debug_bin1949266242

/zot
/runtime
/runtime
/value
1 change: 0 additions & 1 deletion ci/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,6 @@ func (m *HarborSatellite) Service(
AsService()
}


// builds given component from source
func (m *HarborSatellite) build(source *dagger.Directory, component string) *dagger.Directory {
fmt.Printf("Building %s\n", component)
Expand Down
139 changes: 139 additions & 0 deletions cmd/container_runtime/containerd.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,139 @@
package runtime

import (
"fmt"
"os"
"path/filepath"
"strings"

"container-registry.com/harbor-satellite/internal/config"
"container-registry.com/harbor-satellite/internal/utils"
"container-registry.com/harbor-satellite/logger"
"container-registry.com/harbor-satellite/registry"
toml "github.com/pelletier/go-toml"
"github.com/rs/zerolog"
"github.com/spf13/cobra"
)

const (
ContainerDCertPath = "/etc/containerd/certs.d"
DefaultGeneratedTomlName = "config.toml"
ContainerdRuntime = "containerd"
DefaultContainerdConfigPath = "/etc/containerd/config.toml"
DefaultConfigVersion = 2
)

var DefaultContainerDGenPath string

func init() {
cwd, err := os.Getwd()
if err != nil {
fmt.Printf("Error getting current working directory: %v\n", err)
DefaultContainerDGenPath = "/runtime/containerd"
if _, err := os.Stat(DefaultContainerDGenPath); os.IsNotExist(err) {
err := os.MkdirAll(DefaultContainerDGenPath, os.ModePerm)
if err != nil {
fmt.Printf("Error creating default directory: %v\n", err)
}
}
Mehul-Kumar-27 marked this conversation as resolved.
Show resolved Hide resolved
} else {
DefaultContainerDGenPath = filepath.Join(cwd, "runtime/containerd")
}
}

func NewContainerdCommand() *cobra.Command {
var generateConfig bool
var defaultZotConfig registry.DefaultZotConfig
var containerdConfigPath string
var containerDCertPath string

containerdCmd := &cobra.Command{
Use: "containerd",
Short: "Creates the config file for the containerd runtime to fetch the images from the local repository",
PersistentPreRunE: func(cmd *cobra.Command, args []string) error {
return SetupContainerRuntimeCommand(cmd, &defaultZotConfig, DefaultContainerDGenPath)
},
RunE: func(cmd *cobra.Command, args []string) error {
log := logger.FromContext(cmd.Context())
sourceRegistry := config.GetSourceRegistryURL()
satelliteHostConfig := NewSatelliteHostConfig(defaultZotConfig.RemoteURL, sourceRegistry)
if generateConfig {
log.Info().Msg("Generating containerd config file for containerd ...")
log.Info().Msgf("Fetching containerd config from path: %s", containerdConfigPath)
err := GenerateContainerdHostConfig(containerDCertPath, DefaultContainerDGenPath, log, *satelliteHostConfig)
if err != nil {
log.Err(err).Msg("Error generating containerd config")
return fmt.Errorf("could not generate containerd config: %w", err)
}
return GenerateContainerdConfig(log, containerdConfigPath, containerDCertPath)
}
return nil
},
}

containerdCmd.Flags().BoolVarP(&generateConfig, "gen", "g", false, "Generate the containerd config file")
containerdCmd.PersistentFlags().StringVarP(&containerdConfigPath, "path", "p", DefaultContainerdConfigPath, "Path to the containerd config file of the container runtime")
containerdCmd.PersistentFlags().StringVarP(&containerDCertPath, "cert-path", "c", ContainerDCertPath, "Path to the containerd cert directory")
containerdCmd.AddCommand(NewReadConfigCommand(ContainerdRuntime))
return containerdCmd
}

// GenerateContainerdConfig generates the containerd config file for the containerd runtime
// It takes the zot config a logger and the containerd config path
// It reads the containerd config file and adds the local registry to the config file
func GenerateContainerdConfig(log *zerolog.Logger, containerdConfigPath, containerdCertPath string) error {
// First Read the present config file at the configPath
data, err := utils.ReadFile(containerdConfigPath, false)
if err != nil {
if os.IsNotExist(err) {
log.Warn().Msg("Config file does not exist, proceeding with default values")
data = []byte{}
} else {
log.Err(err).Msg("Error reading config file")
return fmt.Errorf("could not read config file: %w", err)
}
}
// Now we marshal the data into the containerd config
containerdConfig := &ContainerdConfigToml{}
err = toml.Unmarshal(data, containerdConfig)
if err != nil {
log.Err(err).Msg("Error unmarshalling config")
return fmt.Errorf("could not unmarshal config: %w", err)
}
// Add the certs.d path to the config
if containerdConfig.Plugins.Cri.Registry.ConfigPath == "" {
containerdConfig.Plugins.Cri.Registry.ConfigPath = containerdCertPath
}
Mehul-Kumar-27 marked this conversation as resolved.
Show resolved Hide resolved
// Set default version
if containerdConfig.Version == 0 {
containerdConfig.Version = DefaultConfigVersion
}
// if config disabled plugins container cri then remove it
if len(containerdConfig.DisabledPlugins) > 0 {
filteredPlugins := make([]string, 0, len(containerdConfig.DisabledPlugins))
for _, plugin := range containerdConfig.DisabledPlugins {
if plugin != "cri" {
filteredPlugins = append(filteredPlugins, plugin)
}
}
containerdConfig.DisabledPlugins = filteredPlugins
}
// ToDo: Find a way to remove the unwanted configuration added to the config file while marshalling
pathToWrite := filepath.Join(DefaultContainerDGenPath, DefaultGeneratedTomlName)
log.Info().Msgf("Writing the containerd config to path: %s", pathToWrite)
// Now we write the config to the file
data, err = toml.Marshal(containerdConfig)
dataStr := string(data)
dataStr = strings.Replace(dataStr, "[plugins]\n", "", 1)
data = []byte(dataStr)
if err != nil {
Comment on lines +125 to +129
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Check for errors immediately after marshalling to prevent using invalid data

The error returned by toml.Marshal should be checked immediately before using the data variable to prevent potential issues if the marshalling fails.

Apply this diff to fix the issue:

 data, err = toml.Marshal(containerdConfig)
+if err != nil {
+	log.Err(err).Msg("Error marshalling config")
+	return fmt.Errorf("could not marshal config: %w", err)
+}
 dataStr := string(data)
 dataStr = strings.Replace(dataStr, "[plugins]\n", "", 1)
 data = []byte(dataStr)
-// Error handling was previously here
-if err != nil {
-	log.Err(err).Msg("Error marshalling config")
-	return fmt.Errorf("could not marshal config: %w", err)
-}
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
data, err = toml.Marshal(containerdConfig)
dataStr := string(data)
dataStr = strings.Replace(dataStr, "[plugins]\n", "", 1)
data = []byte(dataStr)
if err != nil {
data, err = toml.Marshal(containerdConfig)
if err != nil {
log.Err(err).Msg("Error marshalling config")
return fmt.Errorf("could not marshal config: %w", err)
}
dataStr := string(data)
dataStr = strings.Replace(dataStr, "[plugins]\n", "", 1)
data = []byte(dataStr)

log.Err(err).Msg("Error marshalling config")
return fmt.Errorf("could not marshal config: %w", err)
}
err = utils.WriteFile(pathToWrite, data)
if err != nil {
log.Err(err).Msg("Error writing config to file")
return fmt.Errorf("could not write config to file: %w", err)
}
return nil
}
169 changes: 169 additions & 0 deletions cmd/container_runtime/containerd_config.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,169 @@
package runtime

// ContainerdConfigToml provides containerd configuration data for the server
type ContainerdConfigToml struct {
// Version of the config file
Version int `toml:"version,omitempty"`
// Root is the path to a directory where containerd will store persistent data
Root string `toml:"root,omitempty"`
// State is the path to a directory where containerd will store transient data
State string `toml:"state,omitempty"`
// TempDir is the path to a directory where to place containerd temporary files
TempDir string `toml:"temp,omitempty"`
// PluginDir is the directory for dynamic plugins to be stored
//
// Deprecated: Please use proxy or binary external plugins.
PluginDir string `toml:"plugin_dir,omitempty"`
Comment on lines +13 to +16
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Properly annotate deprecated field PluginDir.

The PluginDir field is marked as deprecated, but the deprecation notice should follow Go conventions. The comment should start with "Deprecated:" and be placed immediately before the field declaration. This improves readability and alerts developers to avoid using deprecated fields.

Apply this diff to adjust the deprecation comment:

-	// PluginDir is the directory for dynamic plugins to be stored
-	//
-	// Deprecated: Please use proxy or binary external plugins.
+	// Deprecated: PluginDir is the directory for dynamic plugins to be stored.
+	// Please use proxy or binary external plugins instead.
	PluginDir string `toml:"plugin_dir,omitempty"`
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
// PluginDir is the directory for dynamic plugins to be stored
//
// Deprecated: Please use proxy or binary external plugins.
PluginDir string `toml:"plugin_dir,omitempty"`
// Deprecated: PluginDir is the directory for dynamic plugins to be stored.
// Please use proxy or binary external plugins instead.
PluginDir string `toml:"plugin_dir,omitempty"`

// GRPC configuration settings
GRPC GRPCConfig `toml:"grpc,omitempty"`
// TTRPC configuration settings
TTRPC TTRPCConfig `toml:"ttrpc,omitempty"`
// Debug and profiling settings
Debug Debug `toml:"debug,omitempty"`
// Metrics and monitoring settings
Metrics MetricsConfig `toml:"metrics,omitempty"`
// DisabledPlugins are IDs of plugins to disable. Disabled plugins won't be
// initialized and started.
// DisabledPlugins must use a fully qualified plugin URI.
DisabledPlugins []string `toml:"disabled_plugins,omitempty"`
// RequiredPlugins are IDs of required plugins. Containerd exits if any
// required plugin doesn't exist or fails to be initialized or started.
// RequiredPlugins must use a fully qualified plugin URI.
RequiredPlugins []string `toml:"required_plugins,omitempty"`
// Plugins provides plugin specific configuration for the initialization of a plugin
Plugins PluginsConfig `toml:"plugins,omitempty"`
// OOMScore adjust the containerd's oom score
OOMScore int `toml:"oom_score,omitempty"`
// Cgroup specifies cgroup information for the containerd daemon process
Cgroup CgroupConfig `toml:"cgroup,omitempty"`
// ProxyPlugins configures plugins which are communicated to over GRPC
ProxyPlugins map[string]ProxyPlugin `toml:"proxy_plugins,omitempty"`
// Timeouts specified as a duration
Timeouts map[string]string `toml:"timeouts,omitempty"`
// Imports are additional file path list to config files that can overwrite main config file fields
Imports []string `toml:"imports,omitempty"`
// StreamProcessors configuration
StreamProcessors map[string]StreamProcessor `toml:"stream_processors,omitempty"`
}

type StreamProcessor struct {
// Accepts specific media-types
Accepts []string `toml:"accepts,omitempty"`
// Returns the media-type
Returns string `toml:"returns,omitempty"`
// Path or name of the binary
Path string `toml:"path"`
// Args to the binary
Args []string `toml:"args,omitempty"`
// Environment variables for the binary
Env []string `toml:"env,omitempty"`
}

type GRPCConfig struct {
Address string `toml:"address"`
TCPAddress string `toml:"tcp_address,omitempty"`
TCPTLSCA string `toml:"tcp_tls_ca,omitempty"`
TCPTLSCert string `toml:"tcp_tls_cert,omitempty"`
TCPTLSKey string `toml:"tcp_tls_key,omitempty"`
UID int `toml:"uid,omitempty"`
GID int `toml:"gid,omitempty"`
MaxRecvMsgSize int `toml:"max_recv_message_size,omitempty"`
MaxSendMsgSize int `toml:"max_send_message_size,omitempty"`
}

// TTRPCConfig provides TTRPC configuration for the socket
type TTRPCConfig struct {
Address string `toml:"address"`
UID int `toml:"uid,omitempty"`
GID int `toml:"gid,omitempty"`
}

// Debug provides debug configuration
type Debug struct {
Address string `toml:"address,omitempty"`
UID int `toml:"uid,omitempty"`
GID int `toml:"gid,omitempty"`
Level string `toml:"level,omitempty"`
// Format represents the logging format. Supported values are 'text' and 'json'.
Format string `toml:"format,omitempty"`
}

// MetricsConfig provides metrics configuration
type MetricsConfig struct {
Address string `toml:"address,omitempty"`
GRPCHistogram bool `toml:"grpc_histogram,omitempty"`
}

// CgroupConfig provides cgroup configuration
type CgroupConfig struct {
Path string `toml:"path,omitempty"`
}

// ProxyPlugin provides a proxy plugin configuration
type ProxyPlugin struct {
Type string `toml:"type"`
Address string `toml:"address"`
Platform string `toml:"platform,omitempty"`
Exports map[string]string `toml:"exports,omitempty"`
Capabilities []string `toml:"capabilities,omitempty"`
}

type PluginsConfig struct {
Cri CriConfig `toml:"io.containerd.grpc.v1.cri,omitempty"`
Cgroups MonitorConfig `toml:"io.containerd.monitor.v1.cgroups,omitempty"`
LinuxRuntime interface{} `toml:"io.containerd.runtime.v1.linux,omitempty"`
Scheduler GCSchedulerConfig `toml:"io.containerd.gc.v1.scheduler,omitempty"`
Bolt interface{} `toml:"io.containerd.metadata.v1.bolt,omitempty"`
Task RuntimeV2TaskConfig `toml:"io.containerd.runtime.v2.task,omitempty"`
Opt interface{} `toml:"io.containerd.internal.v1.opt,omitempty"`
Restart interface{} `toml:"io.containerd.internal.v1.restart,omitempty"`
Tracing interface{} `toml:"io.containerd.internal.v1.tracing,omitempty"`
Otlp interface{} `toml:"io.containerd.tracing.processor.v1.otlp,omitempty"`
Aufs interface{} `toml:"io.containerd.snapshotter.v1.aufs,omitempty"`
Btrfs interface{} `toml:"io.containerd.snapshotter.v1.btrfs,omitempty"`
Devmapper interface{} `toml:"io.containerd.snapshotter.v1.devmapper,omitempty"`
Native interface{} `toml:"io.containerd.snapshotter.v1.native,omitempty"`
Overlayfs interface{} `toml:"io.containerd.snapshotter.v1.overlayfs,omitempty"`
Zfs interface{} `toml:"io.containerd.snapshotter.v1.zfs,omitempty"`
}

type MonitorConfig struct {
NoPrometheus bool `toml:"no_prometheus,omitempty"`
}

type GCSchedulerConfig struct {
PauseThreshold float64 `toml:"pause_threshold,omitempty"`
DeletionThreshold int `toml:"deletion_threshold,omitempty"`
MutationThreshold int `toml:"mutation_threshold,omitempty"`
ScheduleDelay string `toml:"schedule_delay,omitempty"`
StartupDelay string `toml:"startup_delay,omitempty"`
}

type RuntimeV2TaskConfig struct {
Platforms []string `toml:"platforms,omitempty"`
SchedCore bool `toml:"sched_core,omitempty"`
}

type CriConfig struct {
Containerd CriContainerdConfig `toml:"containerd,omitempty"`
Registry RegistryConfig `toml:"registry,omitempty"`
}

type CriContainerdConfig struct {
DefaultRuntimeName string `toml:"default_runtime_name,omitempty"`
Runtimes map[string]RuntimeConfig `toml:"runtimes,omitempty"`
}

type RuntimeConfig struct {
PrivilegedWithoutHostDevices bool `toml:"privileged_without_host_devices,omitempty"`
RuntimeType string `toml:"runtime_type"`
Options RuntimeOptions `toml:"options,omitempty"`
}

type RuntimeOptions struct {
BinaryName string `toml:"BinaryName,omitempty"`
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Inconsistent TOML tag casing for BinaryName.

The TOML tag BinaryName uses camel case, whereas other TOML tags use snake case (e.g., runtime_type, default_runtime_name). For consistency and to follow standard naming conventions, consider changing the TOML tag to binary_name.

Apply this diff to correct the TOML tag:

	PrivilegedWithoutHostDevices bool           `toml:"privileged_without_host_devices,omitempty"`
	RuntimeType                  string         `toml:"runtime_type"`
-	Options                      RuntimeOptions `toml:"options,omitempty"`
+	Options                      RuntimeOptions `toml:"options,omitempty"`

-	BinaryName string `toml:"BinaryName,omitempty"`
+	BinaryName string `toml:"binary_name,omitempty"`

Committable suggestion skipped: line range outside the PR's diff.

}

type RegistryConfig struct {
ConfigPath string `toml:"config_path,omitempty"`
}
Loading
Loading