2018-02-20 00:24:10 +02:00
|
|
|
// Copyright 2018 Drone.IO Inc.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2015-10-22 01:14:02 +02:00
|
|
|
package datastore
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
"os"
|
2015-11-10 22:32:09 +02:00
|
|
|
"time"
|
2015-10-22 01:14:02 +02:00
|
|
|
|
|
|
|
"github.com/drone/drone/store"
|
2016-03-25 21:54:16 +02:00
|
|
|
"github.com/drone/drone/store/datastore/ddl"
|
2015-10-22 01:14:02 +02:00
|
|
|
"github.com/russross/meddler"
|
|
|
|
|
2016-03-25 21:54:16 +02:00
|
|
|
"github.com/Sirupsen/logrus"
|
2015-10-22 01:14:02 +02:00
|
|
|
)
|
|
|
|
|
2016-03-25 21:54:16 +02:00
|
|
|
// datastore is an implementation of a model.Store built on top
|
|
|
|
// of the sql/database driver with a relational database backend.
|
|
|
|
type datastore struct {
|
|
|
|
*sql.DB
|
2017-03-28 11:13:13 +02:00
|
|
|
|
|
|
|
driver string
|
|
|
|
config string
|
2016-03-25 21:54:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a database connection for the given driver and datasource
|
|
|
|
// and returns a new Store.
|
2015-10-28 01:48:05 +02:00
|
|
|
func New(driver, config string) store.Store {
|
2017-03-28 11:13:13 +02:00
|
|
|
return &datastore{
|
|
|
|
DB: open(driver, config),
|
|
|
|
driver: driver,
|
|
|
|
config: config,
|
|
|
|
}
|
2015-10-28 03:48:04 +02:00
|
|
|
}
|
|
|
|
|
2016-03-25 21:54:16 +02:00
|
|
|
// From returns a Store using an existing database connection.
|
2015-10-28 03:48:04 +02:00
|
|
|
func From(db *sql.DB) store.Store {
|
2017-03-28 11:13:13 +02:00
|
|
|
return &datastore{DB: db}
|
2015-10-22 01:14:02 +02:00
|
|
|
}
|
|
|
|
|
2016-03-25 21:54:16 +02:00
|
|
|
// open opens a new database connection with the specified
|
2015-10-22 01:14:02 +02:00
|
|
|
// driver and connection string and returns a store.
|
2016-03-25 21:54:16 +02:00
|
|
|
func open(driver, config string) *sql.DB {
|
2015-10-22 01:14:02 +02:00
|
|
|
db, err := sql.Open(driver, config)
|
|
|
|
if err != nil {
|
2016-03-25 21:54:16 +02:00
|
|
|
logrus.Errorln(err)
|
|
|
|
logrus.Fatalln("database connection failed")
|
2015-10-22 01:14:02 +02:00
|
|
|
}
|
2015-11-11 20:23:44 +02:00
|
|
|
if driver == "mysql" {
|
|
|
|
// per issue https://github.com/go-sql-driver/mysql/issues/257
|
|
|
|
db.SetMaxIdleConns(0)
|
|
|
|
}
|
2015-11-10 22:32:09 +02:00
|
|
|
|
2015-10-22 01:14:02 +02:00
|
|
|
setupMeddler(driver)
|
|
|
|
|
2015-11-10 22:32:09 +02:00
|
|
|
if err := pingDatabase(db); err != nil {
|
2016-03-25 21:54:16 +02:00
|
|
|
logrus.Errorln(err)
|
|
|
|
logrus.Fatalln("database ping attempts failed")
|
2015-11-10 22:32:09 +02:00
|
|
|
}
|
|
|
|
|
2015-10-22 01:14:02 +02:00
|
|
|
if err := setupDatabase(driver, db); err != nil {
|
2016-03-25 21:54:16 +02:00
|
|
|
logrus.Errorln(err)
|
|
|
|
logrus.Fatalln("migration failed")
|
2015-10-22 01:14:02 +02:00
|
|
|
}
|
|
|
|
return db
|
|
|
|
}
|
|
|
|
|
2017-03-28 11:13:13 +02:00
|
|
|
// openTest opens a new database connection for testing purposes.
|
2015-10-22 01:14:02 +02:00
|
|
|
// The database driver and connection string are provided by
|
|
|
|
// environment variables, with fallback to in-memory sqlite.
|
|
|
|
func openTest() *sql.DB {
|
|
|
|
var (
|
|
|
|
driver = "sqlite3"
|
|
|
|
config = ":memory:"
|
|
|
|
)
|
|
|
|
if os.Getenv("DATABASE_DRIVER") != "" {
|
|
|
|
driver = os.Getenv("DATABASE_DRIVER")
|
2016-03-25 22:24:16 +02:00
|
|
|
config = os.Getenv("DATABASE_CONFIG")
|
2015-10-22 01:14:02 +02:00
|
|
|
}
|
2016-03-25 21:54:16 +02:00
|
|
|
return open(driver, config)
|
2015-10-22 01:14:02 +02:00
|
|
|
}
|
|
|
|
|
2017-03-28 11:13:13 +02:00
|
|
|
// newTest creates a new database connection for testing purposes.
|
|
|
|
// The database driver and connection string are provided by
|
|
|
|
// environment variables, with fallback to in-memory sqlite.
|
|
|
|
func newTest() *datastore {
|
|
|
|
var (
|
|
|
|
driver = "sqlite3"
|
|
|
|
config = ":memory:"
|
|
|
|
)
|
|
|
|
if os.Getenv("DATABASE_DRIVER") != "" {
|
|
|
|
driver = os.Getenv("DATABASE_DRIVER")
|
|
|
|
config = os.Getenv("DATABASE_CONFIG")
|
|
|
|
}
|
|
|
|
return &datastore{
|
|
|
|
DB: open(driver, config),
|
|
|
|
driver: driver,
|
|
|
|
config: config,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-10 22:32:09 +02:00
|
|
|
// helper function to ping the database with backoff to ensure
|
|
|
|
// a connection can be established before we proceed with the
|
|
|
|
// database setup and migration.
|
|
|
|
func pingDatabase(db *sql.DB) (err error) {
|
|
|
|
for i := 0; i < 30; i++ {
|
|
|
|
err = db.Ping()
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
2016-03-25 21:54:16 +02:00
|
|
|
logrus.Infof("database ping failed. retry in 1s")
|
2015-11-10 22:32:09 +02:00
|
|
|
time.Sleep(time.Second)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-10-22 01:14:02 +02:00
|
|
|
// helper function to setup the databsae by performing
|
|
|
|
// automated database migration steps.
|
|
|
|
func setupDatabase(driver string, db *sql.DB) error {
|
2017-05-13 09:56:23 +02:00
|
|
|
return ddl.Migrate(driver, db)
|
2015-10-22 01:14:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// helper function to setup the meddler default driver
|
|
|
|
// based on the selected driver name.
|
|
|
|
func setupMeddler(driver string) {
|
|
|
|
switch driver {
|
|
|
|
case "sqlite3":
|
|
|
|
meddler.Default = meddler.SQLite
|
|
|
|
case "mysql":
|
|
|
|
meddler.Default = meddler.MySQL
|
|
|
|
case "postgres":
|
|
|
|
meddler.Default = meddler.PostgreSQL
|
|
|
|
}
|
|
|
|
}
|