mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-05-31 23:19:39 +08:00
- 同步分析和预览改为分页扫描差异,避免一次性加载源表和目标表 - 直接导入与源查询同步支持分页读取和分批提交,降低低内存机器 OOM 风险 - 各数据库 ApplyChanges 统一使用参数化批量 INSERT,减少大表同步 SQL 超时 - MySQL 批量写入按行数和参数数量拆分,兼容超宽表场景 - 补充批量插入、分页差异和源查询同步回归测试
588 lines
17 KiB
Go
588 lines
17 KiB
Go
package sync
|
|
|
|
import (
|
|
"GoNavi-Wails/internal/connection"
|
|
"GoNavi-Wails/internal/db"
|
|
"fmt"
|
|
"reflect"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
type pagedDiffCounts struct {
|
|
Inserts int
|
|
Updates int
|
|
Deletes int
|
|
Same int
|
|
}
|
|
|
|
type pagedUpdateDiff struct {
|
|
UpdateRow connection.UpdateRow
|
|
Source map[string]interface{}
|
|
Target map[string]interface{}
|
|
ChangedColumns []string
|
|
}
|
|
|
|
type pagedDiffPage struct {
|
|
Inserts []map[string]interface{}
|
|
Updates []pagedUpdateDiff
|
|
Deletes []map[string]interface{}
|
|
Same int
|
|
}
|
|
|
|
func (s *SyncEngine) tryApplyDiffInPages(config SyncConfig, res *SyncResult, tableIndex, totalTables int, tableName string, sourceDB db.Database, targetDB db.Database, plan SchemaMigrationPlan, sourceCols, targetCols []connection.ColumnDefinition, opts TableOptions, sourceType, targetType, applyTableName, pkCol string) (bool, pagedDiffCounts, error) {
|
|
if normalizeSyncMode(config.Mode) != "insert_update" || !plan.TargetTableExists {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
if !supportsPagedDiffSelect(sourceType) || !supportsPagedDiffSelect(targetType) {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
if opts.Delete && (!supportsPagedDiffKeysetSelect(targetType) || !supportsPagedDiffPKLookup(sourceType)) {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
|
|
applier, ok := targetDB.(db.BatchApplier)
|
|
if !ok {
|
|
return true, pagedDiffCounts{}, fmt.Errorf("目标驱动不支持应用数据变更 (ApplyChanges)")
|
|
}
|
|
|
|
targetColSet, err := s.prepareDirectImportTargetColumnSet(config, res, targetDB, plan, sourceType, targetType, sourceCols, targetCols)
|
|
if err != nil {
|
|
return true, pagedDiffCounts{}, err
|
|
}
|
|
|
|
s.appendLog(config.JobID, res, "info", fmt.Sprintf(" -> 启用分页差异同步:按主键 %s 每批读取 %d 行", pkCol, defaultSyncReadPageSize))
|
|
s.progress(config.JobID, tableIndex, totalTables, tableName, "分页对比数据")
|
|
|
|
applied := pagedDiffCounts{}
|
|
handled, _, err := scanTableDiffInPages(sourceDB, targetDB, sourceType, targetType, plan, sourceCols, targetCols, pkCol, targetColSet, opts.Delete, func(page pagedDiffPage) error {
|
|
changeSet := connection.ChangeSet{
|
|
Inserts: filterRowsByPKSelection(pkCol, page.Inserts, opts.Insert, opts.SelectedInsertPKs),
|
|
Updates: filterPagedUpdatesByPKSelection(pkCol, page.Updates, opts.Update, opts.SelectedUpdatePKs),
|
|
Deletes: filterRowsByPKSelection(pkCol, page.Deletes, opts.Delete, opts.SelectedDeletePKs),
|
|
}
|
|
if len(targetColSet) > 0 {
|
|
changeSet.Inserts = filterInsertRows(changeSet.Inserts, targetColSet)
|
|
changeSet.Updates = filterUpdateRows(changeSet.Updates, targetColSet)
|
|
}
|
|
if len(changeSet.Inserts) == 0 && len(changeSet.Updates) == 0 && len(changeSet.Deletes) == 0 {
|
|
return nil
|
|
}
|
|
if err := s.applyChangesInBatches(config.JobID, res, applyTableName, applier, changeSet); err != nil {
|
|
return err
|
|
}
|
|
applied.Inserts += len(changeSet.Inserts)
|
|
applied.Updates += len(changeSet.Updates)
|
|
applied.Deletes += len(changeSet.Deletes)
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return true, applied, err
|
|
}
|
|
return handled, applied, nil
|
|
}
|
|
|
|
func scanTableDiffInPages(sourceDB db.Database, targetDB db.Database, sourceType, targetType string, plan SchemaMigrationPlan, sourceCols, targetCols []connection.ColumnDefinition, pkCol string, targetColSet map[string]struct{}, includeDeletes bool, consume func(page pagedDiffPage) error) (bool, pagedDiffCounts, error) {
|
|
if !supportsPagedDiffSelect(sourceType) || !supportsPagedDiffPKLookup(targetType) {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
if includeDeletes && (!supportsPagedDiffKeysetSelect(targetType) || !supportsPagedDiffPKLookup(sourceType)) {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
|
|
sourceReadCols := diffReadableColumns(sourceCols, targetColSet, pkCol)
|
|
if len(sourceReadCols) == 0 {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
targetLookupCols := diffLookupColumns(sourceReadCols, targetCols, targetColSet, pkCol)
|
|
if len(targetLookupCols) == 0 {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
|
|
totals := pagedDiffCounts{}
|
|
for offset := 0; ; offset += defaultSyncReadPageSize {
|
|
query := buildPagedSourceTableQuery(sourceType, plan.SourceQueryTable, sourceReadCols, pkCol, defaultSyncReadPageSize, offset)
|
|
if strings.TrimSpace(query) == "" {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
sourceRows, _, err := sourceDB.Query(query)
|
|
if err != nil {
|
|
return true, totals, fmt.Errorf("分页读取源表失败(offset=%d): %w", offset, err)
|
|
}
|
|
if len(sourceRows) == 0 {
|
|
break
|
|
}
|
|
|
|
pkValues := collectPKValues(sourceRows, pkCol)
|
|
targetRows := make([]map[string]interface{}, 0)
|
|
if len(pkValues) > 0 {
|
|
targetQuery := buildPKInSelectQuery(targetType, plan.TargetQueryTable, targetLookupCols, pkCol, pkValues)
|
|
if strings.TrimSpace(targetQuery) == "" {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
targetRows, _, err = targetDB.Query(targetQuery)
|
|
if err != nil {
|
|
return true, totals, fmt.Errorf("按主键读取目标表失败(offset=%d): %w", offset, err)
|
|
}
|
|
}
|
|
|
|
page := diffSourcePageByPK(pkCol, sourceRows, targetRows)
|
|
totals.Inserts += len(page.Inserts)
|
|
totals.Updates += len(page.Updates)
|
|
totals.Same += page.Same
|
|
if consume != nil {
|
|
if err := consume(page); err != nil {
|
|
return true, totals, err
|
|
}
|
|
}
|
|
if len(sourceRows) < defaultSyncReadPageSize {
|
|
break
|
|
}
|
|
}
|
|
|
|
if includeDeletes {
|
|
lastPK, hasLastPK := interface{}(nil), false
|
|
targetPKCols := []connection.ColumnDefinition{{Name: pkCol}}
|
|
for {
|
|
query := buildKeysetPagedTableQuery(targetType, plan.TargetQueryTable, targetPKCols, pkCol, lastPK, hasLastPK, defaultSyncReadPageSize)
|
|
if strings.TrimSpace(query) == "" {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
targetRows, _, err := targetDB.Query(query)
|
|
if err != nil {
|
|
return true, totals, fmt.Errorf("分页读取目标主键失败: %w", err)
|
|
}
|
|
if len(targetRows) == 0 {
|
|
break
|
|
}
|
|
|
|
nextLastPK, ok := lastValidPKValue(targetRows, pkCol)
|
|
if !ok {
|
|
break
|
|
}
|
|
lastPK, hasLastPK = nextLastPK, true
|
|
|
|
pkValues := collectPKValues(targetRows, pkCol)
|
|
sourcePKRows := make([]map[string]interface{}, 0)
|
|
if len(pkValues) > 0 {
|
|
sourceQuery := buildPKInSelectQuery(sourceType, plan.SourceQueryTable, targetPKCols, pkCol, pkValues)
|
|
if strings.TrimSpace(sourceQuery) == "" {
|
|
return false, pagedDiffCounts{}, nil
|
|
}
|
|
sourcePKRows, _, err = sourceDB.Query(sourceQuery)
|
|
if err != nil {
|
|
return true, totals, fmt.Errorf("按主键反查源表失败: %w", err)
|
|
}
|
|
}
|
|
|
|
sourcePKSet := buildPKSet(sourcePKRows, pkCol)
|
|
deletes := make([]map[string]interface{}, 0)
|
|
for _, row := range targetRows {
|
|
pkKey, ok := pkValueKey(row[pkCol])
|
|
if !ok {
|
|
continue
|
|
}
|
|
if _, exists := sourcePKSet[pkKey]; exists {
|
|
continue
|
|
}
|
|
deletes = append(deletes, map[string]interface{}{pkCol: row[pkCol]})
|
|
}
|
|
if len(deletes) > 0 {
|
|
totals.Deletes += len(deletes)
|
|
if consume != nil {
|
|
if err := consume(pagedDiffPage{Deletes: deletes}); err != nil {
|
|
return true, totals, err
|
|
}
|
|
}
|
|
}
|
|
if len(targetRows) < defaultSyncReadPageSize {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
return true, totals, nil
|
|
}
|
|
|
|
func diffSourcePageByPK(pkCol string, sourceRows, targetRows []map[string]interface{}) pagedDiffPage {
|
|
targetMap := make(map[string]map[string]interface{}, len(targetRows))
|
|
for _, row := range targetRows {
|
|
pkKey, ok := pkValueKey(row[pkCol])
|
|
if !ok {
|
|
continue
|
|
}
|
|
targetMap[pkKey] = row
|
|
}
|
|
|
|
page := pagedDiffPage{
|
|
Inserts: make([]map[string]interface{}, 0),
|
|
Updates: make([]pagedUpdateDiff, 0),
|
|
}
|
|
for _, sourceRow := range sourceRows {
|
|
pkKey, ok := pkValueKey(sourceRow[pkCol])
|
|
if !ok {
|
|
continue
|
|
}
|
|
targetRow, exists := targetMap[pkKey]
|
|
if !exists {
|
|
page.Inserts = append(page.Inserts, sourceRow)
|
|
continue
|
|
}
|
|
|
|
changes := make(map[string]interface{})
|
|
changedColumns := make([]string, 0)
|
|
for key, value := range sourceRow {
|
|
if fmt.Sprintf("%v", value) == fmt.Sprintf("%v", targetRow[key]) {
|
|
continue
|
|
}
|
|
changes[key] = value
|
|
changedColumns = append(changedColumns, key)
|
|
}
|
|
if len(changes) == 0 {
|
|
page.Same++
|
|
continue
|
|
}
|
|
sort.Strings(changedColumns)
|
|
page.Updates = append(page.Updates, pagedUpdateDiff{
|
|
UpdateRow: connection.UpdateRow{
|
|
Keys: map[string]interface{}{pkCol: sourceRow[pkCol]},
|
|
Values: changes,
|
|
},
|
|
Source: sourceRow,
|
|
Target: targetRow,
|
|
ChangedColumns: changedColumns,
|
|
})
|
|
}
|
|
return page
|
|
}
|
|
|
|
func filterPagedUpdatesByPKSelection(pkCol string, updates []pagedUpdateDiff, enabled bool, selectedPKs []string) []connection.UpdateRow {
|
|
if !enabled {
|
|
return nil
|
|
}
|
|
if len(updates) == 0 {
|
|
return nil
|
|
}
|
|
out := make([]connection.UpdateRow, 0, len(updates))
|
|
for _, update := range updates {
|
|
out = append(out, update.UpdateRow)
|
|
}
|
|
return filterUpdatesByPKSelection(pkCol, out, true, selectedPKs)
|
|
}
|
|
|
|
func diffReadableColumns(sourceCols []connection.ColumnDefinition, allowedLower map[string]struct{}, pkCol string) []connection.ColumnDefinition {
|
|
out := make([]connection.ColumnDefinition, 0, len(sourceCols))
|
|
seen := map[string]struct{}{}
|
|
add := func(col connection.ColumnDefinition) {
|
|
name := strings.TrimSpace(col.Name)
|
|
lower := strings.ToLower(name)
|
|
if name == "" {
|
|
return
|
|
}
|
|
if _, ok := seen[lower]; ok {
|
|
return
|
|
}
|
|
seen[lower] = struct{}{}
|
|
out = append(out, col)
|
|
}
|
|
for _, col := range sourceCols {
|
|
name := strings.TrimSpace(col.Name)
|
|
lower := strings.ToLower(name)
|
|
if name == "" {
|
|
continue
|
|
}
|
|
if strings.EqualFold(name, pkCol) {
|
|
add(col)
|
|
continue
|
|
}
|
|
if len(allowedLower) > 0 {
|
|
if _, ok := allowedLower[lower]; !ok {
|
|
continue
|
|
}
|
|
}
|
|
add(col)
|
|
}
|
|
if _, ok := seen[strings.ToLower(strings.TrimSpace(pkCol))]; !ok && strings.TrimSpace(pkCol) != "" {
|
|
add(connection.ColumnDefinition{Name: pkCol})
|
|
}
|
|
return out
|
|
}
|
|
|
|
func diffLookupColumns(sourceReadCols, targetCols []connection.ColumnDefinition, allowedLower map[string]struct{}, pkCol string) []connection.ColumnDefinition {
|
|
targetByLower := make(map[string]connection.ColumnDefinition, len(targetCols))
|
|
for _, col := range targetCols {
|
|
name := strings.TrimSpace(col.Name)
|
|
if name != "" {
|
|
targetByLower[strings.ToLower(name)] = col
|
|
}
|
|
}
|
|
|
|
out := make([]connection.ColumnDefinition, 0, len(sourceReadCols))
|
|
seen := map[string]struct{}{}
|
|
for _, sourceCol := range sourceReadCols {
|
|
name := strings.TrimSpace(sourceCol.Name)
|
|
lower := strings.ToLower(name)
|
|
if name == "" {
|
|
continue
|
|
}
|
|
if _, ok := seen[lower]; ok {
|
|
continue
|
|
}
|
|
if !strings.EqualFold(name, pkCol) && len(allowedLower) > 0 {
|
|
if _, ok := allowedLower[lower]; !ok {
|
|
continue
|
|
}
|
|
}
|
|
if targetCol, ok := targetByLower[lower]; ok {
|
|
out = append(out, targetCol)
|
|
} else {
|
|
out = append(out, connection.ColumnDefinition{Name: name})
|
|
}
|
|
seen[lower] = struct{}{}
|
|
}
|
|
if _, ok := seen[strings.ToLower(strings.TrimSpace(pkCol))]; !ok && strings.TrimSpace(pkCol) != "" {
|
|
out = append(out, connection.ColumnDefinition{Name: pkCol})
|
|
}
|
|
return out
|
|
}
|
|
|
|
func collectPKValues(rows []map[string]interface{}, pkCol string) []interface{} {
|
|
values := make([]interface{}, 0, len(rows))
|
|
seen := make(map[string]struct{}, len(rows))
|
|
for _, row := range rows {
|
|
key, ok := pkValueKey(row[pkCol])
|
|
if !ok {
|
|
continue
|
|
}
|
|
if _, exists := seen[key]; exists {
|
|
continue
|
|
}
|
|
seen[key] = struct{}{}
|
|
values = append(values, row[pkCol])
|
|
}
|
|
return values
|
|
}
|
|
|
|
func buildPKSet(rows []map[string]interface{}, pkCol string) map[string]struct{} {
|
|
set := make(map[string]struct{}, len(rows))
|
|
for _, row := range rows {
|
|
key, ok := pkValueKey(row[pkCol])
|
|
if ok {
|
|
set[key] = struct{}{}
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
func lastValidPKValue(rows []map[string]interface{}, pkCol string) (interface{}, bool) {
|
|
for i := len(rows) - 1; i >= 0; i-- {
|
|
if _, ok := pkValueKey(rows[i][pkCol]); ok {
|
|
return rows[i][pkCol], true
|
|
}
|
|
}
|
|
return nil, false
|
|
}
|
|
|
|
func pkValueKey(value interface{}) (string, bool) {
|
|
if value == nil {
|
|
return "", false
|
|
}
|
|
key := strings.TrimSpace(fmt.Sprintf("%v", value))
|
|
if key == "" || key == "<nil>" {
|
|
return "", false
|
|
}
|
|
return key, true
|
|
}
|
|
|
|
func buildPKInSelectQuery(dbType, queryTable string, cols []connection.ColumnDefinition, pkCol string, pkValues []interface{}) string {
|
|
if len(pkValues) == 0 {
|
|
return ""
|
|
}
|
|
selectList := buildColumnSelectListForSync(dbType, cols)
|
|
if strings.TrimSpace(selectList) == "" {
|
|
selectList = "*"
|
|
}
|
|
literals := make([]string, 0, len(pkValues))
|
|
for _, value := range pkValues {
|
|
literal, ok := formatSyncSQLLiteral(value)
|
|
if !ok {
|
|
continue
|
|
}
|
|
literals = append(literals, literal)
|
|
}
|
|
if len(literals) == 0 {
|
|
return ""
|
|
}
|
|
return fmt.Sprintf("SELECT %s FROM %s WHERE %s IN (%s)",
|
|
selectList,
|
|
quoteQualifiedIdentByType(dbType, queryTable),
|
|
quoteIdentByType(dbType, pkCol),
|
|
strings.Join(literals, ", "))
|
|
}
|
|
|
|
func buildKeysetPagedTableQuery(dbType, queryTable string, cols []connection.ColumnDefinition, orderCol string, lastValue interface{}, hasLastValue bool, limit int) string {
|
|
selectList := buildColumnSelectListForSync(dbType, cols)
|
|
if strings.TrimSpace(selectList) == "" {
|
|
selectList = "*"
|
|
}
|
|
safeLimit := limit
|
|
if safeLimit <= 0 {
|
|
safeLimit = defaultSyncReadPageSize
|
|
}
|
|
where := ""
|
|
if hasLastValue {
|
|
literal, ok := formatSyncSQLLiteral(lastValue)
|
|
if !ok {
|
|
return ""
|
|
}
|
|
where = fmt.Sprintf(" WHERE %s > %s", quoteIdentByType(dbType, orderCol), literal)
|
|
}
|
|
orderBy := fmt.Sprintf(" ORDER BY %s ASC", quoteIdentByType(dbType, orderCol))
|
|
if normalizeMigrationDBType(dbType) == "sqlserver" {
|
|
return fmt.Sprintf("SELECT TOP (%d) %s FROM %s%s%s", safeLimit, selectList, quoteQualifiedIdentByType(dbType, queryTable), where, orderBy)
|
|
}
|
|
return fmt.Sprintf("SELECT %s FROM %s%s%s LIMIT %d", selectList, quoteQualifiedIdentByType(dbType, queryTable), where, orderBy, safeLimit)
|
|
}
|
|
|
|
func countTableRowsForSync(database db.Database, dbType, queryTable string) (int, bool, error) {
|
|
query := fmt.Sprintf("SELECT COUNT(*) AS __gonavi_count__ FROM %s", quoteQualifiedIdentByType(dbType, queryTable))
|
|
rows, _, err := database.Query(query)
|
|
if err != nil {
|
|
return 0, true, err
|
|
}
|
|
if len(rows) == 0 {
|
|
return 0, false, nil
|
|
}
|
|
for _, value := range rows[0] {
|
|
count, ok := intFromSyncValue(value)
|
|
if ok {
|
|
return count, true, nil
|
|
}
|
|
}
|
|
return 0, false, nil
|
|
}
|
|
|
|
func intFromSyncValue(value interface{}) (int, bool) {
|
|
if value == nil {
|
|
return 0, false
|
|
}
|
|
switch v := value.(type) {
|
|
case int:
|
|
return v, true
|
|
case int8:
|
|
return int(v), true
|
|
case int16:
|
|
return int(v), true
|
|
case int32:
|
|
return int(v), true
|
|
case int64:
|
|
return int(v), true
|
|
case uint:
|
|
return int(v), true
|
|
case uint8:
|
|
return int(v), true
|
|
case uint16:
|
|
return int(v), true
|
|
case uint32:
|
|
return int(v), true
|
|
case uint64:
|
|
return int(v), true
|
|
case float32:
|
|
return int(v), true
|
|
case float64:
|
|
return int(v), true
|
|
case []byte:
|
|
i, err := strconv.Atoi(strings.TrimSpace(string(v)))
|
|
return i, err == nil
|
|
case string:
|
|
i, err := strconv.Atoi(strings.TrimSpace(v))
|
|
return i, err == nil
|
|
default:
|
|
rv := reflect.ValueOf(value)
|
|
switch rv.Kind() {
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
i := rv.Int()
|
|
if i > int64(^uint(0)>>1) {
|
|
return 0, false
|
|
}
|
|
return int(i), true
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
|
u := rv.Uint()
|
|
if u > uint64(^uint(0)>>1) {
|
|
return 0, false
|
|
}
|
|
return int(u), true
|
|
}
|
|
}
|
|
return 0, false
|
|
}
|
|
|
|
func buildColumnSelectListForSync(dbType string, cols []connection.ColumnDefinition) string {
|
|
quoted := make([]string, 0, len(cols))
|
|
seen := map[string]struct{}{}
|
|
for _, col := range cols {
|
|
name := strings.TrimSpace(col.Name)
|
|
lower := strings.ToLower(name)
|
|
if name == "" {
|
|
continue
|
|
}
|
|
if _, ok := seen[lower]; ok {
|
|
continue
|
|
}
|
|
seen[lower] = struct{}{}
|
|
quoted = append(quoted, quoteIdentByType(dbType, name))
|
|
}
|
|
return strings.Join(quoted, ", ")
|
|
}
|
|
|
|
func formatSyncSQLLiteral(value interface{}) (string, bool) {
|
|
if value == nil {
|
|
return "", false
|
|
}
|
|
switch v := value.(type) {
|
|
case time.Time:
|
|
return quoteSyncSQLString(v.Format("2006-01-02 15:04:05.999999999")), true
|
|
case []byte:
|
|
return quoteSyncSQLString(string(v)), true
|
|
case string:
|
|
if strings.TrimSpace(v) == "" {
|
|
return "", false
|
|
}
|
|
return quoteSyncSQLString(v), true
|
|
case bool:
|
|
if v {
|
|
return "1", true
|
|
}
|
|
return "0", true
|
|
default:
|
|
text := strings.TrimSpace(fmt.Sprintf("%v", value))
|
|
if text == "" || text == "<nil>" {
|
|
return "", false
|
|
}
|
|
switch value.(type) {
|
|
case int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, float32, float64:
|
|
return text, true
|
|
default:
|
|
return quoteSyncSQLString(text), true
|
|
}
|
|
}
|
|
}
|
|
|
|
func quoteSyncSQLString(value string) string {
|
|
return "'" + strings.ReplaceAll(value, "'", "''") + "'"
|
|
}
|
|
|
|
func supportsPagedDiffSelect(dbType string) bool {
|
|
return supportsDirectImportPagination(dbType)
|
|
}
|
|
|
|
func supportsPagedDiffPKLookup(dbType string) bool {
|
|
return supportsDirectImportPagination(dbType)
|
|
}
|
|
|
|
func supportsPagedDiffKeysetSelect(dbType string) bool {
|
|
return supportsDirectImportPagination(dbType)
|
|
}
|