http urls monitor.

builder.go 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621
  1. // Copyright (c) 2012-present The upper.io/db authors. All rights reserved.
  2. //
  3. // Permission is hereby granted, free of charge, to any person obtaining
  4. // a copy of this software and associated documentation files (the
  5. // "Software"), to deal in the Software without restriction, including
  6. // without limitation the rights to use, copy, modify, merge, publish,
  7. // distribute, sublicense, and/or sell copies of the Software, and to
  8. // permit persons to whom the Software is furnished to do so, subject to
  9. // the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be
  12. // included in all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  15. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  16. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  17. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  18. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  19. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  20. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  21. // Package sqlbuilder provides tools for building custom SQL queries.
  22. package sqlbuilder
  23. import (
  24. "context"
  25. "database/sql"
  26. "errors"
  27. "fmt"
  28. "log"
  29. "reflect"
  30. "regexp"
  31. "sort"
  32. "strconv"
  33. "strings"
  34. "upper.io/db.v3"
  35. "upper.io/db.v3/internal/sqladapter/compat"
  36. "upper.io/db.v3/internal/sqladapter/exql"
  37. "upper.io/db.v3/lib/reflectx"
  38. )
  39. // MapOptions represents options for the mapper.
  40. type MapOptions struct {
  41. IncludeZeroed bool
  42. IncludeNil bool
  43. }
  44. var defaultMapOptions = MapOptions{
  45. IncludeZeroed: false,
  46. IncludeNil: false,
  47. }
  48. type compilable interface {
  49. Compile() (string, error)
  50. Arguments() []interface{}
  51. }
  52. type hasIsZero interface {
  53. IsZero() bool
  54. }
  55. type hasArguments interface {
  56. Arguments() []interface{}
  57. }
  58. type hasStatement interface {
  59. statement() *exql.Statement
  60. }
  61. type iterator struct {
  62. sess exprDB
  63. cursor *sql.Rows // This is the main query cursor. It starts as a nil value.
  64. err error
  65. }
  66. type fieldValue struct {
  67. fields []string
  68. values []interface{}
  69. }
  70. var (
  71. reInvisibleChars = regexp.MustCompile(`[\s\r\n\t]+`)
  72. reColumnCompareExclude = regexp.MustCompile(`[^a-zA-Z0-9]`)
  73. )
  74. var (
  75. sqlPlaceholder = exql.RawValue(`?`)
  76. )
  77. var (
  78. errDeprecatedJSONBTag = errors.New(`Tag "jsonb" is deprecated. See "PostgreSQL: jsonb tag" at https://github.com/upper/db/releases/tag/v3.4.0`)
  79. )
  80. type exprDB interface {
  81. StatementExec(ctx context.Context, stmt *exql.Statement, args ...interface{}) (sql.Result, error)
  82. StatementPrepare(ctx context.Context, stmt *exql.Statement) (*sql.Stmt, error)
  83. StatementQuery(ctx context.Context, stmt *exql.Statement, args ...interface{}) (*sql.Rows, error)
  84. StatementQueryRow(ctx context.Context, stmt *exql.Statement, args ...interface{}) (*sql.Row, error)
  85. Context() context.Context
  86. }
  87. type sqlBuilder struct {
  88. sess exprDB
  89. t *templateWithUtils
  90. }
  91. // WithSession returns a query builder that is bound to the given database session.
  92. func WithSession(sess interface{}, t *exql.Template) SQLBuilder {
  93. if sqlDB, ok := sess.(*sql.DB); ok {
  94. sess = sqlDB
  95. }
  96. return &sqlBuilder{
  97. sess: sess.(exprDB), // Let it panic, it will show the developer an informative error.
  98. t: newTemplateWithUtils(t),
  99. }
  100. }
  101. // WithTemplate returns a builder that is based on the given template.
  102. func WithTemplate(t *exql.Template) SQLBuilder {
  103. return &sqlBuilder{
  104. t: newTemplateWithUtils(t),
  105. }
  106. }
  107. // NewIterator creates an iterator using the given *sql.Rows.
  108. func NewIterator(rows *sql.Rows) Iterator {
  109. return &iterator{nil, rows, nil}
  110. }
  111. func (b *sqlBuilder) Iterator(query interface{}, args ...interface{}) Iterator {
  112. return b.IteratorContext(b.sess.Context(), query, args...)
  113. }
  114. func (b *sqlBuilder) IteratorContext(ctx context.Context, query interface{}, args ...interface{}) Iterator {
  115. rows, err := b.QueryContext(ctx, query, args...)
  116. return &iterator{b.sess, rows, err}
  117. }
  118. func (b *sqlBuilder) Prepare(query interface{}) (*sql.Stmt, error) {
  119. return b.PrepareContext(b.sess.Context(), query)
  120. }
  121. func (b *sqlBuilder) PrepareContext(ctx context.Context, query interface{}) (*sql.Stmt, error) {
  122. switch q := query.(type) {
  123. case *exql.Statement:
  124. return b.sess.StatementPrepare(ctx, q)
  125. case string:
  126. return b.sess.StatementPrepare(ctx, exql.RawSQL(q))
  127. case db.RawValue:
  128. return b.PrepareContext(ctx, q.Raw())
  129. default:
  130. return nil, fmt.Errorf("unsupported query type %T", query)
  131. }
  132. }
  133. func (b *sqlBuilder) Exec(query interface{}, args ...interface{}) (sql.Result, error) {
  134. return b.ExecContext(b.sess.Context(), query, args...)
  135. }
  136. func (b *sqlBuilder) ExecContext(ctx context.Context, query interface{}, args ...interface{}) (sql.Result, error) {
  137. switch q := query.(type) {
  138. case *exql.Statement:
  139. return b.sess.StatementExec(ctx, q, args...)
  140. case string:
  141. return b.sess.StatementExec(ctx, exql.RawSQL(q), args...)
  142. case db.RawValue:
  143. return b.ExecContext(ctx, q.Raw(), q.Arguments()...)
  144. default:
  145. return nil, fmt.Errorf("unsupported query type %T", query)
  146. }
  147. }
  148. func (b *sqlBuilder) Query(query interface{}, args ...interface{}) (*sql.Rows, error) {
  149. return b.QueryContext(b.sess.Context(), query, args...)
  150. }
  151. func (b *sqlBuilder) QueryContext(ctx context.Context, query interface{}, args ...interface{}) (*sql.Rows, error) {
  152. switch q := query.(type) {
  153. case *exql.Statement:
  154. return b.sess.StatementQuery(ctx, q, args...)
  155. case string:
  156. return b.sess.StatementQuery(ctx, exql.RawSQL(q), args...)
  157. case db.RawValue:
  158. return b.QueryContext(ctx, q.Raw(), q.Arguments()...)
  159. default:
  160. return nil, fmt.Errorf("unsupported query type %T", query)
  161. }
  162. }
  163. func (b *sqlBuilder) QueryRow(query interface{}, args ...interface{}) (*sql.Row, error) {
  164. return b.QueryRowContext(b.sess.Context(), query, args...)
  165. }
  166. func (b *sqlBuilder) QueryRowContext(ctx context.Context, query interface{}, args ...interface{}) (*sql.Row, error) {
  167. switch q := query.(type) {
  168. case *exql.Statement:
  169. return b.sess.StatementQueryRow(ctx, q, args...)
  170. case string:
  171. return b.sess.StatementQueryRow(ctx, exql.RawSQL(q), args...)
  172. case db.RawValue:
  173. return b.QueryRowContext(ctx, q.Raw(), q.Arguments()...)
  174. default:
  175. return nil, fmt.Errorf("unsupported query type %T", query)
  176. }
  177. }
  178. func (b *sqlBuilder) SelectFrom(table ...interface{}) Selector {
  179. qs := &selector{
  180. builder: b,
  181. }
  182. return qs.From(table...)
  183. }
  184. func (b *sqlBuilder) Select(columns ...interface{}) Selector {
  185. qs := &selector{
  186. builder: b,
  187. }
  188. return qs.Columns(columns...)
  189. }
  190. func (b *sqlBuilder) InsertInto(table string) Inserter {
  191. qi := &inserter{
  192. builder: b,
  193. }
  194. return qi.Into(table)
  195. }
  196. func (b *sqlBuilder) DeleteFrom(table string) Deleter {
  197. qd := &deleter{
  198. builder: b,
  199. }
  200. return qd.setTable(table)
  201. }
  202. func (b *sqlBuilder) Update(table string) Updater {
  203. qu := &updater{
  204. builder: b,
  205. }
  206. return qu.setTable(table)
  207. }
  208. // Map receives a pointer to map or struct and maps it to columns and values.
  209. func Map(item interface{}, options *MapOptions) ([]string, []interface{}, error) {
  210. var fv fieldValue
  211. if options == nil {
  212. options = &defaultMapOptions
  213. }
  214. itemV := reflect.ValueOf(item)
  215. if !itemV.IsValid() {
  216. return nil, nil, nil
  217. }
  218. itemT := itemV.Type()
  219. if itemT.Kind() == reflect.Ptr {
  220. // Single dereference. Just in case the user passes a pointer to struct
  221. // instead of a struct.
  222. item = itemV.Elem().Interface()
  223. itemV = reflect.ValueOf(item)
  224. itemT = itemV.Type()
  225. }
  226. switch itemT.Kind() {
  227. case reflect.Struct:
  228. fieldMap := mapper.TypeMap(itemT).Names
  229. nfields := len(fieldMap)
  230. fv.values = make([]interface{}, 0, nfields)
  231. fv.fields = make([]string, 0, nfields)
  232. for _, fi := range fieldMap {
  233. // Check for deprecated JSONB tag
  234. if _, hasJSONBTag := fi.Options["jsonb"]; hasJSONBTag {
  235. return nil, nil, errDeprecatedJSONBTag
  236. }
  237. // Field options
  238. _, tagOmitEmpty := fi.Options["omitempty"]
  239. fld := reflectx.FieldByIndexesReadOnly(itemV, fi.Index)
  240. if fld.Kind() == reflect.Ptr && fld.IsNil() {
  241. if tagOmitEmpty && !options.IncludeNil {
  242. continue
  243. }
  244. fv.fields = append(fv.fields, fi.Name)
  245. if tagOmitEmpty {
  246. fv.values = append(fv.values, sqlDefault)
  247. } else {
  248. fv.values = append(fv.values, nil)
  249. }
  250. continue
  251. }
  252. value := fld.Interface()
  253. isZero := false
  254. if t, ok := fld.Interface().(hasIsZero); ok {
  255. if t.IsZero() {
  256. isZero = true
  257. }
  258. } else if fld.Kind() == reflect.Array || fld.Kind() == reflect.Slice {
  259. if fld.Len() == 0 {
  260. isZero = true
  261. }
  262. } else if reflect.DeepEqual(fi.Zero.Interface(), value) {
  263. isZero = true
  264. }
  265. if isZero && tagOmitEmpty && !options.IncludeZeroed {
  266. continue
  267. }
  268. fv.fields = append(fv.fields, fi.Name)
  269. v, err := marshal(value)
  270. if err != nil {
  271. return nil, nil, err
  272. }
  273. if isZero && tagOmitEmpty {
  274. v = sqlDefault
  275. }
  276. fv.values = append(fv.values, v)
  277. }
  278. case reflect.Map:
  279. nfields := itemV.Len()
  280. fv.values = make([]interface{}, nfields)
  281. fv.fields = make([]string, nfields)
  282. mkeys := itemV.MapKeys()
  283. for i, keyV := range mkeys {
  284. valv := itemV.MapIndex(keyV)
  285. fv.fields[i] = fmt.Sprintf("%v", keyV.Interface())
  286. v, err := marshal(valv.Interface())
  287. if err != nil {
  288. return nil, nil, err
  289. }
  290. fv.values[i] = v
  291. }
  292. default:
  293. return nil, nil, ErrExpectingPointerToEitherMapOrStruct
  294. }
  295. sort.Sort(&fv)
  296. return fv.fields, fv.values, nil
  297. }
  298. func extractArguments(fragments []interface{}) []interface{} {
  299. args := []interface{}{}
  300. l := len(fragments)
  301. for i := 0; i < l; i++ {
  302. switch v := fragments[i].(type) {
  303. case hasArguments: // TODO: use this on other places where we want to extract arguments.
  304. args = append(args, v.Arguments()...)
  305. }
  306. }
  307. return args
  308. }
  309. func columnFragments(columns []interface{}) ([]exql.Fragment, []interface{}, error) {
  310. l := len(columns)
  311. f := make([]exql.Fragment, l)
  312. args := []interface{}{}
  313. for i := 0; i < l; i++ {
  314. switch v := columns[i].(type) {
  315. case compilable:
  316. c, err := v.Compile()
  317. if err != nil {
  318. return nil, nil, err
  319. }
  320. q, a := Preprocess(c, v.Arguments())
  321. if _, ok := v.(Selector); ok {
  322. q = "(" + q + ")"
  323. }
  324. f[i] = exql.RawValue(q)
  325. args = append(args, a...)
  326. case db.Function:
  327. fnName, fnArgs := v.Name(), v.Arguments()
  328. if len(fnArgs) == 0 {
  329. fnName = fnName + "()"
  330. } else {
  331. fnName = fnName + "(?" + strings.Repeat("?, ", len(fnArgs)-1) + ")"
  332. }
  333. fnName, fnArgs = Preprocess(fnName, fnArgs)
  334. f[i] = exql.RawValue(fnName)
  335. args = append(args, fnArgs...)
  336. case db.RawValue:
  337. q, a := Preprocess(v.Raw(), v.Arguments())
  338. f[i] = exql.RawValue(q)
  339. args = append(args, a...)
  340. case exql.Fragment:
  341. f[i] = v
  342. case string:
  343. f[i] = exql.ColumnWithName(v)
  344. case int:
  345. f[i] = exql.RawValue(fmt.Sprintf("%v", v))
  346. case interface{}:
  347. f[i] = exql.ColumnWithName(fmt.Sprintf("%v", v))
  348. default:
  349. return nil, nil, fmt.Errorf("unexpected argument type %T for Select() argument", v)
  350. }
  351. }
  352. return f, args, nil
  353. }
  354. func prepareQueryForDisplay(in string) (out string) {
  355. j := 1
  356. for i := range in {
  357. if in[i] == '?' {
  358. out = out + "$" + strconv.Itoa(j)
  359. j++
  360. } else {
  361. out = out + string(in[i])
  362. }
  363. }
  364. out = reInvisibleChars.ReplaceAllString(out, ` `)
  365. return strings.TrimSpace(out)
  366. }
  367. func (iter *iterator) NextScan(dst ...interface{}) error {
  368. if ok := iter.Next(); ok {
  369. return iter.Scan(dst...)
  370. }
  371. if err := iter.Err(); err != nil {
  372. return err
  373. }
  374. return db.ErrNoMoreRows
  375. }
  376. func (iter *iterator) ScanOne(dst ...interface{}) error {
  377. defer iter.Close()
  378. return iter.NextScan(dst...)
  379. }
  380. func (iter *iterator) Scan(dst ...interface{}) error {
  381. if err := iter.Err(); err != nil {
  382. return err
  383. }
  384. return iter.cursor.Scan(dst...)
  385. }
  386. func (iter *iterator) setErr(err error) error {
  387. iter.err = err
  388. return iter.err
  389. }
  390. func (iter *iterator) One(dst interface{}) error {
  391. if err := iter.Err(); err != nil {
  392. return err
  393. }
  394. defer iter.Close()
  395. return iter.setErr(iter.next(dst))
  396. }
  397. func (iter *iterator) All(dst interface{}) error {
  398. if err := iter.Err(); err != nil {
  399. return err
  400. }
  401. defer iter.Close()
  402. // Fetching all results within the cursor.
  403. if err := fetchRows(iter, dst); err != nil {
  404. return iter.setErr(err)
  405. }
  406. return nil
  407. }
  408. func (iter *iterator) Err() (err error) {
  409. return iter.err
  410. }
  411. func (iter *iterator) Next(dst ...interface{}) bool {
  412. if err := iter.Err(); err != nil {
  413. return false
  414. }
  415. if err := iter.next(dst...); err != nil {
  416. // ignore db.ErrNoMoreRows, just break.
  417. if err != db.ErrNoMoreRows {
  418. iter.setErr(err)
  419. }
  420. return false
  421. }
  422. return true
  423. }
  424. func (iter *iterator) next(dst ...interface{}) error {
  425. if iter.cursor == nil {
  426. return iter.setErr(db.ErrNoMoreRows)
  427. }
  428. switch len(dst) {
  429. case 0:
  430. if ok := iter.cursor.Next(); !ok {
  431. defer iter.Close()
  432. err := iter.cursor.Err()
  433. if err == nil {
  434. err = db.ErrNoMoreRows
  435. }
  436. return err
  437. }
  438. return nil
  439. case 1:
  440. if err := fetchRow(iter, dst[0]); err != nil {
  441. defer iter.Close()
  442. return err
  443. }
  444. return nil
  445. }
  446. return errors.New("Next does not currently supports more than one parameters")
  447. }
  448. func (iter *iterator) Close() (err error) {
  449. if iter.cursor != nil {
  450. err = iter.cursor.Close()
  451. iter.cursor = nil
  452. }
  453. return err
  454. }
  455. func marshal(v interface{}) (interface{}, error) {
  456. if m, isMarshaler := v.(db.Marshaler); isMarshaler {
  457. var err error
  458. if v, err = m.MarshalDB(); err != nil {
  459. return nil, err
  460. }
  461. }
  462. return v, nil
  463. }
  464. func (fv *fieldValue) Len() int {
  465. return len(fv.fields)
  466. }
  467. func (fv *fieldValue) Swap(i, j int) {
  468. fv.fields[i], fv.fields[j] = fv.fields[j], fv.fields[i]
  469. fv.values[i], fv.values[j] = fv.values[j], fv.values[i]
  470. }
  471. func (fv *fieldValue) Less(i, j int) bool {
  472. return fv.fields[i] < fv.fields[j]
  473. }
  474. type exprProxy struct {
  475. db *sql.DB
  476. t *exql.Template
  477. }
  478. func newSqlgenProxy(db *sql.DB, t *exql.Template) *exprProxy {
  479. return &exprProxy{db: db, t: t}
  480. }
  481. func (p *exprProxy) Context() context.Context {
  482. log.Printf("Missing context")
  483. return context.Background()
  484. }
  485. func (p *exprProxy) StatementExec(ctx context.Context, stmt *exql.Statement, args ...interface{}) (sql.Result, error) {
  486. s, err := stmt.Compile(p.t)
  487. if err != nil {
  488. return nil, err
  489. }
  490. return compat.ExecContext(p.db, ctx, s, args)
  491. }
  492. func (p *exprProxy) StatementPrepare(ctx context.Context, stmt *exql.Statement) (*sql.Stmt, error) {
  493. s, err := stmt.Compile(p.t)
  494. if err != nil {
  495. return nil, err
  496. }
  497. return compat.PrepareContext(p.db, ctx, s)
  498. }
  499. func (p *exprProxy) StatementQuery(ctx context.Context, stmt *exql.Statement, args ...interface{}) (*sql.Rows, error) {
  500. s, err := stmt.Compile(p.t)
  501. if err != nil {
  502. return nil, err
  503. }
  504. return compat.QueryContext(p.db, ctx, s, args)
  505. }
  506. func (p *exprProxy) StatementQueryRow(ctx context.Context, stmt *exql.Statement, args ...interface{}) (*sql.Row, error) {
  507. s, err := stmt.Compile(p.t)
  508. if err != nil {
  509. return nil, err
  510. }
  511. return compat.QueryRowContext(p.db, ctx, s, args), nil
  512. }
  513. var (
  514. _ = SQLBuilder(&sqlBuilder{})
  515. _ = exprDB(&exprProxy{})
  516. )
  517. func joinArguments(args ...[]interface{}) []interface{} {
  518. total := 0
  519. for i := range args {
  520. total += len(args[i])
  521. }
  522. if total == 0 {
  523. return nil
  524. }
  525. flatten := make([]interface{}, 0, total)
  526. for i := range args {
  527. flatten = append(flatten, args[i]...)
  528. }
  529. return flatten
  530. }