2022-10-20 18:45:47 +08:00
package service
import (
2022-11-09 15:08:38 +08:00
"bufio"
2022-10-27 23:09:39 +08:00
"compress/gzip"
2022-10-21 18:50:38 +08:00
"encoding/json"
"fmt"
2022-11-09 15:08:38 +08:00
"io"
2022-11-04 19:02:15 +08:00
"io/ioutil"
2022-11-08 19:03:38 +08:00
"mime/multipart"
2022-10-27 23:09:39 +08:00
"os"
2022-10-25 18:34:33 +08:00
"os/exec"
2022-11-08 19:03:38 +08:00
"path/filepath"
2022-11-04 19:02:15 +08:00
"regexp"
2022-10-24 18:46:19 +08:00
"strconv"
2022-10-25 18:34:33 +08:00
"strings"
2022-10-24 18:46:19 +08:00
"time"
2022-10-21 18:50:38 +08:00
2022-10-20 18:45:47 +08:00
"github.com/1Panel-dev/1Panel/backend/app/dto"
2022-10-25 18:34:33 +08:00
"github.com/1Panel-dev/1Panel/backend/app/model"
2022-10-20 18:45:47 +08:00
"github.com/1Panel-dev/1Panel/backend/constant"
2022-10-27 23:09:39 +08:00
"github.com/1Panel-dev/1Panel/backend/global"
2022-11-04 19:02:15 +08:00
"github.com/1Panel-dev/1Panel/backend/utils/compose"
2022-11-09 15:08:38 +08:00
"github.com/1Panel-dev/1Panel/backend/utils/files"
2022-10-21 18:50:38 +08:00
_ "github.com/go-sql-driver/mysql"
2022-10-20 18:45:47 +08:00
"github.com/jinzhu/copier"
"github.com/pkg/errors"
)
type MysqlService struct { }
type IMysqlService interface {
2022-10-25 18:34:33 +08:00
SearchWithPage ( search dto . SearchDBWithPage ) ( int64 , interface { } , error )
2022-10-31 23:52:39 +08:00
ListDBByVersion ( name string ) ( [ ] string , error )
2022-10-28 18:46:14 +08:00
SearchBackupsWithPage ( search dto . SearchBackupsWithPage ) ( int64 , interface { } , error )
2022-10-20 18:45:47 +08:00
Create ( mysqlDto dto . MysqlDBCreate ) error
2022-10-24 18:46:19 +08:00
ChangeInfo ( info dto . ChangeDBInfo ) error
2022-11-04 19:02:15 +08:00
UpdateVariables ( mysqlName string , updatas [ ] dto . MysqlVariablesUpdate ) error
2022-11-09 15:08:38 +08:00
UpdateConfByFile ( info dto . MysqlConfUpdateByFile ) error
2022-10-27 23:09:39 +08:00
2022-11-08 19:03:38 +08:00
UpFile ( mysqlName string , files [ ] * multipart . FileHeader ) error
2022-11-09 15:08:38 +08:00
RecoverByUpload ( req dto . UploadRecover ) error
2022-11-08 19:03:38 +08:00
SearchUpListWithPage ( req dto . SearchDBWithPage ) ( int64 , interface { } , error )
2022-10-27 23:09:39 +08:00
Backup ( db dto . BackupDB ) error
Recover ( db dto . RecoverDB ) error
2022-10-31 23:52:39 +08:00
Delete ( name string , ids [ ] uint ) error
LoadStatus ( name string ) ( * dto . MysqlStatus , error )
LoadVariables ( vernamesion string ) ( * dto . MysqlVariables , error )
2022-10-25 18:34:33 +08:00
LoadRunningVersion ( ) ( [ ] string , error )
2022-10-31 23:52:39 +08:00
LoadBaseInfo ( name string ) ( * dto . DBBaseInfo , error )
2022-10-20 18:45:47 +08:00
}
func NewIMysqlService ( ) IMysqlService {
return & MysqlService { }
}
2022-10-25 18:34:33 +08:00
func ( u * MysqlService ) SearchWithPage ( search dto . SearchDBWithPage ) ( int64 , interface { } , error ) {
2022-10-31 23:52:39 +08:00
total , mysqls , err := mysqlRepo . Page ( search . Page , search . PageSize , mysqlRepo . WithByMysqlName ( search . MysqlName ) )
2022-10-20 18:45:47 +08:00
var dtoMysqls [ ] dto . MysqlDBInfo
for _ , mysql := range mysqls {
var item dto . MysqlDBInfo
if err := copier . Copy ( & item , & mysql ) ; err != nil {
return 0 , nil , errors . WithMessage ( constant . ErrStructTransform , err . Error ( ) )
}
dtoMysqls = append ( dtoMysqls , item )
}
return total , dtoMysqls , err
}
2022-11-08 19:03:38 +08:00
func ( u * MysqlService ) SearchUpListWithPage ( req dto . SearchDBWithPage ) ( int64 , interface { } , error ) {
var (
2022-11-09 15:08:38 +08:00
list [ ] dto . DatabaseFileRecords
backDatas [ ] dto . DatabaseFileRecords
2022-11-08 19:03:38 +08:00
)
2022-11-09 15:08:38 +08:00
localDir , appKey , err := loadBackupDirAndKey ( req . MysqlName )
2022-11-08 19:03:38 +08:00
if err != nil {
2022-11-09 15:08:38 +08:00
return 0 , list , nil
2022-11-08 19:03:38 +08:00
}
2022-11-09 15:08:38 +08:00
uploadDir := fmt . Sprintf ( "%s/database/%s/%s/upload" , localDir , appKey , req . MysqlName )
2022-11-08 19:03:38 +08:00
if _ , err := os . Stat ( uploadDir ) ; err != nil {
return 0 , list , nil
}
_ = filepath . Walk ( uploadDir , func ( path string , info os . FileInfo , err error ) error {
2022-11-09 15:08:38 +08:00
if err != nil {
return nil
}
2022-11-08 19:03:38 +08:00
if ! info . IsDir ( ) {
2022-11-09 15:08:38 +08:00
list = append ( list , dto . DatabaseFileRecords {
2022-11-08 19:03:38 +08:00
CreatedAt : info . ModTime ( ) . Format ( "2006-01-02 15:04:05" ) ,
Size : int ( info . Size ( ) ) ,
FileDir : uploadDir ,
FileName : info . Name ( ) ,
} )
}
return nil
} )
total , start , end := len ( list ) , ( req . Page - 1 ) * req . PageSize , req . Page * req . PageSize
if start > total {
2022-11-09 15:08:38 +08:00
backDatas = make ( [ ] dto . DatabaseFileRecords , 0 )
2022-11-08 19:03:38 +08:00
} else {
if end >= total {
end = total
}
backDatas = list [ start : end ]
}
return int64 ( total ) , backDatas , nil
}
func ( u * MysqlService ) UpFile ( mysqlName string , files [ ] * multipart . FileHeader ) error {
2022-11-09 15:08:38 +08:00
localDir , appKey , err := loadBackupDirAndKey ( mysqlName )
2022-11-08 19:03:38 +08:00
if err != nil {
return err
}
2022-11-09 15:08:38 +08:00
dstDir := fmt . Sprintf ( "%s/database/%s/%s/upload" , localDir , appKey , mysqlName )
2022-11-08 19:03:38 +08:00
if _ , err := os . Stat ( dstDir ) ; err != nil && os . IsNotExist ( err ) {
if err = os . MkdirAll ( dstDir , os . ModePerm ) ; err != nil {
if err != nil {
return fmt . Errorf ( "mkdir %s failed, err: %v" , dstDir , err )
}
}
}
for _ , file := range files {
src , err := file . Open ( )
if err != nil {
return err
}
defer src . Close ( )
out , err := os . Create ( dstDir + "/" + file . Filename )
if err != nil {
return err
}
defer out . Close ( )
2022-11-09 15:08:38 +08:00
_ , _ = io . Copy ( out , src )
}
return nil
}
func ( u * MysqlService ) RecoverByUpload ( req dto . UploadRecover ) error {
app , err := mysqlRepo . LoadBaseInfoByName ( req . MysqlName )
if err != nil {
return err
}
localDir , err := loadLocalDir ( )
if err != nil {
return err
}
file := req . FileDir + "/" + req . FileName
if ! strings . HasSuffix ( req . FileName , ".sql" ) && ! strings . HasSuffix ( req . FileName , ".gz" ) {
fileOp := files . NewFileOp ( )
fileNameItem := time . Now ( ) . Format ( "20060102150405" )
dstDir := fmt . Sprintf ( "%s/database/%s/%s/upload/tmp/%s" , localDir , app . Key , req . MysqlName , fileNameItem )
if _ , err := os . Stat ( dstDir ) ; err != nil && os . IsNotExist ( err ) {
if err = os . MkdirAll ( dstDir , os . ModePerm ) ; err != nil {
if err != nil {
return fmt . Errorf ( "mkdir %s failed, err: %v" , dstDir , err )
}
}
}
var compressType files . CompressType
switch {
case strings . HasSuffix ( req . FileName , ".tar.gz" ) , strings . HasSuffix ( req . FileName , ".tgz" ) :
compressType = files . TarGz
case strings . HasSuffix ( req . FileName , ".zip" ) :
compressType = files . Zip
}
if err := fileOp . Decompress ( req . FileDir + "/" + req . FileName , dstDir , compressType ) ; err != nil {
_ = os . RemoveAll ( dstDir )
return err
}
hasTestSql := false
_ = filepath . Walk ( dstDir , func ( path string , info os . FileInfo , err error ) error {
if err != nil {
return nil
}
if ! info . IsDir ( ) && info . Name ( ) == "test.sql" {
hasTestSql = true
file = path
}
return nil
} )
if ! hasTestSql {
_ = os . RemoveAll ( dstDir )
return fmt . Errorf ( "no such file named test.sql in %s, err: %v" , req . FileName , err )
}
defer func ( ) {
_ = os . RemoveAll ( dstDir )
} ( )
}
fi , _ := os . Open ( file )
defer fi . Close ( )
cmd := exec . Command ( "docker" , "exec" , "-i" , app . ContainerName , "mysql" , "-uroot" , "-p" + app . Password , req . DBName )
if strings . HasSuffix ( req . FileName , ".gz" ) {
gzipFile , err := os . Open ( file )
if err != nil {
return err
}
defer gzipFile . Close ( )
gzipReader , err := gzip . NewReader ( gzipFile )
if err != nil {
return err
}
defer gzipReader . Close ( )
cmd . Stdin = gzipReader
} else {
cmd . Stdin = fi
}
stdout , err := cmd . CombinedOutput ( )
stdStr := strings . ReplaceAll ( string ( stdout ) , "mysql: [Warning] Using a password on the command line interface can be insecure.\n" , "" )
if err != nil || strings . HasPrefix ( string ( stdStr ) , "ERROR " ) {
return errors . New ( stdStr )
2022-11-08 19:03:38 +08:00
}
return nil
}
2022-10-31 23:52:39 +08:00
func ( u * MysqlService ) ListDBByVersion ( name string ) ( [ ] string , error ) {
2022-11-08 19:03:38 +08:00
mysqls , err := mysqlRepo . List ( mysqlRepo . WithByMysqlName ( name ) )
2022-10-28 18:46:14 +08:00
var dbNames [ ] string
for _ , mysql := range mysqls {
dbNames = append ( dbNames , mysql . Name )
}
return dbNames , err
}
func ( u * MysqlService ) SearchBackupsWithPage ( search dto . SearchBackupsWithPage ) ( int64 , interface { } , error ) {
2022-10-31 23:52:39 +08:00
app , err := mysqlRepo . LoadBaseInfoByName ( search . MysqlName )
2022-10-27 23:09:39 +08:00
if err != nil {
return 0 , nil , err
}
searchDto := dto . BackupSearch {
Type : "database-mysql" ,
PageInfo : search . PageInfo ,
Name : app . Name ,
DetailName : search . DBName ,
}
return NewIBackupService ( ) . SearchRecordWithPage ( searchDto )
}
2022-10-25 18:34:33 +08:00
func ( u * MysqlService ) LoadRunningVersion ( ) ( [ ] string , error ) {
2022-10-31 23:52:39 +08:00
return mysqlRepo . LoadRunningVersion ( [ ] string { "Mysql5.7" , "Mysql8.0" } )
2022-10-25 18:34:33 +08:00
}
2022-10-20 18:45:47 +08:00
func ( u * MysqlService ) Create ( mysqlDto dto . MysqlDBCreate ) error {
2022-10-24 18:46:19 +08:00
if mysqlDto . Username == "root" {
return errors . New ( "Cannot set root as user name" )
}
2022-11-09 15:08:38 +08:00
app , err := mysqlRepo . LoadBaseInfoByName ( mysqlDto . MysqlName )
if err != nil {
return err
}
mysql , _ := mysqlRepo . Get ( commonRepo . WithByName ( mysqlDto . Name ) , mysqlRepo . WithByMysqlName ( app . Key ) )
2022-10-20 18:45:47 +08:00
if mysql . ID != 0 {
return constant . ErrRecordExist
}
if err := copier . Copy ( & mysql , & mysqlDto ) ; err != nil {
return errors . WithMessage ( constant . ErrStructTransform , err . Error ( ) )
}
2022-10-25 18:34:33 +08:00
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "create database if not exists %s character set=%s" , mysqlDto . Name , mysqlDto . Format ) ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
tmpPermission := mysqlDto . Permission
2022-11-09 15:08:38 +08:00
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "create user if not exists '%s'@'%s' identified by '%s';" , mysqlDto . Username , tmpPermission , mysqlDto . Password ) ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-25 18:34:33 +08:00
grantStr := fmt . Sprintf ( "grant all privileges on %s.* to '%s'@'%s'" , mysqlDto . Name , mysqlDto . Username , tmpPermission )
2022-10-31 23:52:39 +08:00
if app . Key == "mysql5.7" {
2022-10-25 18:34:33 +08:00
grantStr = fmt . Sprintf ( "%s identified by '%s' with grant option;" , grantStr , mysqlDto . Password )
2022-10-24 18:46:19 +08:00
}
2022-10-25 18:34:33 +08:00
if err := excuteSql ( app . ContainerName , app . Password , grantStr ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-20 18:45:47 +08:00
if err := mysqlRepo . Create ( & mysql ) ; err != nil {
return err
}
return nil
}
2022-10-27 23:09:39 +08:00
func ( u * MysqlService ) Backup ( db dto . BackupDB ) error {
2022-11-09 15:08:38 +08:00
localDir , appKey , err := loadBackupDirAndKey ( db . MysqlName )
2022-11-08 19:03:38 +08:00
if err != nil {
return err
}
2022-11-09 15:08:38 +08:00
backupDir := fmt . Sprintf ( "database/%s/%s/%s" , appKey , db . MysqlName , db . DBName )
2022-10-28 18:46:14 +08:00
fileName := fmt . Sprintf ( "%s_%s.sql.gz" , db . DBName , time . Now ( ) . Format ( "20060102150405" ) )
2022-10-31 23:52:39 +08:00
if err := backupMysql ( "LOCAL" , localDir , backupDir , db . MysqlName , db . DBName , fileName ) ; err != nil {
2022-10-28 18:46:14 +08:00
return err
2022-10-27 23:09:39 +08:00
}
return nil
}
func ( u * MysqlService ) Recover ( db dto . RecoverDB ) error {
2022-10-31 23:52:39 +08:00
app , err := mysqlRepo . LoadBaseInfoByName ( db . MysqlName )
2022-10-27 23:09:39 +08:00
if err != nil {
return err
}
gzipFile , err := os . Open ( db . BackupName )
if err != nil {
2022-11-09 15:08:38 +08:00
return err
2022-10-27 23:09:39 +08:00
}
defer gzipFile . Close ( )
gzipReader , err := gzip . NewReader ( gzipFile )
if err != nil {
2022-11-09 15:08:38 +08:00
return err
2022-10-27 23:09:39 +08:00
}
defer gzipReader . Close ( )
cmd := exec . Command ( "docker" , "exec" , "-i" , app . ContainerName , "mysql" , "-uroot" , "-p" + app . Password , db . DBName )
cmd . Stdin = gzipReader
stdout , err := cmd . CombinedOutput ( )
stdStr := strings . ReplaceAll ( string ( stdout ) , "mysql: [Warning] Using a password on the command line interface can be insecure.\n" , "" )
if err != nil || strings . HasPrefix ( string ( stdStr ) , "ERROR " ) {
return errors . New ( stdStr )
}
return nil
}
2022-10-31 23:52:39 +08:00
func ( u * MysqlService ) Delete ( name string , ids [ ] uint ) error {
app , err := mysqlRepo . LoadBaseInfoByName ( name )
2022-10-24 18:46:19 +08:00
if err != nil {
return err
}
2022-10-25 18:34:33 +08:00
2022-10-24 18:46:19 +08:00
dbs , err := mysqlRepo . List ( commonRepo . WithIdsIn ( ids ) )
if err != nil {
return err
}
for _ , db := range dbs {
if len ( db . Name ) != 0 {
2022-10-25 18:34:33 +08:00
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "drop user if exists '%s'@'%s'" , db . Name , db . Permission ) ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-25 18:34:33 +08:00
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "drop database if exists %s" , db . Name ) ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-20 18:45:47 +08:00
}
2022-10-24 18:46:19 +08:00
_ = mysqlRepo . Delete ( commonRepo . WithByID ( db . ID ) )
2022-10-20 18:45:47 +08:00
}
2022-10-24 18:46:19 +08:00
return nil
2022-10-20 18:45:47 +08:00
}
2022-10-21 18:50:38 +08:00
2022-10-24 18:46:19 +08:00
func ( u * MysqlService ) ChangeInfo ( info dto . ChangeDBInfo ) error {
2022-10-25 18:34:33 +08:00
var (
mysql model . DatabaseMysql
err error
)
if info . ID != 0 {
mysql , err = mysqlRepo . Get ( commonRepo . WithByID ( info . ID ) )
if err != nil {
return err
}
2022-10-24 18:46:19 +08:00
}
2022-10-31 23:52:39 +08:00
app , err := mysqlRepo . LoadBaseInfoByName ( info . MysqlName )
2022-10-24 18:46:19 +08:00
if err != nil {
return err
2022-10-21 18:50:38 +08:00
}
2022-10-24 18:46:19 +08:00
if info . Operation == "password" {
2022-10-25 18:34:33 +08:00
if info . ID != 0 {
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "set password for %s@%s = password('%s')" , mysql . Username , mysql . Permission , info . Value ) ) ; err != nil {
return err
}
_ = mysqlRepo . Update ( mysql . ID , map [ string ] interface { } { "password" : info . Value } )
return nil
}
hosts , err := excuteSqlForRows ( app . ContainerName , app . Password , "select host from mysql.user where user='root';" )
if err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-25 18:34:33 +08:00
for _ , host := range hosts {
if host == "%" || host == "localhost" {
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "set password for root@'%s' = password('%s')" , host , info . Value ) ) ; err != nil {
return err
}
}
}
2022-11-08 14:34:41 +08:00
_ = mysqlRepo . UpdateDatabaseInfo ( app . ID , map [ string ] interface { } {
2022-10-25 18:34:33 +08:00
"param" : strings . ReplaceAll ( app . Param , app . Password , info . Value ) ,
"env" : strings . ReplaceAll ( app . Env , app . Password , info . Value ) ,
} )
2022-10-24 18:46:19 +08:00
return nil
}
2022-10-21 18:50:38 +08:00
2022-10-25 18:34:33 +08:00
if info . ID == 0 {
mysql . Name = "*"
mysql . Username = "root"
mysql . Permission = "%"
mysql . Password = app . Password
}
if info . Value != mysql . Permission {
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "drop user if exists '%s'@'%s'" , mysql . Username , mysql . Permission ) ) ; err != nil {
return err
}
if info . ID == 0 {
return nil
}
}
if err := excuteSql ( app . ContainerName , app . Password , fmt . Sprintf ( "create user if not exists '%s'@'%s' identified by '%s';" , mysql . Username , info . Value , mysql . Password ) ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-25 18:34:33 +08:00
grantStr := fmt . Sprintf ( "grant all privileges on %s.* to '%s'@'%s'" , mysql . Name , mysql . Username , info . Value )
2022-10-31 23:52:39 +08:00
if app . Key == "mysql5.7" {
2022-10-25 18:34:33 +08:00
grantStr = fmt . Sprintf ( "%s identified by '%s' with grant option;" , grantStr , mysql . Password )
2022-10-24 18:46:19 +08:00
}
2022-10-25 18:34:33 +08:00
if err := excuteSql ( app . ContainerName , app . Password , grantStr ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-25 18:34:33 +08:00
if err := excuteSql ( app . ContainerName , app . Password , "flush privileges" ) ; err != nil {
2022-10-24 18:46:19 +08:00
return err
}
2022-10-25 18:34:33 +08:00
if info . ID == 0 {
return nil
}
2022-10-24 18:46:19 +08:00
_ = mysqlRepo . Update ( mysql . ID , map [ string ] interface { } { "permission" : info . Value } )
return nil
}
2022-11-09 15:08:38 +08:00
func ( u * MysqlService ) UpdateConfByFile ( info dto . MysqlConfUpdateByFile ) error {
app , err := mysqlRepo . LoadBaseInfoByName ( info . MysqlName )
if err != nil {
return err
}
path := fmt . Sprintf ( "%s/%s/%s/conf/my.cnf" , constant . AppInstallDir , app . Key , app . Name )
file , err := os . OpenFile ( path , os . O_WRONLY | os . O_TRUNC , 0640 )
if err != nil {
return err
}
defer file . Close ( )
write := bufio . NewWriter ( file )
_ , _ = write . WriteString ( info . File )
write . Flush ( )
if _ , err := compose . Restart ( fmt . Sprintf ( "%s/%s/%s/docker-compose.yml" , constant . AppInstallDir , app . Key , app . Name ) ) ; err != nil {
return err
}
return nil
}
2022-11-04 19:02:15 +08:00
func ( u * MysqlService ) UpdateVariables ( mysqlName string , updatas [ ] dto . MysqlVariablesUpdate ) error {
app , err := mysqlRepo . LoadBaseInfoByName ( mysqlName )
2022-10-25 11:41:19 +08:00
if err != nil {
return err
}
2022-11-04 19:02:15 +08:00
var files [ ] string
2022-10-25 11:41:19 +08:00
2022-11-04 19:02:15 +08:00
path := fmt . Sprintf ( "%s/%s/%s/conf/my.cnf" , constant . AppInstallDir , app . Key , app . Name )
lineBytes , err := ioutil . ReadFile ( path )
if err != nil {
2022-10-25 11:41:19 +08:00
return err
2022-11-04 19:02:15 +08:00
} else {
files = strings . Split ( string ( lineBytes ) , "\n" )
}
group := ""
for _ , info := range updatas {
switch info . Param {
case "key_buffer_size" , "sort_buffer_size" :
group = "[myisamchk]"
default :
group = "[mysqld]"
}
files = updateMyCnf ( files , group , info . Param , info . Value )
2022-10-25 11:41:19 +08:00
}
2022-11-04 19:02:15 +08:00
file , err := os . OpenFile ( path , os . O_WRONLY , 0666 )
if err != nil {
2022-10-25 11:41:19 +08:00
return err
}
2022-11-04 19:02:15 +08:00
defer file . Close ( )
_ , err = file . WriteString ( strings . Join ( files , "\n" ) )
if err != nil {
2022-10-25 11:41:19 +08:00
return err
}
2022-11-04 19:02:15 +08:00
if _ , err := compose . Restart ( fmt . Sprintf ( "%s/%s/%s/docker-compose.yml" , constant . AppInstallDir , app . Key , app . Name ) ) ; err != nil {
2022-10-25 11:41:19 +08:00
return err
}
return nil
}
2022-10-31 23:52:39 +08:00
func ( u * MysqlService ) LoadBaseInfo ( name string ) ( * dto . DBBaseInfo , error ) {
2022-10-25 18:34:33 +08:00
var data dto . DBBaseInfo
2022-10-31 23:52:39 +08:00
app , err := mysqlRepo . LoadBaseInfoByName ( name )
2022-10-21 18:50:38 +08:00
if err != nil {
return nil , err
}
2022-11-04 19:02:15 +08:00
data . ContainerName = app . ContainerName
2022-10-25 18:34:33 +08:00
data . Name = app . Name
data . Port = int64 ( app . Port )
data . Password = app . Password
2022-10-31 23:52:39 +08:00
data . MysqlKey = app . Key
2022-10-21 18:50:38 +08:00
2022-10-25 18:34:33 +08:00
hosts , err := excuteSqlForRows ( app . ContainerName , app . Password , "select host from mysql.user where user='root';" )
if err != nil {
return nil , err
}
for _ , host := range hosts {
if host == "%" {
data . RemoteConn = true
break
}
}
return & data , nil
}
2022-10-31 23:52:39 +08:00
func ( u * MysqlService ) LoadVariables ( name string ) ( * dto . MysqlVariables , error ) {
app , err := mysqlRepo . LoadBaseInfoByName ( name )
2022-10-25 18:34:33 +08:00
if err != nil {
return nil , err
}
variableMap , err := excuteSqlForMaps ( app . ContainerName , app . Password , "show global variables;" )
2022-10-24 18:46:19 +08:00
if err != nil {
return nil , err
}
var info dto . MysqlVariables
2022-10-21 18:50:38 +08:00
arr , err := json . Marshal ( variableMap )
if err != nil {
return nil , err
}
_ = json . Unmarshal ( arr , & info )
return & info , nil
}
2022-10-31 23:52:39 +08:00
func ( u * MysqlService ) LoadStatus ( name string ) ( * dto . MysqlStatus , error ) {
app , err := mysqlRepo . LoadBaseInfoByName ( name )
2022-10-21 18:50:38 +08:00
if err != nil {
return nil , err
}
2022-10-25 18:34:33 +08:00
statusMap , err := excuteSqlForMaps ( app . ContainerName , app . Password , "show global status;" )
2022-10-21 18:50:38 +08:00
if err != nil {
return nil , err
}
2022-10-25 18:34:33 +08:00
2022-10-21 18:50:38 +08:00
var info dto . MysqlStatus
2022-10-25 18:34:33 +08:00
arr , err := json . Marshal ( statusMap )
2022-10-21 18:50:38 +08:00
if err != nil {
return nil , err
}
_ = json . Unmarshal ( arr , & info )
2022-10-24 18:46:19 +08:00
2022-10-25 18:34:33 +08:00
if value , ok := statusMap [ "Run" ] ; ok {
2022-10-24 18:46:19 +08:00
uptime , _ := strconv . Atoi ( value )
info . Run = time . Unix ( time . Now ( ) . Unix ( ) - int64 ( uptime ) , 0 ) . Format ( "2006-01-02 15:04:05" )
} else {
2022-10-25 18:34:33 +08:00
if value , ok := statusMap [ "Uptime" ] ; ok {
2022-10-24 18:46:19 +08:00
uptime , _ := strconv . Atoi ( value )
info . Run = time . Unix ( time . Now ( ) . Unix ( ) - int64 ( uptime ) , 0 ) . Format ( "2006-01-02 15:04:05" )
}
}
2022-10-25 18:34:33 +08:00
info . File = "OFF"
info . Position = "OFF"
rows , err := excuteSqlForRows ( app . ContainerName , app . Password , "show master status;" )
if err != nil {
return nil , err
}
if len ( rows ) > 2 {
itemValue := strings . Split ( rows [ 1 ] , "\t" )
if len ( itemValue ) > 2 {
info . File = itemValue [ 0 ]
info . Position = itemValue [ 1 ]
}
}
return & info , nil
}
func excuteSqlForMaps ( containerName , password , command string ) ( map [ string ] string , error ) {
2022-10-27 23:09:39 +08:00
cmd := exec . Command ( "docker" , "exec" , containerName , "mysql" , "-uroot" , "-p" + password , "-e" , command )
2022-10-25 18:34:33 +08:00
stdout , err := cmd . CombinedOutput ( )
2022-10-27 23:09:39 +08:00
stdStr := strings . ReplaceAll ( string ( stdout ) , "mysql: [Warning] Using a password on the command line interface can be insecure.\n" , "" )
if err != nil || strings . HasPrefix ( string ( stdStr ) , "ERROR " ) {
return nil , errors . New ( stdStr )
2022-10-24 18:46:19 +08:00
}
2022-10-25 18:34:33 +08:00
rows := strings . Split ( stdStr , "\n" )
rowMap := make ( map [ string ] string )
for _ , v := range rows {
itemRow := strings . Split ( v , "\t" )
if len ( itemRow ) == 2 {
rowMap [ itemRow [ 0 ] ] = itemRow [ 1 ]
2022-10-24 18:46:19 +08:00
}
}
2022-10-25 18:34:33 +08:00
return rowMap , nil
}
func excuteSqlForRows ( containerName , password , command string ) ( [ ] string , error ) {
2022-10-27 23:09:39 +08:00
cmd := exec . Command ( "docker" , "exec" , containerName , "mysql" , "-uroot" , "-p" + password , "-e" , command )
2022-10-25 18:34:33 +08:00
stdout , err := cmd . CombinedOutput ( )
stdStr := strings . ReplaceAll ( string ( stdout ) , "mysql: [Warning] Using a password on the command line interface can be insecure.\n" , "" )
2022-10-27 23:09:39 +08:00
if err != nil || strings . HasPrefix ( string ( stdStr ) , "ERROR " ) {
return nil , errors . New ( stdStr )
}
2022-10-25 18:34:33 +08:00
return strings . Split ( stdStr , "\n" ) , nil
}
func excuteSql ( containerName , password , command string ) error {
2022-10-27 23:09:39 +08:00
cmd := exec . Command ( "docker" , "exec" , containerName , "mysql" , "-uroot" , "-p" + password , "-e" , command )
2022-10-25 18:34:33 +08:00
stdout , err := cmd . CombinedOutput ( )
stdStr := strings . ReplaceAll ( string ( stdout ) , "mysql: [Warning] Using a password on the command line interface can be insecure.\n" , "" )
2022-10-27 23:09:39 +08:00
if err != nil || strings . HasPrefix ( string ( stdStr ) , "ERROR " ) {
return errors . New ( stdStr )
2022-10-25 18:34:33 +08:00
}
return nil
2022-10-21 18:50:38 +08:00
}
2022-10-28 18:46:14 +08:00
2022-11-03 23:42:42 +08:00
func backupMysql ( backupType , baseDir , backupDir , mysqlName , dbName , fileName string ) error {
app , err := mysqlRepo . LoadBaseInfoByName ( mysqlName )
2022-10-28 18:46:14 +08:00
if err != nil {
return err
}
fullDir := baseDir + "/" + backupDir
if _ , err := os . Stat ( fullDir ) ; err != nil && os . IsNotExist ( err ) {
if err = os . MkdirAll ( fullDir , os . ModePerm ) ; err != nil {
if err != nil {
return fmt . Errorf ( "mkdir %s failed, err: %v" , fullDir , err )
}
}
}
outfile , _ := os . OpenFile ( fullDir + "/" + fileName , os . O_RDWR | os . O_CREATE , 0755 )
cmd := exec . Command ( "docker" , "exec" , app . ContainerName , "mysqldump" , "-uroot" , "-p" + app . Password , dbName )
gzipCmd := exec . Command ( "gzip" , "-cf" )
gzipCmd . Stdin , _ = cmd . StdoutPipe ( )
gzipCmd . Stdout = outfile
_ = gzipCmd . Start ( )
_ = cmd . Run ( )
_ = gzipCmd . Wait ( )
record := & model . BackupRecord {
Type : "database-mysql" ,
Name : app . Name ,
DetailName : dbName ,
Source : backupType ,
2022-10-31 17:26:15 +08:00
BackupType : backupType ,
2022-10-28 18:46:14 +08:00
FileDir : backupDir ,
FileName : fileName ,
}
if baseDir != constant . TmpDir || backupType == "LOCAL" {
record . Source = "LOCAL"
record . FileDir = fullDir
}
if err := backupRepo . CreateRecord ( record ) ; err != nil {
global . LOG . Errorf ( "save backup record failed, err: %v" , err )
}
return nil
}
2022-11-04 19:02:15 +08:00
func updateMyCnf ( oldFiles [ ] string , group string , param string , value interface { } ) [ ] string {
isOn := false
2022-11-09 15:08:38 +08:00
hasGroup := false
2022-11-04 19:02:15 +08:00
hasKey := false
regItem , _ := regexp . Compile ( ` \[*\] ` )
var newFiles [ ] string
2022-11-09 15:08:38 +08:00
i := 0
2022-11-04 19:02:15 +08:00
for _ , line := range oldFiles {
2022-11-09 15:08:38 +08:00
i ++
2022-11-04 19:02:15 +08:00
if strings . HasPrefix ( line , group ) {
isOn = true
2022-11-09 15:08:38 +08:00
hasGroup = true
2022-11-04 19:02:15 +08:00
newFiles = append ( newFiles , line )
continue
}
if ! isOn {
newFiles = append ( newFiles , line )
continue
}
if strings . HasPrefix ( line , param ) || strings . HasPrefix ( line , "# " + param ) {
newFiles = append ( newFiles , fmt . Sprintf ( "%s=%v" , param , value ) )
hasKey = true
continue
}
2022-11-09 15:08:38 +08:00
if regItem . Match ( [ ] byte ( line ) ) || i == len ( oldFiles ) {
isOn = false
if ! hasKey {
newFiles = append ( newFiles , fmt . Sprintf ( "%s=%v" , param , value ) )
}
2022-11-04 19:02:15 +08:00
newFiles = append ( newFiles , line )
continue
}
2022-11-09 15:08:38 +08:00
newFiles = append ( newFiles , line )
2022-11-04 19:02:15 +08:00
}
2022-11-09 15:08:38 +08:00
if ! hasGroup {
2022-11-04 19:02:15 +08:00
newFiles = append ( newFiles , group + "\n" )
newFiles = append ( newFiles , fmt . Sprintf ( "%s=%v\n" , param , value ) )
}
return newFiles
}
2022-11-09 15:08:38 +08:00
func loadBackupDirAndKey ( mysqlName string ) ( string , string , error ) {
app , err := mysqlRepo . LoadBaseInfoByName ( mysqlName )
if err != nil {
return "" , "" , err
}
localDir , err := loadLocalDir ( )
if err != nil {
return "" , "" , err
}
return localDir , app . Key , nil
}