f8f46b3c48
Added the Account Dashboard and merged a few account views into it. BREAKING CHANGE: We now use config/config.json instead of config/config.go, be sure to setup one of these files, you can config_default.json as an example of what a config.json should look like. If you don't have an existing installation, you can just rely on the installer to do this for you. CSS Changes (does not include Nox Theme): Sidebar should no longer show up in the account manager in some odd situations or themes. Made a few CSS rules more generic. Forms have a new look in Cosora now. Config Changes: Removed the DefaultRoute config field. Added the DefaultPath config field. Added the MaxRequestSizeStr config field to make it easier for users to input custom max request sizes without having to use a calculator or figure out how many bytes there are in a megabyte. Removed the CacheTopicUser config field. Added the UserCache config field. Added the TopicCache config field Phrases: Removed ten english phrases. Added 21 english phrases. Changed eleven english phrases. Removed some duplicate indices in the english phrase pack. Removed some old benchmark code. Tweaked some things to make the linter happy. Added comments for all the MemoryUserCache and MemoryTopicCache methods. Added a comment for the null caches, consult the other caches for further information on the methods. Added a client-side check to make sure the user doesn't upload too much data in a single post. The server already did this, but it might be a while before feedback arrives from it. Simplified a lot of the control panel route code with the buildBasePage function. Renamed /user/edit/critical/ to /user/edit/password/ Renamed /user/edit/critical/submit/ to /user/edit/password/submit/ Made some small improvements to SEO with a couple of meta tags. Renamed some of the control panel templates so that they use _ instead of -. Fixed a bug where notices were being moved to the wrong place in some areas in Cosora. Added the writeJsonError function to help abstract writing json errors. Moved routePanelUsers to panel.Users Moved routePanelUsersEdit to panel.UsersEdit Moved routePanelUsersEditSubmit to panel.UsersEditSubmit Renamed routes.AccountEditCritical to routes.AccountEditPassword Renamed routes.AccountEditCriticalSubmit to routes.AccountEditPasswordSubmit Removed the routes.AccountEditAvatar and routes.AccountEditUsername routes. Fixed a data race in MemoryTopicCache.Add which could lead to the capacity limit being bypassed. Tweaked MemoryTopicCache.AddUnsafe under the assumption that it's not going to be safe anyway, but we might as-well try in case this call is properly synchronised. Fixed a data race in MemoryTopicCache.Remove which could lead to the length counter being decremented twice. Tweaked the behaviour of MemoryTopicCache.RemoveUnsafe to mirror that of Remove. Fixed a data race in MemoryUserCache.Add which could lead to the capacity limit being bypassed. User can no longer change their usernames to blank. Made a lot of progress on the Nox theme. Added modified FA5 SVGs as a dependency for Nox. Be sure to run the patcher or update script and don't forget to create a customised config/config.json file.
322 lines
7.4 KiB
Go
322 lines
7.4 KiB
Go
/*
|
|
*
|
|
* Gosora Installer
|
|
* Copyright Azareal 2017 - 2018
|
|
*
|
|
*/
|
|
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"runtime/debug"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"./install"
|
|
)
|
|
|
|
var scanner *bufio.Scanner
|
|
|
|
var siteShortName string
|
|
var siteName string
|
|
var siteURL string
|
|
var serverPort string
|
|
|
|
var defaultAdapter = "mysql"
|
|
var defaultHost = "localhost"
|
|
var defaultUsername = "root"
|
|
var defaultDbname = "gosora"
|
|
var defaultSiteShortName = "SN"
|
|
var defaultSiteName = "Site Name"
|
|
var defaultsiteURL = "localhost"
|
|
var defaultServerPort = "80" // 8080's a good one, if you're testing and don't want it to clash with port 80
|
|
|
|
func main() {
|
|
// Capture panics instead of closing the window at a superhuman speed before the user can read the message on Windows
|
|
defer func() {
|
|
r := recover()
|
|
if r != nil {
|
|
fmt.Println(r)
|
|
debug.PrintStack()
|
|
pressAnyKey()
|
|
return
|
|
}
|
|
}()
|
|
|
|
scanner = bufio.NewScanner(os.Stdin)
|
|
fmt.Println("Welcome to Gosora's Installer")
|
|
fmt.Println("We're going to take you through a few steps to help you get started :)")
|
|
adap, ok := handleDatabaseDetails()
|
|
if !ok {
|
|
err := scanner.Err()
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
} else {
|
|
err = errors.New("Something went wrong!")
|
|
}
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
if !getSiteDetails() {
|
|
err := scanner.Err()
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
} else {
|
|
err = errors.New("Something went wrong!")
|
|
}
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
err := adap.InitDatabase()
|
|
if err != nil {
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
err = adap.TableDefs()
|
|
if err != nil {
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
err = adap.CreateAdmin()
|
|
if err != nil {
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
err = adap.InitialData()
|
|
if err != nil {
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
configContents := []byte(`{
|
|
"Site": {
|
|
"ShortName":"` + siteShortName + `",
|
|
"Name":"` + siteName + `",
|
|
"URL":"` + siteURL + `",
|
|
"Port":"` + serverPort + `",
|
|
"EnableSsl":false,
|
|
"EnableEmails":false,
|
|
"HasProxy":false,
|
|
"Language": "english"
|
|
},
|
|
"Config": {
|
|
"SslPrivkey": "",
|
|
"SslFullchain": "",
|
|
"SMTPServer": "",
|
|
"SMTPUsername": "",
|
|
"SMTPPassword": "",
|
|
"SMTPPort": "25",
|
|
|
|
"MaxRequestSizeStr":"5MB",
|
|
"UserCache":"static",
|
|
"TopicCache":"static",
|
|
"UserCacheCapacity":120,
|
|
"TopicCacheCapacity":200,
|
|
"DefaultPath":"/topics/",
|
|
"DefaultGroup":3,
|
|
"ActivationGroup":5,
|
|
"StaffCSS":"staff_post",
|
|
"DefaultForum":2,
|
|
"MinifyTemplates":true,
|
|
"BuildSlugs":true,
|
|
"ServerCount":1,
|
|
"Noavatar":"https://api.adorable.io/avatars/285/{id}@{site_url}.png",
|
|
"ItemsPerPage":25
|
|
},
|
|
"Database": {
|
|
"Adapter": "` + adap.Name() + `",
|
|
"Host": "` + adap.DBHost() + `",
|
|
"Username": "` + adap.DBUsername() + `",
|
|
"Password": "` + adap.DBPassword() + `",
|
|
"Dbname": "` + adap.DBName() + `",
|
|
"Port": "` + adap.DBPort() + `",
|
|
|
|
"TestAdapter": "` + adap.Name() + `",
|
|
"TestHost": "",
|
|
"TestUsername": "",
|
|
"TestPassword": "",
|
|
"TestDbname": "",
|
|
"TestPort": ""
|
|
},
|
|
"Dev": {
|
|
"DebugMode":true,
|
|
"SuperDebug":false
|
|
}
|
|
}`)
|
|
|
|
//"Noavatar": "https://api.adorable.io/avatars/{width}/{id}@{site_url}.png" Maybe allow this sort of syntax?
|
|
fmt.Println("Opening the configuration file")
|
|
configFile, err := os.Create("./config/config.json")
|
|
if err != nil {
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
fmt.Println("Writing to the configuration file...")
|
|
_, err = configFile.Write(configContents)
|
|
if err != nil {
|
|
abortError(err)
|
|
return
|
|
}
|
|
|
|
configFile.Sync()
|
|
configFile.Close()
|
|
fmt.Println("Finished writing to the configuration file")
|
|
|
|
fmt.Println("Yay, you have successfully installed Gosora!")
|
|
fmt.Println("Your name is Admin and you can login with the password 'password'. Don't forget to change it! Seriously. It's really insecure.")
|
|
pressAnyKey()
|
|
}
|
|
|
|
func abortError(err error) {
|
|
fmt.Println(err)
|
|
fmt.Println("Aborting installation...")
|
|
pressAnyKey()
|
|
}
|
|
|
|
func handleDatabaseDetails() (adap install.InstallAdapter, ok bool) {
|
|
var dbAdapter string
|
|
var dbHost string
|
|
var dbUsername string
|
|
var dbPassword string
|
|
var dbName string
|
|
// TODO: Let the admin set the database port?
|
|
//var dbPort string
|
|
|
|
for {
|
|
fmt.Println("Which database adapter do you wish to use? mysql, mssql, or mysql? Default: mysql")
|
|
if !scanner.Scan() {
|
|
return nil, false
|
|
}
|
|
dbAdapter := strings.TrimSpace(scanner.Text())
|
|
if dbAdapter == "" {
|
|
dbAdapter = defaultAdapter
|
|
}
|
|
adap, ok = install.Lookup(dbAdapter)
|
|
if ok {
|
|
break
|
|
}
|
|
fmt.Println("That adapter doesn't exist")
|
|
}
|
|
fmt.Println("Set database adapter to " + dbAdapter)
|
|
|
|
fmt.Println("Database Host? Default: " + defaultHost)
|
|
if !scanner.Scan() {
|
|
return nil, false
|
|
}
|
|
dbHost = scanner.Text()
|
|
if dbHost == "" {
|
|
dbHost = defaultHost
|
|
}
|
|
fmt.Println("Set database host to " + dbHost)
|
|
|
|
fmt.Println("Database Username? Default: " + defaultUsername)
|
|
if !scanner.Scan() {
|
|
return nil, false
|
|
}
|
|
dbUsername = scanner.Text()
|
|
if dbUsername == "" {
|
|
dbUsername = defaultUsername
|
|
}
|
|
fmt.Println("Set database username to " + dbUsername)
|
|
|
|
fmt.Println("Database Password? Default: ''")
|
|
if !scanner.Scan() {
|
|
return nil, false
|
|
}
|
|
dbPassword = scanner.Text()
|
|
if len(dbPassword) == 0 {
|
|
fmt.Println("You didn't set a password for this user. This won't block the installation process, but it might create security issues in the future.")
|
|
fmt.Println("")
|
|
} else {
|
|
fmt.Println("Set password to " + obfuscatePassword(dbPassword))
|
|
}
|
|
|
|
fmt.Println("Database Name? Pick a name you like or one provided to you. Default: " + defaultDbname)
|
|
if !scanner.Scan() {
|
|
return nil, false
|
|
}
|
|
dbName = scanner.Text()
|
|
if dbName == "" {
|
|
dbName = defaultDbname
|
|
}
|
|
fmt.Println("Set database name to " + dbName)
|
|
|
|
adap.SetConfig(dbHost, dbUsername, dbPassword, dbName, adap.DefaultPort())
|
|
return adap, true
|
|
}
|
|
|
|
func getSiteDetails() bool {
|
|
fmt.Println("Okay. We also need to know some actual information about your site!")
|
|
fmt.Println("What's your site's name? Default: " + defaultSiteName)
|
|
if !scanner.Scan() {
|
|
return false
|
|
}
|
|
siteName = scanner.Text()
|
|
if siteName == "" {
|
|
siteName = defaultSiteName
|
|
}
|
|
fmt.Println("Set the site name to " + siteName)
|
|
|
|
// ? - We could compute this based on the first letter of each word in the site's name, if it's name spans multiple words. I'm not sure how to do this for single word names.
|
|
fmt.Println("Can we have a short abbreviation for your site? Default: " + defaultSiteShortName)
|
|
if !scanner.Scan() {
|
|
return false
|
|
}
|
|
siteShortName = scanner.Text()
|
|
if siteShortName == "" {
|
|
siteShortName = defaultSiteShortName
|
|
}
|
|
fmt.Println("Set the short name to " + siteShortName)
|
|
|
|
fmt.Println("What's your site's url? Default: " + defaultsiteURL)
|
|
if !scanner.Scan() {
|
|
return false
|
|
}
|
|
siteURL = scanner.Text()
|
|
if siteURL == "" {
|
|
siteURL = defaultsiteURL
|
|
}
|
|
fmt.Println("Set the site url to " + siteURL)
|
|
|
|
fmt.Println("What port do you want the server to listen on? If you don't know what this means, you should probably leave it on the default. Default: " + defaultServerPort)
|
|
if !scanner.Scan() {
|
|
return false
|
|
}
|
|
serverPort = scanner.Text()
|
|
if serverPort == "" {
|
|
serverPort = defaultServerPort
|
|
}
|
|
_, err := strconv.Atoi(serverPort)
|
|
if err != nil {
|
|
fmt.Println("That's not a valid number!")
|
|
return false
|
|
}
|
|
fmt.Println("Set the server port to " + serverPort)
|
|
return true
|
|
}
|
|
|
|
func obfuscatePassword(password string) (out string) {
|
|
for i := 0; i < len(password); i++ {
|
|
out += "*"
|
|
}
|
|
return out
|
|
}
|
|
|
|
func pressAnyKey() {
|
|
//fmt.Println("Press any key to exit...")
|
|
fmt.Println("Please press enter to exit...")
|
|
for scanner.Scan() {
|
|
_ = scanner.Text()
|
|
return
|
|
}
|
|
}
|