Files
EdgeAdmin/internal/web/actions/default/setup/mysql/mysqlinstallers/mysql_installer.go

759 lines
20 KiB
Go
Raw Normal View History

// Copyright 2023 Liuxiangchao iwind.liu@gmail.com. All rights reserved. Official site: https://goedge.cn .
package mysqlinstallers
import (
"bytes"
"crypto/rand"
"errors"
"fmt"
2023-12-26 09:09:21 +08:00
executils "github.com/TeaOSLab/EdgeAdmin/internal/utils/exec"
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/default/setup/mysql/mysqlinstallers/utils"
stringutil "github.com/iwind/TeaGo/utils/string"
timeutil "github.com/iwind/TeaGo/utils/time"
"io"
"net"
"net/http"
"os"
2024-03-08 20:02:47 +08:00
"os/exec"
"path/filepath"
"regexp"
"runtime"
"strconv"
"strings"
"time"
)
type MySQLInstaller struct {
password string
}
func NewMySQLInstaller() *MySQLInstaller {
return &MySQLInstaller{}
}
func (this *MySQLInstaller) InstallFromFile(xzFilePath string, targetDir string) error {
// check whether mysql already running
this.log("checking mysqld ...")
var oldPid = utils.FindPidWithName("mysqld")
if oldPid > 0 {
return errors.New("there is already a running mysql server process, pid: '" + strconv.Itoa(oldPid) + "'")
}
// check target dir
this.log("checking target dir '" + targetDir + "' ...")
_, err := os.Stat(targetDir)
if err == nil {
// check target dir
matches, _ := filepath.Glob(targetDir + "/*")
if len(matches) > 0 {
return errors.New("target dir '" + targetDir + "' already exists and not empty")
} else {
err = os.Remove(targetDir)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("clean target dir '%s' failed: %w", targetDir, err)
}
}
}
// check 'tar' command
this.log("checking 'tar' command ...")
2023-12-26 09:09:21 +08:00
var tarExe, _ = executils.LookPath("tar")
if len(tarExe) == 0 {
this.log("installing 'tar' command ...")
err = this.installTarCommand()
if err != nil {
this.log("WARN: failed to install 'tar' ...")
}
}
// check commands
this.log("checking system commands ...")
var cmdList = []string{"tar" /** again **/, "chown", "sh"}
for _, cmd := range cmdList {
2023-12-26 09:09:21 +08:00
cmdPath, err := executils.LookPath(cmd)
if err != nil || len(cmdPath) == 0 {
return errors.New("could not find '" + cmd + "' command in this system")
}
}
groupAddExe, err := this.lookupGroupAdd()
if err != nil {
return errors.New("could not find 'groupadd' command in this system")
}
userAddExe, err := this.lookupUserAdd()
if err != nil {
return errors.New("could not find 'useradd' command in this system")
}
// ubuntu apt
aptGetExe, err := exec.LookPath("apt-get")
if err == nil && len(aptGetExe) > 0 {
2024-05-12 08:56:28 +08:00
for _, lib := range []string{"libaio1", "libncurses5", "libnuma1"} {
this.log("checking " + lib + " ...")
var cmd = utils.NewCmd(aptGetExe, "-y", "install", lib)
cmd.WithStderr()
err = cmd.Run()
if err != nil {
// try apt
aptExe, aptErr := exec.LookPath("apt")
if aptErr == nil && len(aptExe) > 0 {
cmd = utils.NewCmd(aptExe, "-y", "install", lib)
cmd.WithStderr()
err = cmd.Run()
}
if err != nil {
2024-05-12 08:56:28 +08:00
if lib == "libnuma1" {
err = nil
} else {
return errors.New("install " + lib + " failed: " + cmd.Stderr())
}
}
}
time.Sleep(1 * time.Second)
}
} else { // yum
2023-12-26 09:09:21 +08:00
yumExe, err := executils.LookPath("yum")
if err == nil && len(yumExe) > 0 {
for _, lib := range []string{"libaio", "ncurses-libs", "ncurses-compat-libs", "numactl-libs"} {
var cmd = utils.NewCmd("yum", "-y", "install", lib)
_ = cmd.Run()
time.Sleep(1 * time.Second)
}
}
// create symbolic links
{
var libFile = "/usr/lib64/libncurses.so.5"
_, err = os.Stat(libFile)
if err != nil && os.IsNotExist(err) {
var latestLibFile = this.findLatestVersionFile("/usr/lib64", "libncurses.so.")
if len(latestLibFile) > 0 {
this.log("link '" + latestLibFile + "' to '" + libFile + "'")
_ = os.Symlink(latestLibFile, libFile)
}
}
}
{
var libFile = "/usr/lib64/libtinfo.so.5"
_, err = os.Stat(libFile)
if err != nil && os.IsNotExist(err) {
var latestLibFile = this.findLatestVersionFile("/usr/lib64", "libtinfo.so.")
if len(latestLibFile) > 0 {
this.log("link '" + latestLibFile + "' to '" + libFile + "'")
_ = os.Symlink(latestLibFile, libFile)
}
}
}
}
// create 'mysql' user group
this.log("checking 'mysql' user group ...")
{
data, err := os.ReadFile("/etc/group")
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("check user group failed: %w", err)
}
if !bytes.Contains(data, []byte("\nmysql:")) {
var cmd = utils.NewCmd(groupAddExe, "mysql")
cmd.WithStderr()
err = cmd.Run()
if err != nil {
return errors.New("add 'mysql' user group failed: " + cmd.Stderr())
}
}
}
// create 'mysql' user
this.log("checking 'mysql' user ...")
{
data, err := os.ReadFile("/etc/passwd")
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("check user failed: %w", err)
}
if !bytes.Contains(data, []byte("\nmysql:")) {
var cmd *utils.Cmd
if strings.HasSuffix(userAddExe, "useradd") {
cmd = utils.NewCmd(userAddExe, "mysql", "-g", "mysql")
} else { // adduser
cmd = utils.NewCmd(userAddExe, "-S", "-G", "mysql", "mysql")
}
cmd.WithStderr()
err = cmd.Run()
if err != nil {
return errors.New("add 'mysql' user failed: " + cmd.Stderr())
}
}
}
// mkdir
{
var parentDir = filepath.Dir(targetDir)
stat, err := os.Stat(parentDir)
if err != nil {
if os.IsNotExist(err) {
err = os.MkdirAll(parentDir, 0777)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("try to create dir '%s' failed: %w", parentDir, err)
}
} else {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("check dir '%s' failed: %w", parentDir, err)
}
} else {
if !stat.IsDir() {
return errors.New("'" + parentDir + "' should be a directory")
}
}
}
// check installer file .xz
this.log("checking installer file ...")
{
stat, err := os.Stat(xzFilePath)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("could not open the installer file: %w", err)
}
if stat.IsDir() {
return errors.New("'" + xzFilePath + "' not a valid file")
}
var basename = filepath.Base(xzFilePath)
if !strings.HasSuffix(basename, ".xz") {
return errors.New("installer file should has '.xz' extension")
}
}
// extract
this.log("extracting installer file ...")
var tmpDir = os.TempDir() + "/goedge-mysql-tmp"
{
_, err := os.Stat(tmpDir)
if err == nil {
err = os.RemoveAll(tmpDir)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("clean temporary directory '%s' failed: %w", tmpDir, err)
}
}
err = os.Mkdir(tmpDir, 0777)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("create temporary directory '%s' failed: %w", tmpDir, err)
}
}
{
var cmd = utils.NewCmd("tar", "-xJvf", xzFilePath, "-C", tmpDir)
cmd.WithStderr()
err = cmd.Run()
if err != nil {
return errors.New("extract installer file '" + xzFilePath + "' failed: " + cmd.Stderr())
}
}
// create datadir
matches, err := filepath.Glob(tmpDir + "/mysql-*")
if err != nil || len(matches) == 0 {
return errors.New("could not find mysql installer directory from '" + tmpDir + "'")
}
var baseDir = matches[0]
var dataDir = baseDir + "/data"
_, err = os.Stat(dataDir)
if err != nil {
if os.IsNotExist(err) {
err = os.Mkdir(dataDir, 0777)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("create data dir '%s' failed: %w", dataDir, err)
}
} else {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("check data dir '%s' failed: %w", dataDir, err)
}
}
// chown datadir
{
var cmd = utils.NewCmd("chown", "mysql:mysql", dataDir)
cmd.WithStderr()
err = cmd.Run()
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("chown data dir '%s' failed: %w", dataDir, err)
}
}
// create my.cnf
var myCnfFile = "/etc/my.cnf"
_, err = os.Stat(myCnfFile)
if err == nil {
// backup it
err = os.Rename(myCnfFile, "/etc/my.cnf."+timeutil.Format("YmdHis"))
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("backup '/etc/my.cnf' failed: %w", err)
}
}
// mysql server options https://dev.mysql.com/doc/refman/8.0/en/server-system-variables.html
var myCnfTemplate = this.createMyCnf(baseDir, dataDir)
err = os.WriteFile(myCnfFile, []byte(myCnfTemplate), 0666)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("write '%s' failed: %w", myCnfFile, err)
}
// initialize
this.log("initializing mysql ...")
2023-08-08 14:17:16 +08:00
var generatedPassword string
{
var cmd = utils.NewCmd(baseDir+"/bin/mysqld", "--initialize", "--user=mysql")
cmd.WithStderr()
cmd.WithStdout()
err = cmd.Run()
if err != nil {
return errors.New("initialize failed: " + cmd.Stderr())
}
// read from stdout
var match = regexp.MustCompile(`temporary password is.+:\s*(.+)`).FindStringSubmatch(cmd.Stdout())
if len(match) == 0 {
// read from stderr
match = regexp.MustCompile(`temporary password is.+:\s*(.+)`).FindStringSubmatch(cmd.Stderr())
if len(match) == 0 {
return errors.New("initialize successfully, but could not find generated password, please report to developer")
}
}
generatedPassword = strings.TrimSpace(match[1])
// write password to file
var passwordFile = baseDir + "/generated-password.txt"
err = os.WriteFile(passwordFile, []byte(generatedPassword), 0666)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("write password failed: %w", err)
}
}
// move to right place
this.log("moving files to target dir ...")
err = os.Rename(baseDir, targetDir)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("move '%s' to '%s' failed: %w", baseDir, targetDir, err)
}
baseDir = targetDir
// change my.cnf
myCnfTemplate = this.createMyCnf(baseDir, baseDir+"/data")
err = os.WriteFile(myCnfFile, []byte(myCnfTemplate), 0666)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("create new '%s' failed: %w", myCnfFile, err)
}
// start mysql
this.log("starting mysql ...")
{
var cmd = utils.NewCmd(baseDir+"/bin/mysqld_safe", "--user=mysql")
cmd.WithStderr()
err = cmd.Start()
if err != nil {
return errors.New("start failed '" + cmd.String() + "': " + cmd.Stderr())
}
// waiting for startup
2023-09-01 11:57:28 +08:00
for i := 0; i < 30; i++ {
var conn net.Conn
conn, err = net.Dial("tcp", "127.0.0.1:3306")
if err != nil {
time.Sleep(1 * time.Second)
} else {
_ = conn.Close()
break
}
}
time.Sleep(1 * time.Second)
}
// change password
newPassword, err := this.generatePassword()
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("generate new password failed: %w", err)
}
this.log("changing mysql password ...")
var passwordSQL = "ALTER USER 'root'@'localhost' IDENTIFIED BY '" + newPassword + "';"
{
2023-09-01 11:57:28 +08:00
var cmd = utils.NewCmd("sh", "-c", baseDir+"/bin/mysql --host=\"127.0.0.1\" --user=root --password=\""+generatedPassword+"\" --execute=\""+passwordSQL+"\" --connect-expired-password")
cmd.WithStderr()
err = cmd.Run()
if err != nil {
return errors.New("change password failed: " + cmd.String() + ": " + cmd.Stderr())
}
}
this.password = newPassword
var passwordFile = baseDir + "/generated-password.txt"
err = os.WriteFile(passwordFile, []byte(this.password), 0666)
if err != nil {
2023-08-11 16:41:43 +08:00
return fmt.Errorf("write generated file failed: %w", err)
}
// remove temporary directory
_ = os.Remove(tmpDir)
// create link to 'mysql' client command
var clientExe = "/usr/local/bin/mysql"
_, err = os.Stat(clientExe)
if err != nil && os.IsNotExist(err) {
err = os.Symlink(baseDir+"/bin/mysql", clientExe)
if err == nil {
this.log("created symbolic link '" + clientExe + "' to '" + baseDir + "/bin/mysql'")
} else {
this.log("WARN: failed to create symbolic link '" + clientExe + "' to '" + baseDir + "/bin/mysql': " + err.Error())
}
}
// install service
// this is not required, so we ignore all errors
err = this.installService(baseDir)
if err != nil {
this.log("WARN: install service failed: " + err.Error())
}
this.log("finished")
return nil
}
func (this *MySQLInstaller) Download() (path string, err error) {
var client = &http.Client{}
// check latest version
this.log("checking mysql latest version ...")
2023-10-26 08:33:35 +08:00
var latestVersion = "8.2.0" // default version
var majorVersion = "8.2"
{
req, err := http.NewRequest(http.MethodGet, "https://dev.mysql.com/downloads/mysql/", nil)
if err != nil {
return "", err
}
req.Header.Set("User-Agent", "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Ubuntu Chromium/78.0.3904.108 Chrome/78.0.3904.108 Safari/537.36")
resp, err := client.Do(req)
if err != nil {
return "", errors.New("check latest version failed: " + err.Error())
}
defer func() {
_ = resp.Body.Close()
}()
if resp.StatusCode == http.StatusOK {
data, err := io.ReadAll(resp.Body)
if err != nil {
return "", errors.New("read latest version failed: " + err.Error())
}
var reg = regexp.MustCompile(`<h1>MySQL Community Server ([\d.]+) `)
var matches = reg.FindSubmatch(data)
if len(matches) > 0 {
latestVersion = string(matches[1])
2023-10-26 08:33:35 +08:00
var matchPieces = strings.Split(latestVersion, ".")
if len(matchPieces) >= 2 {
majorVersion = strings.Join(matchPieces[:2], ".")
}
}
}
}
this.log("found version: v" + latestVersion)
// download
this.log("start downloading ...")
2023-10-26 08:33:35 +08:00
var downloadURL = "https://cdn.mysql.com/Downloads/MySQL-" + majorVersion + "/mysql-" + latestVersion + "-linux-glibc2.17-x86_64-minimal.tar.xz"
{
2023-07-19 15:14:33 +08:00
this.log("downloading from url '" + downloadURL + "' ...")
req, err := http.NewRequest(http.MethodGet, downloadURL, nil)
if err != nil {
return "", err
}
req.Header.Set("User-Agent", "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36")
resp, err := client.Do(req)
if err != nil {
return "", errors.New("check latest version failed: " + err.Error())
}
defer func() {
_ = resp.Body.Close()
}()
if resp.StatusCode != http.StatusOK {
return "", errors.New("check latest version failed: invalid response code: " + strconv.Itoa(resp.StatusCode))
}
path = filepath.Base(downloadURL)
fp, err := os.OpenFile(path, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666)
if err != nil {
return "", errors.New("create download file '" + path + "' failed: " + err.Error())
}
var writer = utils.NewProgressWriter(fp, resp.ContentLength)
var ticker = time.NewTicker(1 * time.Second)
var done = make(chan bool, 1)
go func() {
var lastProgress float32 = -1
for {
select {
case <-ticker.C:
var progress = writer.Progress()
if lastProgress < 0 || progress-lastProgress > 0.1 || progress == 1 {
lastProgress = progress
this.log(fmt.Sprintf("%.2f%%", progress*100))
}
case <-done:
return
}
}
}()
_, err = io.Copy(writer, resp.Body)
if err != nil {
_ = fp.Close()
done <- true
return "", errors.New("download failed: " + err.Error())
}
err = fp.Close()
if err != nil {
done <- true
return "", errors.New("download failed: " + err.Error())
}
time.Sleep(1 * time.Second) // waiting for progress printing
done <- true
}
return path, nil
}
// Password get generated password
func (this *MySQLInstaller) Password() string {
return this.password
}
// create my.cnf content
func (this *MySQLInstaller) createMyCnf(baseDir string, dataDir string) string {
var memoryTotalG = 1
if runtime.GOOS == "linux" {
memoryTotalG = this.sysMemoryGB() / 2
if memoryTotalG <= 0 {
memoryTotalG = 1
}
}
return `
[mysqld]
port=3306
basedir="` + baseDir + `"
datadir="` + dataDir + `"
max_connections=256
innodb_flush_log_at_trx_commit=2
max_prepared_stmt_count=65535
binlog_cache_size=1M
binlog_stmt_cache_size=1M
thread_cache_size=32
binlog_expire_logs_seconds=604800
innodb_sort_buffer_size=8M
innodb_buffer_pool_size=` + strconv.Itoa(memoryTotalG) + "G"
}
// generate random password
func (this *MySQLInstaller) generatePassword() (string, error) {
var p = make([]byte, 16)
n, err := rand.Read(p)
if err != nil {
return "", err
}
return fmt.Sprintf("%x", p[:n]), nil
}
// print log
func (this *MySQLInstaller) log(message string) {
utils.SharedLogger.Push("[" + timeutil.Format("H:i:s") + "]" + message)
}
// copy file
func (this *MySQLInstaller) installService(baseDir string) error {
2023-12-26 09:09:21 +08:00
_, err := executils.LookPath("systemctl")
if err != nil {
return err
}
this.log("registering systemd service ...")
2024-03-08 20:02:47 +08:00
var startCmd = "${BASE_DIR}/support-files/mysql.server start"
bashPath, _ := exec.LookPath("bash")
if len(bashPath) > 0 {
startCmd = bashPath + " -c \"" + startCmd + "\""
}
var desc = `### BEGIN INIT INFO
# Provides: mysql
# Required-Start: $local_fs $network $remote_fs
# Should-Start: ypbind nscd ldap ntpd xntpd
# Required-Stop: $local_fs $network $remote_fs
# Default-Start: 2 3 4 5
# Default-Stop: 0 1 6
# Short-Description: start and stop MySQL
# Description: MySQL is a very fast and reliable SQL database engine.
### END INIT INFO
[Unit]
Description=MySQL Service
Before=shutdown.target
After=network-online.target
[Service]
Type=simple
2023-04-03 10:01:35 +08:00
Restart=on-failure
RestartSec=5s
2024-03-08 20:02:47 +08:00
RemainAfterExit=yes
ExecStart=` + startCmd + `
ExecStop=${BASE_DIR}/support-files/mysql.server stop
ExecRestart=${BASE_DIR}/support-files/mysql.server restart
ExecStatus=${BASE_DIR}/support-files/mysql.server status
ExecReload=${BASE_DIR}/support-files/mysql.server reload
[Install]
WantedBy=multi-user.target`
desc = strings.ReplaceAll(desc, "${BASE_DIR}", baseDir)
err = os.WriteFile("/etc/systemd/system/mysqld.service", []byte(desc), 0666)
if err != nil {
return err
}
var cmd = utils.NewTimeoutCmd(5*time.Second, "systemctl", "enable", "mysqld.service")
cmd.WithStderr()
err = cmd.Run()
if err != nil {
return errors.New("enable mysqld.service failed: " + cmd.Stderr())
}
return nil
}
// install 'tar' command automatically
func (this *MySQLInstaller) installTarCommand() error {
// dnf
dnfExe, err := exec.LookPath("dnf")
if err == nil && len(dnfExe) > 0 {
var cmd = utils.NewTimeoutCmd(10*time.Second, dnfExe, "-y", "install", "tar")
return cmd.Run()
}
// yum
2023-12-26 09:09:21 +08:00
yumExe, err := executils.LookPath("yum")
if err == nil && len(yumExe) > 0 {
var cmd = utils.NewTimeoutCmd(10*time.Second, yumExe, "-y", "install", "tar")
return cmd.Run()
}
// apt-get
aptGetExe, err := exec.LookPath("apt-get")
if err == nil && len(aptGetExe) > 0 {
var cmd = utils.NewTimeoutCmd(10*time.Second, aptGetExe, "-y", "install", "tar")
err = cmd.Run()
if err != nil {
aptExe, aptErr := exec.LookPath("apt")
if aptErr == nil {
cmd = utils.NewTimeoutCmd(10*time.Second, aptExe, "-y", "install", "tar")
err = cmd.Run()
}
}
return err
}
return nil
}
func (this *MySQLInstaller) lookupGroupAdd() (string, error) {
for _, cmd := range []string{"groupadd", "addgroup"} {
2023-12-26 09:09:21 +08:00
path, err := executils.LookPath(cmd)
if err == nil && len(path) > 0 {
return path, nil
}
}
return "", errors.New("not found")
}
func (this *MySQLInstaller) lookupUserAdd() (string, error) {
for _, cmd := range []string{"useradd", "adduser"} {
2023-12-26 09:09:21 +08:00
path, err := executils.LookPath(cmd)
if err == nil && len(path) > 0 {
return path, nil
}
}
return "", errors.New("not found")
}
func (this *MySQLInstaller) findLatestVersionFile(dir string, prefix string) string {
files, err := filepath.Glob(filepath.Clean(dir + "/" + prefix + "*"))
if err != nil {
return ""
}
var resultFile = ""
var lastVersion = ""
var reg = regexp.MustCompile(`\.([\d.]+)`)
for _, file := range files {
var filename = filepath.Base(file)
var matches = reg.FindStringSubmatch(filename)
if len(matches) > 1 {
var version = matches[1]
if len(lastVersion) == 0 || stringutil.VersionCompare(lastVersion, version) < 0 {
lastVersion = version
resultFile = file
}
}
}
return resultFile
}
func (this *MySQLInstaller) sysMemoryGB() int {
if runtime.GOOS != "linux" {
return 0
}
meminfoData, err := os.ReadFile("/proc/meminfo")
if err != nil {
return 0
}
for _, line := range bytes.Split(meminfoData, []byte{'\n'}) {
line = bytes.TrimSpace(line)
if bytes.Contains(line, []byte{':'}) {
name, value, found := bytes.Cut(line, []byte{':'})
if found {
name = bytes.TrimSpace(name)
if bytes.Equal(name, []byte("MemTotal")) {
for _, unit := range []string{"gB", "mB", "kB"} {
if bytes.Contains(value, []byte(unit)) {
value = bytes.TrimSpace(bytes.ReplaceAll(value, []byte(unit), nil))
valueInt, err := strconv.Atoi(string(value))
if err != nil {
return 0
}
switch unit {
case "gB":
return valueInt
case "mB":
return valueInt / 1024
case "kB":
return valueInt / 1024 / 1024
}
return 0
}
}
}
}
}
}
return 0
}