/* Copyright 2023 The KubeSphere Authors. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package modules import ( "context" "github.com/cockroachdb/errors" kkprojectv1 "github.com/kubesphere/kubekey/api/project/v1" "k8s.io/apimachinery/pkg/runtime" utilruntime "k8s.io/apimachinery/pkg/util/runtime" "github.com/kubesphere/kubekey/v4/pkg/converter/tmpl" "github.com/kubesphere/kubekey/v4/pkg/variable" ) /* The Assert module evaluates boolean conditions and returns success/failure messages. This module allows users to perform assertions in playbooks and control flow based on conditions. Configuration: Users can specify conditions to evaluate and customize success/failure messages: assert: that: # List of conditions to evaluate (required) - "{{ condition1 }}" - "{{ condition2 }}" success_msg: "Success" # optional: message to return on success (default: "True") fail_msg: "Failed" # optional: high priority failure message msg: "Error" # optional: fallback failure message (default: "False") Usage Examples in Playbook Tasks: 1. Basic assertion: ```yaml - name: Check if service is running assert: that: - "{{ service_status == 'running' }}" register: check_result ``` 2. Custom messages: ```yaml - name: Verify deployment assert: that: - "{{ deployment_ready }}" - "{{ pods_running }}" success_msg: "Deployment is healthy" fail_msg: "Deployment check failed" register: verify_result ``` 3. Multiple conditions: ```yaml - name: Validate configuration assert: that: - "{{ config_valid }}" - "{{ required_fields_present }}" - "{{ values_in_range }}" msg: "Configuration validation failed" register: config_check ``` Return Values: - On success: Returns success_msg (or "True") in stdout - On failure: Returns fail_msg (or msg or "False") in stderr */ type assertArgs struct { that []string successMsg string failMsg string // high priority than msg msg string } func newAssertArgs(_ context.Context, raw runtime.RawExtension, vars map[string]any) (*assertArgs, error) { var err error var aa = &assertArgs{} args := variable.Extension2Variables(raw) if aa.that, err = variable.StringSliceVar(vars, args, "that"); err != nil { return nil, errors.New("\"that\" should be []string or string") } for i, s := range aa.that { if !kkprojectv1.IsTmplSyntax(s) { aa.that[i] = kkprojectv1.ParseTmplSyntax(s) } } aa.successMsg, _ = variable.StringVar(vars, args, "success_msg") if aa.successMsg == "" { aa.successMsg = StdoutSuccess } aa.failMsg, _ = variable.StringVar(vars, args, "fail_msg") if aa.failMsg == "" { aa.failMsg = StdoutFailed } aa.msg, _ = variable.StringVar(vars, args, "msg") if aa.msg == "" { aa.msg = StdoutFailed } return aa, nil } // ModuleAssert handles the "assert" module, evaluating boolean conditions and returning appropriate messages func ModuleAssert(ctx context.Context, options ExecOptions) (string, string, error) { // get host variable ha, err := options.getAllVariables() if err != nil { return StdoutFailed, StderrGetHostVariable, err } aa, err := newAssertArgs(ctx, options.Args, ha) if err != nil { return StdoutFailed, StderrParseArgument, err } ok, err := tmpl.ParseBool(ha, aa.that...) if err != nil { return StdoutFailed, "failed to parse argument of that", err } // condition is true if ok { return aa.successMsg, "", nil } // condition is false and fail_msg is not empty if aa.failMsg != "" { return StdoutFailed, aa.failMsg, errors.New(aa.failMsg) } // condition is false and msg is not empty if aa.msg != "" { return StdoutFailed, aa.msg, errors.New(aa.msg) } return StdoutFailed, StdoutFailed, errors.New(StdoutFailed) } func init() { utilruntime.Must(RegisterModule("assert", ModuleAssert)) }