2014-08-18 16:41:43 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// 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
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// 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.
2013-07-30 20:32:38 +08:00
package orm
2013-08-11 00:15:26 +08:00
import (
2013-08-11 22:27:45 +08:00
"fmt"
2013-08-11 00:15:26 +08:00
"strconv"
)
2014-01-17 17:25:17 +08:00
// postgresql operators.
2013-08-11 00:15:26 +08:00
var postgresOperators = map [ string ] string {
"exact" : "= ?" ,
"iexact" : "= UPPER(?)" ,
"contains" : "LIKE ?" ,
"icontains" : "LIKE UPPER(?)" ,
"gt" : "> ?" ,
"gte" : ">= ?" ,
"lt" : "< ?" ,
"lte" : "<= ?" ,
"startswith" : "LIKE ?" ,
"endswith" : "LIKE ?" ,
"istartswith" : "LIKE UPPER(?)" ,
"iendswith" : "LIKE UPPER(?)" ,
}
2014-01-17 17:25:17 +08:00
// postgresql column field types.
2013-08-19 22:37:39 +08:00
var postgresTypes = map [ string ] string {
"auto" : "serial NOT NULL PRIMARY KEY" ,
"pk" : "NOT NULL PRIMARY KEY" ,
"bool" : "bool" ,
"string" : "varchar(%d)" ,
"string-text" : "text" ,
"time.Time-date" : "date" ,
"time.Time" : "timestamp with time zone" ,
"int8" : ` smallint CHECK("%COL%" >= -127 AND "%COL%" <= 128) ` ,
"int16" : "smallint" ,
"int32" : "integer" ,
"int64" : "bigint" ,
"uint8" : ` smallint CHECK("%COL%" >= 0 AND "%COL%" <= 255) ` ,
"uint16" : ` integer CHECK("%COL%" >= 0) ` ,
"uint32" : ` bigint CHECK("%COL%" >= 0) ` ,
"uint64" : ` bigint CHECK("%COL%" >= 0) ` ,
"float64" : "double precision" ,
"float64-decimal" : "numeric(%d, %d)" ,
}
2014-01-17 17:25:17 +08:00
// postgresql dbBaser.
2013-07-30 20:32:38 +08:00
type dbBasePostgres struct {
dbBase
}
2013-08-11 00:15:26 +08:00
var _ dbBaser = new ( dbBasePostgres )
2014-01-17 17:25:17 +08:00
// get postgresql operator.
2013-08-11 00:15:26 +08:00
func ( d * dbBasePostgres ) OperatorSql ( operator string ) string {
return postgresOperators [ operator ]
}
2014-01-17 17:25:17 +08:00
// generate functioned sql string, such as contains(text).
2013-08-13 17:16:12 +08:00
func ( d * dbBasePostgres ) GenerateOperatorLeftCol ( fi * fieldInfo , operator string , leftCol * string ) {
2013-08-11 22:27:45 +08:00
switch operator {
case "contains" , "startswith" , "endswith" :
* leftCol = fmt . Sprintf ( "%s::text" , * leftCol )
case "iexact" , "icontains" , "istartswith" , "iendswith" :
* leftCol = fmt . Sprintf ( "UPPER(%s::text)" , * leftCol )
}
}
2014-01-17 17:25:17 +08:00
// postgresql unsupports updating joined record.
2013-08-11 22:27:45 +08:00
func ( d * dbBasePostgres ) SupportUpdateJoin ( ) bool {
return false
}
func ( d * dbBasePostgres ) MaxLimit ( ) uint64 {
return 0
}
2014-01-17 17:25:17 +08:00
// postgresql quote is ".
2013-08-11 00:15:26 +08:00
func ( d * dbBasePostgres ) TableQuote ( ) string {
return ` " `
}
2014-01-17 17:25:17 +08:00
// postgresql value placeholder is $n.
// replace default ? to $n.
2013-08-11 00:15:26 +08:00
func ( d * dbBasePostgres ) ReplaceMarks ( query * string ) {
q := * query
num := 0
for _ , c := range q {
if c == '?' {
num += 1
}
}
if num == 0 {
return
}
data := make ( [ ] byte , 0 , len ( q ) + num )
num = 1
for i := 0 ; i < len ( q ) ; i ++ {
c := q [ i ]
if c == '?' {
data = append ( data , '$' )
data = append ( data , [ ] byte ( strconv . Itoa ( num ) ) ... )
num += 1
} else {
data = append ( data , c )
}
}
* query = string ( data )
}
2014-01-17 17:25:17 +08:00
// make returning sql support for postgresql.
2013-08-11 22:27:45 +08:00
func ( d * dbBasePostgres ) HasReturningID ( mi * modelInfo , query * string ) ( has bool ) {
if mi . fields . pk . auto {
if query != nil {
* query = fmt . Sprintf ( ` %s RETURNING "%s" ` , * query , mi . fields . pk . column )
}
has = true
}
return
}
2013-08-11 00:15:26 +08:00
2014-01-17 17:25:17 +08:00
// show table sql for postgresql.
2013-08-27 12:33:27 +08:00
func ( d * dbBasePostgres ) ShowTablesQuery ( ) string {
return "SELECT table_name FROM information_schema.tables WHERE table_type = 'BASE TABLE' AND table_schema NOT IN ('pg_catalog', 'information_schema')"
}
2014-01-17 17:25:17 +08:00
// show table columns sql for postgresql.
2013-08-27 12:33:27 +08:00
func ( d * dbBasePostgres ) ShowColumnsQuery ( table string ) string {
return fmt . Sprintf ( "SELECT column_name, data_type, is_nullable FROM information_schema.columns where table_schema NOT IN ('pg_catalog', 'information_schema') and table_name = '%s'" , table )
}
2014-01-17 17:25:17 +08:00
// get column types of postgresql.
2013-08-19 22:37:39 +08:00
func ( d * dbBasePostgres ) DbTypes ( ) map [ string ] string {
return postgresTypes
}
2014-01-17 17:25:17 +08:00
// check index exist in postgresql.
2013-08-27 12:33:27 +08:00
func ( d * dbBasePostgres ) IndexExists ( db dbQuerier , table string , name string ) bool {
query := fmt . Sprintf ( "SELECT COUNT(*) FROM pg_indexes WHERE tablename = '%s' AND indexname = '%s'" , table , name )
row := db . QueryRow ( query )
var cnt int
row . Scan ( & cnt )
return cnt > 0
}
2014-01-17 17:25:17 +08:00
// create new postgresql dbBaser.
2013-07-30 20:32:38 +08:00
func newdbBasePostgres ( ) dbBaser {
b := new ( dbBasePostgres )
b . ins = b
return b
}