/*
Copyright 2023 ceph-csi 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 healthchecker

import (
	"errors"
	"os"
	"path"
	"time"
)

type fileChecker struct {
	checker

	// filename contains the filename that is used for checking.
	filename string
}

func newFileChecker(dir string) ConditionChecker {
	fc := &fileChecker{
		filename: path.Join(dir, "csi-volume-condition.ts"),
	}
	fc.initDefaults()

	fc.checker.runChecker = func() {
		fc.isRunning = true

		ticker := time.NewTicker(fc.interval)
		defer ticker.Stop()

		for {
			select {
			case <-fc.commands: // STOP command received
				fc.isRunning = false

				return
			case now := <-ticker.C:
				err := fc.writeTimestamp(now)
				if err != nil {
					fc.mutex.Lock()
					fc.healthy = false
					fc.err = err
					fc.mutex.Unlock()

					continue
				}

				ts, err := fc.readTimestamp()
				if err != nil {
					fc.mutex.Lock()
					fc.healthy = false
					fc.err = err
					fc.mutex.Unlock()

					continue
				}

				// verify that the written timestamp is read back
				if now.Compare(ts) != 0 {
					fc.mutex.Lock()
					fc.healthy = false
					fc.err = errors.New("timestamp read from file does not match what was written")
					fc.mutex.Unlock()

					continue
				}

				// run health check, write a timestamp to a file, read it back
				fc.mutex.Lock()
				fc.healthy = true
				fc.err = nil
				fc.lastUpdate = ts
				fc.mutex.Unlock()
			}
		}
	}

	return fc
}

// readTimestamp reads the JSON formatted timestamp from the file.
func (fc *fileChecker) readTimestamp() (time.Time, error) {
	var ts time.Time

	data, err := os.ReadFile(fc.filename)
	if err != nil {
		return ts, err
	}

	err = ts.UnmarshalJSON(data)

	return ts, err
}

// writeTimestamp writes the timestamp to the file in JSON format.
func (fc *fileChecker) writeTimestamp(ts time.Time) error {
	data, err := ts.MarshalJSON()
	if err != nil {
		return err
	}

	//nolint:gosec // allow reading of the timestamp for debugging
	return os.WriteFile(fc.filename, data, 0o644)
}