1
0
mirror of https://github.com/ManyakRus/starter.git synced 2025-11-24 22:53:52 +02:00
Files
starter/micro/microfunctions.go

1662 lines
37 KiB
Go
Raw Normal View History

2023-03-17 11:20:09 +03:00
// модуль с вспомогательными небольшими функциями
package micro
import (
2024-04-10 13:12:11 +03:00
"bytes"
2023-05-02 09:44:46 +03:00
"context"
2024-04-10 13:12:11 +03:00
"encoding/gob"
2023-03-17 11:20:09 +03:00
"errors"
"fmt"
2025-04-23 13:56:09 +03:00
"github.com/ManyakRus/starter/constants"
2025-04-23 14:47:16 +03:00
"github.com/dromara/carbon/v2"
"github.com/google/uuid"
2024-10-28 11:07:06 +03:00
"golang.org/x/exp/constraints"
"google.golang.org/protobuf/types/known/timestamppb"
2023-07-20 15:54:11 +03:00
"hash/fnv"
2024-10-11 14:48:02 +03:00
"os/exec"
2023-07-20 15:54:11 +03:00
"reflect"
2023-03-17 11:20:09 +03:00
"runtime"
"sort"
2023-07-20 15:54:11 +03:00
"strconv"
2023-03-17 11:20:09 +03:00
"strings"
"unicode"
//"log"
"os"
"path/filepath"
"time"
)
2025-02-12 15:12:33 +03:00
//// Time - тип для хранения времени
//type Time time.Time
2023-03-17 11:20:09 +03:00
//var log = logger.GetLog()
2025-04-23 14:47:16 +03:00
func init() {
//время всегда московское (из константы)
carbon.SetLocation(constants.Loc)
}
2023-03-17 11:20:09 +03:00
// IsTestApp - возвращает true если это тестовая среда выполнения приложения
func IsTestApp() bool {
Otvet := true
stage, ok := os.LookupEnv("STAGE")
if ok == false {
panic(fmt.Errorf("Not found Env 'STAGE' !"))
}
switch stage {
case "local", "dev", "test", "preprod":
Otvet = true
case "prod":
Otvet = false
default:
panic(fmt.Errorf("Error, unknown stage(%v) !", stage))
}
return Otvet
}
// FileExists - возвращает true если файл существует
func FileExists(name string) (bool, error) {
_, err := os.Stat(name)
if err == nil {
return true, nil
}
if errors.Is(err, os.ErrNotExist) {
return false, nil
}
return false, err
}
// AddSeparator - добавляет в конец строки сеператор "/", если его там нет
func AddSeparator(dir string) string {
otvet := dir
if otvet == "" {
return SeparatorFile()
}
if otvet[len(otvet)-1:] != SeparatorFile() {
otvet = otvet + SeparatorFile()
}
return otvet
}
// SeparatorFile - возвращает символ сепаратора каталогов= / или \
func SeparatorFile() string {
return string(filepath.Separator)
}
// Sleep - приостановка работы программы на нужное число миллисекунд
func Sleep(ms int) {
duration := time.Duration(ms) * time.Millisecond
time.Sleep(duration)
}
// Pause - приостановка работы программы на нужное число миллисекунд
func Pause(ms int) {
Sleep(ms)
}
2024-09-27 10:04:44 +03:00
// Pause_ctx - приостановка работы программы на нужное число миллисекунд, с учётом глобального контекста
func Pause_ctx(ctx context.Context, ms int) {
Duration := time.Duration(ms) * time.Millisecond
select {
case <-ctx.Done():
case <-time.After(Duration):
}
}
2025-03-18 15:23:24 +03:00
// Pause_duration - приостановка работы программы на время duration
func Pause_duration(duration time.Duration) {
time.Sleep(duration)
}
// Pause_duration_ctx - приостановка работы программы на время duration, с учётом глобального контекста
func Pause_duration_ctx(ctx context.Context, duration time.Duration) {
select {
case <-ctx.Done():
case <-time.After(duration):
}
}
2023-03-17 11:20:09 +03:00
// FindDirUp - возвращает строку с именем каталога на уровень выше
func FindDirUp(dir string) string {
otvet := dir
if dir == "" {
return otvet
}
if otvet[len(otvet)-1:] == SeparatorFile() {
otvet = otvet[:len(otvet)-1]
}
pos1 := strings.LastIndex(otvet, SeparatorFile())
if pos1 > 0 {
otvet = otvet[0 : pos1+1]
}
return otvet
}
// ErrorJoin - возвращает ошибку из объединения текста двух ошибок
func ErrorJoin(err1, err2 error) error {
var err error
if err1 == nil && err2 == nil {
} else if err1 == nil {
err = err2
} else if err2 == nil {
err = err1
} else {
err = errors.New(err1.Error() + ", " + err2.Error())
}
return err
}
// SubstringLeft - возвращает левые символы строки
func SubstringLeft(str string, num int) string {
if num <= 0 {
return ``
}
if num > len(str) {
num = len(str)
}
return str[:num]
}
// SubstringRight - возвращает правые символы строки
func SubstringRight(str string, num int) string {
if num <= 0 {
return ``
}
max := len(str)
if num > max {
num = max
}
num = max - num
return str[num:]
}
// StringBetween - GetStringInBetween Returns empty string if no start string found
func StringBetween(str string, start string, end string) string {
otvet := ""
if str == "" {
return otvet
}
pos1 := strings.Index(str, start)
if pos1 == -1 {
return otvet
}
pos1 += len(start)
pos2 := strings.Index(str[pos1:], end)
if pos2 == -1 {
return otvet
}
pos2 = pos1 + pos2
otvet = str[pos1:pos2]
return otvet
}
// LastWord - возвращает последнее слово из строки
func LastWord(StringFrom string) string {
Otvet := ""
if StringFrom == "" {
return Otvet
}
r := []rune(StringFrom)
2024-05-27 16:10:39 +03:00
for f := len(r); f > 0; f-- {
2023-03-17 11:20:09 +03:00
r1 := r[f-1]
if r1 == '_' {
} else if unicode.IsLetter(r1) == false && unicode.IsDigit(r1) == false {
break
}
Otvet = string(r1) + Otvet
}
return Otvet
}
// CurrentFilename - возвращает полное имя текущего исполняемого файла
func CurrentFilename() string {
_, filename, _, _ := runtime.Caller(0)
return filename
}
// ProgramDir - возвращает главный каталог программы, в конце "/"
func ProgramDir_Common() string {
2023-09-13 13:48:19 +03:00
//filename := os.Args[0]
filename, err := os.Executable()
if err != nil {
panic(err)
}
2023-03-17 11:20:09 +03:00
dir := filepath.Dir(filename)
sdir := strings.ToLower(dir)
2023-04-25 17:28:10 +03:00
substr := "/tmp/"
pos1 := strings.Index(sdir, substr)
if pos1 >= 0 {
2023-06-01 18:02:36 +03:00
//linux
2023-03-17 11:20:09 +03:00
filename = CurrentFilename()
dir = filepath.Dir(filename)
substr := SeparatorFile() + "vendor" + SeparatorFile()
pos_vendor := strings.Index(strings.ToLower(dir), substr)
if pos_vendor >= 0 {
dir = dir[0:pos_vendor]
} else if dir[len(dir)-5:] == "micro" {
dir = FindDirUp(dir)
//dir = FindDirUp(dir)
//dir = FindDirUp(dir)
}
2023-06-01 18:02:36 +03:00
} else {
//Windows
substr = "\\temp\\"
pos1 = strings.Index(sdir, substr)
if pos1 >= 0 {
filename = CurrentFilename()
dir = filepath.Dir(filename)
substr := SeparatorFile() + "vendor" + SeparatorFile()
pos_vendor := strings.Index(strings.ToLower(dir), substr)
if pos_vendor >= 0 {
dir = dir[0:pos_vendor]
} else if dir[len(dir)-5:] == "micro" {
dir = FindDirUp(dir)
//dir = FindDirUp(dir)
//dir = FindDirUp(dir)
}
2023-05-27 19:26:23 +03:00
}
}
2023-03-17 11:20:09 +03:00
//dir, err := os.Getwd()
//if err != nil {
// log.Fatalln(err)
// dir = ""
//}
dir = AddSeparator(dir)
return dir
}
// ProgramDir - возвращает главный каталог программы, в конце "/"
func ProgramDir() string {
Otvet := ProgramDir_Common()
return Otvet
}
// FileNameWithoutExtension - возвращает имя файла без расширения
func FileNameWithoutExtension(fileName string) string {
return strings.TrimSuffix(fileName, filepath.Ext(fileName))
}
func BeginningOfMonth(date time.Time) time.Time {
return date.AddDate(0, 0, -date.Day()+1)
}
func EndOfMonth(date time.Time) time.Time {
Otvet := date.AddDate(0, 1, -date.Day())
return Otvet
//return date.AddDate(0, 1, -date.Day())
}
//// GetPackageName - возвращает имя пакета
//func GetPackageName(temp interface{}) string {
// strs := strings.Split((runtime.FuncForPC(reflect.ValueOf(temp).Pointer()).Name()), ".")
// strs = strings.Split(strs[len(strs)-2], "/")
// return strs[len(strs)-1]
//}
// StringAfter - возвращает строку, начиная после субстроки StringAfter
func StringAfter(StringFull, StringAfter string) string {
Otvet := StringFull
pos1 := strings.Index(StringFull, StringAfter)
if pos1 == -1 {
return Otvet
}
Otvet = Otvet[pos1+len(StringAfter):]
return Otvet
}
// StringFrom - возвращает строку, начиная со субстроки StringAfter
func StringFrom(StringFull, StringAfter string) string {
Otvet := StringFull
pos1 := strings.Index(StringFull, StringAfter)
if pos1 == -1 {
return Otvet
}
Otvet = Otvet[pos1:]
return Otvet
}
func Trim(s string) string {
Otvet := ""
Otvet = strings.Trim(s, " \n\r\t")
return Otvet
}
2023-04-25 17:28:10 +03:00
2025-01-24 11:06:24 +03:00
// Max returns the largest value
func Max(Mass ...int) int {
var Otvet int
//
if len(Mass) == 0 {
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
//
Otvet = Mass[0]
for _, val := range Mass {
if val > Otvet {
Otvet = val
}
}
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
// Min returns the smallest value
func Min(Mass ...int) int {
var Otvet int
//
if len(Mass) == 0 {
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
//
Otvet = Mass[0]
for _, val := range Mass {
if val < Otvet {
Otvet = val
}
}
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
// MaxInt64 returns the largest value
func MaxInt64(Mass ...int64) int64 {
var Otvet int64
//
if len(Mass) == 0 {
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
//
Otvet = Mass[0]
for _, val := range Mass {
if val > Otvet {
Otvet = val
}
}
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
// MinInt64 returns the smallest value
func MinInt64(Mass ...int64) int64 {
var Otvet int64
//
if len(Mass) == 0 {
return Otvet
2023-04-25 17:28:10 +03:00
}
2025-01-24 11:06:24 +03:00
//
Otvet = Mass[0]
for _, val := range Mass {
if val < Otvet {
Otvet = val
}
}
return Otvet
2023-04-25 17:28:10 +03:00
}
2023-05-02 09:44:46 +03:00
2025-04-09 16:49:11 +03:00
// MaxFloat64 returns the largest value
func MaxFloat64(Mass ...float64) float64 {
var Otvet float64
//
if len(Mass) == 0 {
return Otvet
}
//
Otvet = Mass[0]
for _, val := range Mass {
if val > Otvet {
Otvet = val
}
}
return Otvet
}
// MinFloat64 returns the smallest value
func MinFloat64(Mass ...float64) float64 {
var Otvet float64
//
if len(Mass) == 0 {
return Otvet
}
//
Otvet = Mass[0]
for _, val := range Mass {
if val < Otvet {
Otvet = val
}
}
return Otvet
}
2023-05-19 16:05:54 +03:00
// MaxDate returns the largest of x or y.
func MaxDate(x, y time.Time) time.Time {
if x.Before(y) == true {
return y
}
return x
}
// MinDate returns the smallest of x or y.
func MinDate(x, y time.Time) time.Time {
if x.Before(y) == false {
return y
}
return x
}
2023-05-02 09:44:46 +03:00
// GoGo - запускает функцию в отдельном потоке
func GoGo(ctx context.Context, fn func() error) error {
var err error
chanErr := make(chan error)
go gogo_chan(fn, chanErr)
select {
case <-ctx.Done():
Text1 := "error: TimeOut"
err = errors.New(Text1)
return err
case err = <-chanErr:
//print("err: ", err)
break
}
return err
}
// gogo_chan - запускает функцию и возвращает ошибку в поток
// только совместно с GoGo()
func gogo_chan(fn func() error, chanErr chan error) {
err := fn()
chanErr <- err
}
2023-06-06 18:05:30 +03:00
2023-07-20 15:54:11 +03:00
// CheckInnKpp - проверяет правильность ИНН и КПП
func CheckInnKpp(Inn, Kpp string, is_individual bool) error {
2023-06-06 18:05:30 +03:00
2023-07-20 15:54:11 +03:00
var err error
if Inn == "" {
Text1 := "ИНН не должен быть пустой"
err = errors.New(Text1)
return err
}
if is_individual == true {
if len(Inn) != 12 {
Text1 := "Длина ИНН должна быть 12 символов"
err = errors.New(Text1)
return err
}
if len(Kpp) != 0 {
Text1 := "КПП должен быть пустой"
err = errors.New(Text1)
return err
}
} else {
if len(Inn) != 10 {
Text1 := "Длина ИНН должна быть 10 символов"
err = errors.New(Text1)
return err
}
if len(Kpp) != 9 {
Text1 := "КПП должен быть 9 символов"
err = errors.New(Text1)
return err
}
err = CheckINNControlSum(Inn)
}
return err
}
// CheckINNControlSum - проверяет правильность ИНН по контрольной сумме
func CheckINNControlSum(Inn string) error {
var err error
if len(Inn) == 10 {
err = CheckINNControlSum10(Inn)
} else if len(Inn) == 12 {
err = CheckINNControlSum12(Inn)
} else {
err = errors.New("ИНН должен быть 10 или 12 символов")
}
return err
}
// CheckINNControlSum10 - проверяет правильность 10-значного ИНН по контрольной сумме
func CheckINNControlSum10(Inn string) error {
var err error
MassKoef := [10]int{2, 4, 10, 3, 5, 9, 4, 6, 8, 0}
var sum int
var x int
for i, _ := range Inn {
s := Inn[i : i+1]
var err1 error
x, err1 = strconv.Atoi(s)
if err1 != nil {
err = errors.New("Неправильная цифра в ИНН: " + s)
return err
}
sum = sum + x*MassKoef[i]
}
ControlSum := sum % 11
ControlSum = ControlSum % 10
if ControlSum != x {
err = errors.New("Неправильная контрольная сумма ИНН")
return err
}
return err
}
// CheckINNControlSum2 - проверяет правильность 12-значного ИНН по контрольной сумме
func CheckINNControlSum12(Inn string) error {
var err error
//контрольное чилос по 11 знакам
MassKoef := [11]int{7, 2, 4, 10, 3, 5, 9, 4, 6, 8, 0}
var sum int
var x11 int
for i := 0; i < 11; i++ {
s := Inn[i : i+1]
var err1 error
x, err1 := strconv.Atoi(s)
if err1 != nil {
err = errors.New("Неправильная цифра в ИНН: " + s)
return err
}
if i == 10 {
x11 = x
}
sum = sum + x*MassKoef[i]
}
ControlSum := sum % 11
ControlSum = ControlSum % 10
if ControlSum != x11 {
err = errors.New("Неправильная контрольная сумма ИНН")
return err
}
//контрольное чилос по 12 знакам
MassKoef2 := [12]int{3, 7, 2, 4, 10, 3, 5, 9, 4, 6, 8, 0}
var sum2 int
var x12 int
for i := 0; i < 12; i++ {
s := Inn[i : i+1]
var err1 error
x, err1 := strconv.Atoi(s)
if err1 != nil {
err = errors.New("Неправильная цифра в ИНН: " + s)
return err
}
if i == 11 {
x12 = x
}
sum2 = sum2 + x*MassKoef2[i]
}
ControlSum2 := sum2 % 11
ControlSum2 = ControlSum2 % 10
if ControlSum2 != x12 {
err = errors.New("Неправильная контрольная сумма ИНН")
return err
}
return err
}
2024-12-27 14:16:34 +03:00
// StringFromInt64 - возвращает строку из числа int64
2023-07-20 15:54:11 +03:00
func StringFromInt64(i int64) string {
Otvet := ""
Otvet = strconv.FormatInt(i, 10)
return Otvet
}
2024-12-27 14:16:34 +03:00
// StringFromInt32 - возвращает строку из числа int32
func StringFromInt32(i int32) string {
Otvet := ""
Otvet = fmt.Sprintf("%d", i)
return Otvet
}
2023-07-20 15:54:11 +03:00
// StringDate - возвращает строку дата без времени
func StringDate(t time.Time) string {
Otvet := ""
Otvet = t.Format("02.01.2006")
return Otvet
}
2024-11-08 12:55:49 +03:00
// StringDateTime - возвращает строку дата и время, без миллисекунд
func StringDateTime(t time.Time) string {
Otvet := ""
Otvet = t.Format("02.01.2006 15:04:05")
return Otvet
}
2025-04-21 11:05:25 +03:00
// ProgramDir_bin - возвращает каталог "bin" или каталог программы, в конце "/" (или "\")
2023-07-20 15:54:11 +03:00
func ProgramDir_bin() string {
Otvet := ""
dir := ProgramDir()
FileName := dir + "bin" + SeparatorFile()
ok, _ := FileExists(FileName)
if ok == true {
return FileName
}
Otvet = dir
return Otvet
}
// SaveTempFile - записывает массив байт в файл
func SaveTempFile(bytes []byte) string {
Otvet, err := SaveTempFile_err(bytes)
if err != nil {
TextError := fmt.Sprint("SaveTempFile() error: ", err)
print(TextError)
panic(TextError)
}
return Otvet
}
// SaveTempFile_err - записывает массив байт в файл, возвращает ошибку
func SaveTempFile_err(bytes []byte) (string, error) {
Otvet := ""
// create and open a temporary file
f, err := os.CreateTemp("", "") // in Go version older than 1.17 you can use ioutil.TempFile
if err != nil {
return Otvet, err
}
// close and remove the temporary file at the end of the program
defer f.Close()
2023-09-13 13:48:19 +03:00
//defer os.Remove(f.Name())
2023-07-20 15:54:11 +03:00
// write data to the temporary file
if _, err := f.Write(bytes); err != nil {
return Otvet, err
}
Otvet = f.Name()
return Otvet, err
}
// Hash - возвращает число хэш из строки
func Hash(s string) uint32 {
h := fnv.New32a()
h.Write([]byte(s))
return h.Sum32()
}
// TextError - возвращает текст ошибки из error
func TextError(err error) string {
Otvet := ""
if err != nil {
Otvet = err.Error()
}
return Otvet
}
// GetType - возвращает строку тип объекта
func GetType(myvar interface{}) string {
return reflect.TypeOf(myvar).String()
}
2023-09-13 13:48:19 +03:00
// FindFileNameShort - возвращает имя файла(каталога) без пути
func FindFileNameShort(path string) string {
Otvet := ""
if path == "" {
return Otvet
}
Otvet = filepath.Base(path)
return Otvet
}
// CurrentDirectory - возвращает текущую директорию ОС
func CurrentDirectory() string {
Otvet, err := os.Getwd()
if err != nil {
//log.Println(err)
}
return Otvet
}
// BoolFromInt64 - возвращает true если число <>0
func BoolFromInt64(i int64) bool {
Otvet := false
if i != 0 {
Otvet = true
}
return Otvet
}
// BoolFromInt - возвращает true если число <>0
func BoolFromInt(i int) bool {
Otvet := false
if i != 0 {
Otvet = true
}
return Otvet
}
2023-11-10 13:01:08 +03:00
// BoolFromString - возвращает true если строка = true, или =1
func BoolFromString(s string) bool {
Otvet := false
s = strings.TrimLeft(s, " ")
s = strings.TrimRight(s, " ")
s = strings.ToLower(s)
if s == "true" || s == "1" {
Otvet = true
}
return Otvet
}
2023-09-13 13:48:19 +03:00
// DeleteFileSeperator - убирает в конце / или \
func DeleteFileSeperator(dir string) string {
Otvet := dir
len1 := len(Otvet)
if len1 == 0 {
return Otvet
}
LastWord := Otvet[len1-1 : len1]
if LastWord == SeparatorFile() {
Otvet = Otvet[0 : len1-1]
}
return Otvet
}
2023-11-10 13:01:08 +03:00
// CreateFolder - создаёт папку на диске
func CreateFolder(FilenameFull string, FilePermissions uint32) error {
var err error
FileMode1 := os.FileMode(FilePermissions)
if FilePermissions == 0 {
FileMode1 = os.FileMode(0700)
}
if _, err := os.Stat(FilenameFull); errors.Is(err, os.ErrNotExist) {
err := os.Mkdir(FilenameFull, FileMode1)
if err != nil {
return err
}
}
return err
}
// DeleteFolder - создаёт папку на диске
func DeleteFolder(FilenameFull string) error {
var err error
if _, err := os.Stat(FilenameFull); errors.Is(err, os.ErrNotExist) {
return err
}
err = os.RemoveAll(FilenameFull)
if err != nil {
return err
}
return err
}
// ContextDone - возвращает true если контекст завершен
func ContextDone(ctx context.Context) bool {
select {
case <-ctx.Done():
return true
default:
return false
}
}
2023-11-15 11:29:45 +03:00
// StringFromUpperCase - возвращает строку, первая буква в верхнем регистре
func StringFromUpperCase(s string) string {
Otvet := s
if Otvet == "" {
return Otvet
}
Otvet = strings.ToUpper(Otvet[:1]) + Otvet[1:]
return Otvet
}
// StringFromLowerCase - возвращает строку, первая буква в нижнем регистре
func StringFromLowerCase(s string) string {
Otvet := s
if Otvet == "" {
return Otvet
}
Otvet = strings.ToLower(Otvet[:1]) + Otvet[1:]
return Otvet
}
2023-11-17 13:05:14 +03:00
// DeleteEndSlash - убирает в конце / или \
func DeleteEndSlash(Text string) string {
Otvet := Text
if Otvet == "" {
return Otvet
}
LastSymbol := Otvet[len(Otvet)-1:]
if LastSymbol == "/" || LastSymbol == `\` {
Otvet = Otvet[0 : len(Otvet)-1]
}
return Otvet
}
2023-12-19 11:30:26 +03:00
// Int64FromString - возвращает int64 из строки
func Int64FromString(s string) (int64, error) {
var Otvet int64
var err error
Otvet, err = strconv.ParseInt(s, 10, 64)
return Otvet, err
}
2024-02-12 13:29:20 +03:00
// FindLastPos - возвращает позицию последнего вхождения
func FindLastPos(s, TextFind string) int {
Otvet := strings.LastIndex(s, TextFind)
return Otvet
}
// StringFromFloat64_Dimension2 - возвращает строку с 2 знака после запятой
func StringFromFloat64_Dimension2(f float64) string {
Otvet := fmt.Sprintf("%.2f", f)
return Otvet
}
// StringFromFloat32_Dimension2 - возвращает строку с 2 знака после запятой
func StringFromFloat32_Dimension2(f float32) string {
Otvet := fmt.Sprintf("%.2f", f)
return Otvet
}
2024-04-02 11:10:46 +03:00
// StringFromFloat64_Dimension0 - возвращает строку с 0 знаков после запятой
func StringFromFloat64_Dimension0(f float64) string {
Otvet := fmt.Sprintf("%.0f", f)
return Otvet
}
// StringFromFloat32_Dimension0 - возвращает строку с 0 знаков после запятой
func StringFromFloat32_Dimension0(f float32) string {
Otvet := fmt.Sprintf("%.0f", f)
return Otvet
}
// StringFromFloat64_Dimension - возвращает строку с Dimension знаков после запятой
func StringFromFloat64_Dimension(f float64, Dimension int) string {
Otvet := fmt.Sprintf("%."+strconv.Itoa(Dimension)+"f", f)
return Otvet
}
// StringFromFloat32_Dimension - возвращает строку с Dimension знаков после запятой
func StringFromFloat32_Dimension(f float32, Dimension int) string {
Otvet := fmt.Sprintf("%."+strconv.Itoa(Dimension)+"f", f)
return Otvet
}
2024-04-02 11:10:46 +03:00
// ShowTimePassed - показывает время прошедшее с момента старта
// запускать:
// defer micro.ShowTimePassed(time.Now())
func ShowTimePassed(StartAt time.Time) {
2024-05-21 17:50:26 +03:00
fmt.Printf("Time passed: %s\n", time.Since(StartAt))
2024-04-02 11:10:46 +03:00
}
2024-04-10 13:12:11 +03:00
2024-11-29 14:33:29 +03:00
// ShowTimePassed_FormatText - показывает время прошедшее с момента старта
// запускать:
// defer micro.ShowTimePassed(time.Now())
func ShowTimePassed_FormatText(FormatText string, StartAt time.Time) {
fmt.Printf(FormatText, time.Since(StartAt))
}
2024-05-21 17:33:42 +03:00
// ShowTimePassedSeconds - показывает время секунд прошедшее с момента старта
// запускать:
2024-05-21 17:39:47 +03:00
// defer micro.ShowTimePassedSeconds(time.Now())
2024-05-21 17:33:42 +03:00
func ShowTimePassedSeconds(StartAt time.Time) {
2024-05-21 17:50:26 +03:00
fmt.Printf("Time passed: %s\n", time.Since(StartAt).Round(time.Second))
2024-05-21 17:39:47 +03:00
}
// ShowTimePassedMilliSeconds - показывает время миллисекунд прошедшее с момента старта
// запускать:
// defer micro.ShowTimePassedMilliSeconds(time.Now())
func ShowTimePassedMilliSeconds(StartAt time.Time) {
2024-05-21 17:50:26 +03:00
fmt.Printf("Time passed: %s\n", time.Since(StartAt).Round(time.Millisecond))
2024-05-21 17:33:42 +03:00
}
2024-04-10 13:12:11 +03:00
// StructDeepCopy - копирует структуру из src в dist
// dist - обязательно ссылка &
func StructDeepCopy(src, dist interface{}) (err error) {
buf := bytes.Buffer{}
if err = gob.NewEncoder(&buf).Encode(src); err != nil {
return
}
return gob.NewDecoder(&buf).Decode(dist)
}
2024-04-12 15:34:20 +03:00
// IsEmptyValue - возвращает true если значение по умолчанию (0, пустая строка, пустой слайс)
func IsEmptyValue(v any) bool {
rv := reflect.ValueOf(v)
Otvet := !rv.IsValid() || reflect.DeepEqual(rv.Interface(), reflect.Zero(rv.Type()).Interface())
return Otvet
}
// StringIdentifierFromUUID - возвращает строку из UUID
func StringIdentifierFromUUID() string {
Otvet := uuid.New().String()
Otvet = strings.ReplaceAll(Otvet, "-", "")
return Otvet
}
// IndexSubstringMin - возвращает индекс первого вхождения в строке
func IndexSubstringMin(s string, MassSubstr ...string) int {
Otvet := -1
for _, v := range MassSubstr {
Otvet1 := -1
if v != "" {
Otvet1 = strings.Index(s, v)
}
if Otvet1 != -1 && (Otvet1 < Otvet || Otvet == -1) {
Otvet = Otvet1
}
}
return Otvet
}
// IndexSubstringMin2 - возвращает индекс первого вхождения в строке
func IndexSubstringMin2(s string, substr1, substr2 string) int {
Otvet := -1
Otvet1 := -1
Otvet2 := -1
if substr1 != "" {
Otvet1 = strings.Index(s, substr1)
}
if substr2 != "" {
Otvet2 = strings.Index(s, substr2)
}
if Otvet1 != -1 && (Otvet1 < Otvet2 || Otvet2 == -1) {
Otvet = Otvet1
} else {
Otvet = Otvet2
}
return Otvet
}
// SortMapStringInt_Desc - сортирует map по значению, по убыванию
func SortMapStringInt_Desc(values map[string]int) []string {
type kv struct {
Key string
Value int
}
var ss []kv
for k, v := range values {
ss = append(ss, kv{k, v})
}
sort.Slice(ss, func(i, j int) bool {
return ss[i].Value > ss[j].Value
})
ranked := make([]string, len(values))
for i, kv := range ss {
ranked[i] = kv.Key
}
return ranked
}
// IsNilInterface - проверка интерфейса на nil
func IsNilInterface(i any) bool {
iv := reflect.ValueOf(i)
if !iv.IsValid() {
return true
}
switch iv.Kind() {
case reflect.Ptr, reflect.Slice, reflect.Map, reflect.Func, reflect.Interface:
return iv.IsNil()
default:
return false
}
}
2024-07-17 17:11:11 +03:00
// StringFromMassInt64 - преобразование массива int64 в строку
func StringFromMassInt64(A []int64, delim string) string {
var buffer bytes.Buffer
for i := 0; i < len(A); i++ {
s1 := StringFromInt64(A[i])
buffer.WriteString(s1)
if i != len(A)-1 {
buffer.WriteString(delim)
}
}
return buffer.String()
}
2024-09-19 15:54:59 +03:00
// IsInt - проверяет, является ли строка целым числом
func IsInt(s string) bool {
Otvet := false
if s == "" {
return Otvet
}
for _, c := range s {
if !unicode.IsDigit(c) {
return Otvet
}
}
Otvet = true
return Otvet
}
2024-09-26 14:49:25 +03:00
// Int32FromString - возвращает int32 из строки
2024-09-26 15:27:33 +03:00
func Int32FromString(s string) (int32, error) {
var Otvet int32
2024-09-26 14:49:25 +03:00
var err error
2024-09-26 15:27:33 +03:00
Otvet64, err := strconv.ParseInt(s, 10, 32)
if err != nil {
return Otvet, err
}
Otvet = int32(Otvet64)
2024-09-26 14:49:25 +03:00
return Otvet, err
}
2024-10-11 14:48:02 +03:00
// ExecuteShellCommand - выполняет команду в shell, и возвращает строку результата
func ExecuteShellCommand(TextCommand string, args ...string) (string, error) {
Otvet := ""
var err error
MassByte, err := exec.Command(TextCommand, args...).CombinedOutput()
Otvet = string(MassByte)
if err != nil {
return Otvet, err
}
return Otvet, err
}
// DeleteEndEndline - убирает в конце "\n"
func DeleteEndEndline(Text string) string {
Otvet := Text
if Otvet == "" {
return Otvet
}
LastSymbol := Otvet[len(Otvet)-1:]
if LastSymbol == "\n" {
Otvet = Otvet[0 : len(Otvet)-1]
}
return Otvet
}
2024-10-14 14:25:20 +03:00
// Find_Directory_ModifiedTime - возвращает дату последнего изменения в папке internal
func Find_Directory_ModifiedTime(FolderName string) (time.Time, error) {
var Otvet time.Time
var err error
dir := ProgramDir()
dir = dir + FolderName
ok, err := FileExists(dir)
if err != nil {
err = fmt.Errorf("Find_Directory_ModifiedTime() FileExists() error: %w", err)
return Otvet, err
}
if ok == false {
err = fmt.Errorf("Find_Directory_ModifiedTime() FileExists() error: file not exists: %s", dir)
return Otvet, err
}
//найдём дату папки
f, err := os.Open(dir)
if err != nil {
err = fmt.Errorf("Find_Directory_ModifiedTime() os.Open() error: %w", err)
return Otvet, err
}
defer f.Close()
stat, err := f.Stat()
if err != nil {
err = fmt.Errorf("Find_Directory_ModifiedTime() f.Stat() error: %w", err)
return Otvet, err
}
Otvet = stat.ModTime()
return Otvet, err
}
// Show_Repository_Code_ModifiedTime - выводит дату последнего изменения в папках cmd, internal, pkg, vendor
func Show_Repository_Code_ModifiedTime() {
Date, err := Find_Repository_Code_ModifiedTime()
if err != nil {
println("Find_Repository_Code_ModifiedTime() error: ", err.Error())
return
}
if Date.IsZero() {
println("Last repository code modified time: not found")
return
}
println("Last repository code modified time: ", Date.String())
}
// Find_Repository_Code_ModifiedTime - возвращает дату последнего изменения в папках cmd, internal, pkg, vendor
func Find_Repository_Code_ModifiedTime() (time.Time, error) {
var Otvet time.Time
var err error
//cmd
Time_cmd, err := Find_Directory_ModifiedTime("cmd")
if err != nil {
//return Otvet, err
}
//internal
Time_internal, err := Find_Directory_ModifiedTime("internal")
if err != nil {
//return Otvet, err
}
//pkg
Time_pkg, err := Find_Directory_ModifiedTime("pkg")
if err != nil {
//return Otvet, err
}
//vendor
Time_vendor, err := Find_Directory_ModifiedTime("vendor")
if err != nil {
//return Otvet, err
}
//выбираем максимальную дату
Otvet = TimeMax(Time_cmd, Time_internal, Time_pkg, Time_vendor)
return Otvet, err
}
// TimeMax - возвращает максимальную дату
func TimeMax(x time.Time, y ...time.Time) time.Time {
maxTime := x
for _, val := range y {
if val.After(maxTime) {
maxTime = val
}
}
return maxTime
}
// TimeMin - возвращает минимальную дату
func TimeMin(x time.Time, y ...time.Time) time.Time {
minTime := x
for _, val := range y {
if val.Before(minTime) {
minTime = val
}
}
return minTime
}
// Show_Version - выводит версию сервиса на экран
func Show_Version(Version string) {
println("Service version: ", Version)
}
2024-10-28 11:07:06 +03:00
// MassFrom_MapString - сортирует map по названию колонок и возвращает слайс
func MassFrom_MapString[V any](Map map[string]V) []V {
Otvet := make([]V, 0)
//сортировка по названию колонок
keys := make([]string, 0, len(Map))
for k := range Map {
keys = append(keys, k)
}
sort.Strings(keys)
//
for _, key1 := range keys {
Value, ok := Map[key1]
if ok == false {
fmt.Printf("Map[%s] not found\n", key1)
}
Otvet = append(Otvet, Value)
}
return Otvet
}
// SortMass - сортирует слайс
func SortMass[T constraints.Ordered](s []T) {
sort.Slice(s, func(i, j int) bool {
return s[i] < s[j]
})
}
2024-11-29 09:35:31 +03:00
// SortMass_DESC - сортирует слайс, в обратном порядке
func SortMass_DESC[T constraints.Ordered](s []T) {
sort.Slice(s, func(i, j int) bool {
return s[i] > s[j]
})
}
2024-10-28 11:07:06 +03:00
// MassFrom_Map - сортирует map по названию колонок и возвращает слайс
func MassFrom_Map[C constraints.Ordered, V any](Map map[C]V) []V {
Otvet := make([]V, 0)
//сортировка по названию колонок
keys := make([]C, 0, len(Map))
for k := range Map {
keys = append(keys, k)
}
SortMass(keys)
//
for _, key1 := range keys {
Value, ok := Map[key1]
if ok == false {
fmt.Printf("Map[%v] not found\n", key1)
}
Otvet = append(Otvet, Value)
}
return Otvet
}
2024-11-14 14:49:44 +03:00
2024-11-29 09:35:31 +03:00
// MassFrom_Map_DESC - сортирует map по названию колонок и возвращает слайс, с обратной сортировкой
func MassFrom_Map_DESC[C constraints.Ordered, V any](Map map[C]V) []V {
Otvet := make([]V, 0)
//сортировка по названию колонок
keys := make([]C, 0, len(Map))
for k := range Map {
keys = append(keys, k)
}
SortMass_DESC(keys)
//
for _, key1 := range keys {
Value, ok := Map[key1]
if ok == false {
fmt.Printf("Map[%v] not found\n", key1)
}
Otvet = append(Otvet, Value)
}
return Otvet
}
2024-11-14 14:49:44 +03:00
// Substring - take at most last n characters, from start index
func Substring(input string, StartIndex int, length int) string {
2024-11-18 15:06:05 +03:00
//asRunes := []rune(input)
2024-11-14 14:49:44 +03:00
2024-11-18 15:06:05 +03:00
if StartIndex >= len(input) {
2024-11-14 14:49:44 +03:00
return ""
}
2024-11-18 15:06:05 +03:00
if (StartIndex + length) >= len(input) {
length = len(input) - StartIndex
2024-11-14 14:49:44 +03:00
}
2024-11-18 15:06:05 +03:00
//if StartIndex+length > len(asRunes) {
// length = len(asRunes) - StartIndex
//}
Otvet := string(input[StartIndex : StartIndex+length])
return Otvet
2024-11-14 14:49:44 +03:00
}
2024-11-18 13:27:46 +03:00
// IntNot0 - возвращает первое ненулевое значение
func IntNot0(MassInt ...int) int {
Otvet := 0
for _, v := range MassInt {
if v != 0 {
Otvet = v
break
}
}
return Otvet
}
2024-11-18 13:36:44 +03:00
// InsertTextFrom - вставляет текст в середину строки
func InsertTextFrom(Text string, TextAdd string, IndexFrom int) string {
var buffer bytes.Buffer
//
if IndexFrom >= len(Text) {
return Text + TextAdd
}
//
if IndexFrom < 0 {
return TextAdd + Text
}
//
2024-11-18 15:06:05 +03:00
s2 := SubstringLeft(Text, IndexFrom+1)
buffer.WriteString(s2)
2024-11-18 13:36:44 +03:00
buffer.WriteString(TextAdd)
2024-11-18 15:06:05 +03:00
s3 := Substring(Text, IndexFrom, len(Text+TextAdd))
buffer.WriteString(s3)
2024-11-18 13:36:44 +03:00
2024-11-18 15:06:05 +03:00
Otvet := buffer.String()
return Otvet
2024-11-18 13:36:44 +03:00
}
// Date_from_TimestampReference - возвращает дату из *Timestamp
func Date_from_TimestampReference(Timestamp *timestamppb.Timestamp) time.Time {
Otvet := time.Time{}
if Timestamp != nil {
Otvet = Timestamp.AsTime()
}
return Otvet
}
2024-11-28 16:59:40 +03:00
// SetFieldValue - устанавливает значение поля в структуре
// Параметры:
// Object - ссылка(&) на структуру
// FieldName - название поля
// Value - значение нужного типа
// Возвращает ошибку
func SetFieldValue(Object any, FieldName string, Value any) error {
var err error
ref := reflect.ValueOf(Object)
//sanek
if ref.Kind() != reflect.Ptr {
err = fmt.Errorf("expected pointer but got %s", ref.Kind().String())
return err
}
// if its a pointer, resolve its Value
if ref.Kind() == reflect.Ptr {
ref = reflect.Indirect(ref)
}
if ref.Kind() == reflect.Interface {
ref = ref.Elem()
}
//should double check we now have a struct (could still be anything)
if ref.Kind() != reflect.Struct {
err = fmt.Errorf("expected struct but got %s", ref.Kind().String())
return err
}
prop := ref.FieldByName(FieldName)
2024-11-29 14:33:29 +03:00
ValueNew := reflect.ValueOf(Value)
prop.Set(ValueNew)
2024-11-28 16:59:40 +03:00
return err
}
2024-12-09 17:28:19 +03:00
// Float64FromString - возвращает float64 из строки
func Float64FromString(s string) (float64, error) {
var Otvet float64
var err error
Otvet, err = strconv.ParseFloat(s, 64)
return Otvet, err
}
2025-01-14 10:44:22 +03:00
// Abs - возвращает абсолютное значение
func Abs[T constraints.Integer](x T) T {
if x < 0 {
return -x
}
return x
}
2025-01-29 16:58:12 +03:00
// StringFromBool - возвращает строку из булевского значения
func StringFromBool(value bool) string {
Otvet := "true"
if value == false {
Otvet = "false"
}
return Otvet
}
2025-02-12 15:12:33 +03:00
2025-03-14 13:06:41 +03:00
// StringFromBool_Rus - возвращает строку из булевского значения, Да/Нет
func StringFromBool_Rus(value bool) string {
Otvet := "Да"
if value == false {
Otvet = "Нет"
}
return Otvet
}
// StringFromBool_Rus_lower - возвращает строку из булевского значения, да/нет
func StringFromBool_Rus_lower(value bool) string {
Otvet := "да"
if value == false {
Otvet = "нет"
}
return Otvet
}
2025-02-12 15:17:19 +03:00
//// UnmarshalJSON - преобразует строку время в time.Time
//func (d *Time) UnmarshalJSON(b []byte) error {
2025-02-12 15:12:33 +03:00
// str := string(b)
// if str != "" && str[0] == '"' && str[len(str)-1] == '"' {
// str = str[1 : len(str)-1]
// }
//
// // parse string
// t, err := time.ParseInLocation(constants.LayoutTime, str, constants.Loc)
// if err != nil {
// err = fmt.Errorf("invalid time string: %s, error: %w", b, err)
// return err
// }
//
// //
// *d = Time(t)
// return nil
//}
//// UnmarshalString - преобразует строку время в time.Time
//func (d *Time) UnmarshalString(str string) error {
// if str != "" && str[0] == '"' && str[len(str)-1] == '"' {
// str = str[1 : len(str)-1]
// }
//
// // parse string
// t, err := time.Parse(constants.LayoutTime, str)
// if err != nil {
// err = fmt.Errorf("invalid time string: %s, error: %w", str, err)
// return err
// }
//
// //
// *d = Time(t)
// return nil
//}
2025-03-04 14:32:24 +03:00
// IsFalseString - возвращает true если строка = false, или =0
func IsFalseString(s string) bool {
Otvet := false
s = strings.Trim(s, " ")
s = strings.Trim(s, "\n")
s = strings.ToLower(s)
switch s {
case "0", "нет", "no", "off", "false":
Otvet = true
}
return Otvet
}
// IsTrueString - возвращает true если строка = true, или =1
func IsTrueString(s string) bool {
Otvet := false
s = strings.Trim(s, " ")
s = strings.Trim(s, "\n")
s = strings.ToLower(s)
switch s {
case "1", "да", "yes", "on", "true":
Otvet = true
}
return Otvet
}
2025-04-23 13:56:09 +03:00
// DateTimeFromString_rus - возвращает дату из строки, из формата "02.01.2006 15:04:05"
func DateTimeFromString_rus(s string) (time.Time, error) {
2025-04-23 14:47:16 +03:00
t, err := time.ParseInLocation(constants.LayoutDateTimeRus, s, constants.Loc)
2025-04-23 13:56:09 +03:00
return t, err
}
// DateFromString_rus - возвращает дату из строки, из формата "02.01.2006"
func DateFromString_rus(s string) (time.Time, error) {
//
if len(s) > 10 {
s = s[:10]
}
//
2025-04-23 14:47:16 +03:00
t, err := time.ParseInLocation(constants.LayoutDateRus, s, constants.Loc)
2025-04-23 13:56:09 +03:00
return t, err
}
2025-04-23 14:47:16 +03:00
// DateFromToToday_rus - возвращает дату начала и конца дня
func DateFromToToday_rus() (time.Time, time.Time) {
//carbon.SetLocation(constants.Loc)
Date1 := carbon.Now().StartOfDay().StdTime()
Date2 := carbon.CreateFromStdTime(Date1).EndOfDay().StdTime()
return Date1, Date2
}