@@ -170,22 +170,7 @@ else if ($step==2) { | |||||
$query .= $line."\n"; | $query .= $line."\n"; | ||||
$query = str_replace('#@__', $dbprefix, $query); | $query = str_replace('#@__', $dbprefix, $query); | ||||
if ($dbtype == 'sqlite') { | if ($dbtype == 'sqlite') { | ||||
$query = preg_replace('/character set (.*?) /i','', $query); | |||||
$query = preg_replace('/unsigned/i','', $query); | |||||
$query = str_replace('TYPE=MyISAM','', $query); | |||||
$query = preg_replace ('/TINYINT\(([\d]+)\)/i','INTEGER', $query); | |||||
$query = preg_replace ('/mediumint\(([\d]+)\)/i','INTEGER', $query); | |||||
$query = preg_replace ('/smallint\(([\d]+)\)/i','INTEGER', $query); | |||||
$query = preg_replace('/int\(([\d]+)\)/i','INTEGER', $query); | |||||
$query = preg_replace('/auto_increment/i','PRIMARY KEY AUTOINCREMENT', $query); | |||||
$query = preg_replace('/,([\t\s ]+)KEY(.*?)MyISAM;/','', $query); | |||||
$query = preg_replace('/,([\t\s ]+)KEY(.*?);/',');', $query); | |||||
$query = preg_replace('/,([\t\s ]+)UNIQUE KEY(.*?);/',');', $query); | |||||
$query = preg_replace('/set\(([^\)]*?)\)/','varchar', $query); | |||||
$query = preg_replace('/enum\(([^\)]*?)\)/','varchar', $query); | |||||
if (preg_match("/PRIMARY KEY AUTOINCREMENT/", $query)) { | |||||
$query = preg_replace('/,([\t\s ]+)PRIMARY KEY([\t\s ]+)\(`([0-9a-zA-Z]+)`\)/i','', $query); | |||||
} | |||||
$query = ConvertMysqlToSqlite($query); | |||||
$db->exec($query); | $db->exec($query); | ||||
} else { | } else { | ||||
if (preg_match('#CREATE#i', $query)) { | if (preg_match('#CREATE#i', $query)) { | ||||
@@ -113,8 +113,10 @@ | |||||
$("#dbtype").change(function() { | $("#dbtype").change(function() { | ||||
if ($(this).val() === 'sqlite') { | if ($(this).val() === 'sqlite') { | ||||
$(".form-group.server").hide(); | $(".form-group.server").hide(); | ||||
$("#dbpwd").prop("required", false); | |||||
} else { | } else { | ||||
$(".form-group.server").show(); | $(".form-group.server").show(); | ||||
$("#dbpwd").prop("required", true); | |||||
} | } | ||||
}); | }); | ||||
</script> | </script> | ||||
@@ -365,14 +365,24 @@ function ShowMsg($msg, $gourl, $onlymsg = 0, $limittime = 0) | |||||
*/ | */ | ||||
function TableHasField($tablename, $field) | function TableHasField($tablename, $field) | ||||
{ | { | ||||
global $dsql; | |||||
$dsql->GetTableFields($tablename,"tfd"); | |||||
while ($r = $dsql->GetFieldObject("tfd")) { | |||||
if ($r->name === $field) { | |||||
return true; | |||||
global $dsql,$cfg_dbtype; | |||||
if ($cfg_dbtype === 'mysql') { | |||||
$dsql->GetTableFields($tablename,"tfd"); | |||||
while ($r = $dsql->GetFieldObject("tfd")) { | |||||
if ($r->name === $field) { | |||||
return true; | |||||
} | |||||
} | |||||
return false; | |||||
} elseif ($cfg_dbtype === 'sqlite') { | |||||
$k = $dsql->GetTableFields($tablename); | |||||
while ($r = $dsql->GetFieldObject($k)) { | |||||
if ($r->name === $field) { | |||||
return true; | |||||
} | |||||
} | } | ||||
return false; | |||||
} | } | ||||
return false; | |||||
} | } | ||||
function GetSimpleServerSoftware() | function GetSimpleServerSoftware() | ||||
{ | { | ||||
@@ -459,7 +469,7 @@ function GetUpdateSQL() | |||||
$result = array(); | $result = array(); | ||||
$query = ''; | $query = ''; | ||||
$sql4tmp = "ENGINE=MyISAM DEFAULT CHARSET=".$cfg_db_language; | $sql4tmp = "ENGINE=MyISAM DEFAULT CHARSET=".$cfg_db_language; | ||||
$fp = fopen(DEDEROOT.'/install/update.txt','r'); | |||||
$fp = fopen(DEDEDATA.'/admin/update.txt','r'); | |||||
$sqls = array(); | $sqls = array(); | ||||
$current_ver = ''; | $current_ver = ''; | ||||
while(!feof($fp)) | while(!feof($fp)) | ||||
@@ -692,6 +702,108 @@ function DedeSearchAPIURL($action, $parms=array()) | |||||
return $finalUrl; | return $finalUrl; | ||||
} | } | ||||
function ConvertMysqlToSqlite($mysqlQuery) { | |||||
// 移除 CHARACTER SET 和 COLLATE | |||||
$query = preg_replace('/character set \S+/i', '', $mysqlQuery); | |||||
$query = preg_replace('/collate \S+/i', '', $query); | |||||
// 移除 unsigned 关键字 | |||||
$query = str_replace('unsigned', '', $query); | |||||
// 替换 MySQL 的整数类型为 SQLite 的 INTEGER | |||||
$query = preg_replace('/\b(TINY|SMALL|MEDIUM|BIG)?INT\(\d+\)/i', 'INTEGER', $query); | |||||
// 替换 AUTO_INCREMENT 为 PRIMARY KEY AUTOINCREMENT (仅适用于 INTEGER 类型) | |||||
$query = preg_replace('/\bINTEGER\s+NOT NULL\s+PRIMARY KEY AUTOINCREMENT/i', 'INTEGER PRIMARY KEY AUTOINCREMENT', $query); | |||||
$query = preg_replace('/\bAUTO_INCREMENT\b/i', '', $query); | |||||
// 移除 MySQL 特有的 ENGINE、CHARSET、COLLATE、TYPE 选项 | |||||
$query = preg_replace('/ENGINE=\S+/i', '', $query); | |||||
$query = preg_replace('/DEFAULT CHARSET=\S+/i', '', $query); | |||||
$query = preg_replace('/COLLATE=\S+/i', '', $query); | |||||
$query = preg_replace('/TYPE=MyISAM;/i', '', $query); | |||||
// 移除 COMMENT 语法(SQLite 不支持) | |||||
$query = preg_replace('/COMMENT\s+\'[^\']*\'/i', '', $query); | |||||
// 移除 KEY 和 UNIQUE KEY 定义(SQLite 会自动管理索引),同时处理 USING BTREE | |||||
$query = preg_replace('/,?\s*KEY\s+\S+\s*\([^)]*\)\s*(USING BTREE)?/i', '', $query); | |||||
$query = preg_replace('/,?\s*UNIQUE KEY\s+\S+\s*\([^)]*\)\s*(USING BTREE)?/i', '', $query); | |||||
// 移除不完整的 UNIQUE 定义 | |||||
$query = preg_replace('/,?\s*UNIQUE\s*(?!\()/', '', $query); | |||||
// 替换 ENUM 和 SET 为 TEXT | |||||
$query = preg_replace('/\b(ENUM|SET)\([^)]*\)/i', 'TEXT', $query); | |||||
// 替换 MEDIUMTEXT 为 TEXT | |||||
$query = preg_replace('/\bMEDIUMTEXT\b/i', 'TEXT', $query); | |||||
// 处理 DEFAULT CURRENT_TIMESTAMP | |||||
$query = preg_replace('/DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP/i', 'DEFAULT CURRENT_TIMESTAMP', $query); | |||||
// 处理 DEFAULT 值 | |||||
$query = preg_replace('/DEFAULT\s+\'([^\']+)\'/i', 'DEFAULT "$1"', $query); | |||||
// 处理 PRIMARY KEY 只能用于 INTEGER | |||||
if (preg_match('/PRIMARY KEY \(`(\w+)`\)/', $query, $matches)) { | |||||
$primaryKeyColumn = $matches[1]; | |||||
$query = preg_replace('/,?\s*PRIMARY KEY\s*\(`' . $primaryKeyColumn . '`\)/i', '', $query); | |||||
$query = preg_replace('/(`' . $primaryKeyColumn . '`\s+INTEGER)/i', '$1 PRIMARY KEY', $query); | |||||
} | |||||
// 处理 CONCAT 替换为 SQLite 兼容形式 | |||||
if (preg_match('/CONCAT\(([^)]*?)\)/i', $query, $matches)) { | |||||
$query = preg_replace('/CONCAT\(([^)]*?)\)/i', str_replace(",", "||", $matches[1]), $query); | |||||
$query = str_replace("'||'", "','", $query); | |||||
} | |||||
// 修正 FIND_IN_SET 替换 | |||||
$query = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)>0/i", "(',' || arc.flag || ',') LIKE '%,\\1,%'", $query); | |||||
$query = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)<1/i", "(',' || arc.flag || ',') NOT LIKE '%,\\1,%'", $query); | |||||
// 修正 FIND_IN_SET 替换(允许列名包含点号) | |||||
$query = preg_replace_callback( | |||||
"/FIND_IN_SET\s*\(\s*'([^']+)'\s*,\s*([a-zA-Z0-9_`\.]+)\s*\)/i", | |||||
function ($matches) { | |||||
// 返回 SQLite 兼容的 LIKE 语法 | |||||
return "(',' || " . $matches[2] . " || ',' LIKE '%," . $matches[1] . ",%')"; | |||||
}, | |||||
$query | |||||
); | |||||
// 替换 FIELD 函数为 CASE 表达式 | |||||
$query = preg_replace_callback( | |||||
'/\bFIELD\s*\(\s*([^,]+)\s*,\s*((?:\'[^\']+\'|`[^`]+`|[^),]+(?:,\s*)?)+)\s*\)/i', | |||||
function ($matches) { | |||||
$field = trim($matches[1]); | |||||
$values = trim($matches[2]); | |||||
// 更精确分割值列表(支持带引号、反引号及无空格分隔的数值) | |||||
preg_match_all('/\'[^\']+\'|`[^`]+`|\d+|\w+/', $values, $valueParts); | |||||
$cases = []; | |||||
$position = 1; | |||||
foreach ($valueParts[0] as $value) { | |||||
$cases[] = "WHEN $field = $value THEN $position"; | |||||
$position++; | |||||
} | |||||
return "(CASE " . implode(' ', $cases) . " ELSE 0 END)"; | |||||
}, | |||||
$query | |||||
); | |||||
// 新增的转换逻辑 | |||||
$query = preg_replace("/SHOW fields FROM `([\w]+)`/i", "PRAGMA table_info('\\1') ", $query); | |||||
$query = preg_replace("/SHOW CREATE TABLE `([\w]+)`/i", "SELECT 0,sql FROM sqlite_master WHERE name='\\1'; ", $query); | |||||
$query = preg_replace("/Show Tables/i", "SELECT name FROM sqlite_master WHERE type = \"table\"", $query); | |||||
$query = str_replace("\'", "\"", $query); | |||||
$query = str_replace('\t\n', "", $query); | |||||
$query = str_ireplace('rand', 'RANDOM', $query); | |||||
return trim($query); | |||||
} | |||||
//自定义函数接口 | //自定义函数接口 | ||||
if (file_exists(DEDEINC.'/extend.func.php')) { | if (file_exists(DEDEINC.'/extend.func.php')) { | ||||
require_once(DEDEINC.'/extend.func.php'); | require_once(DEDEINC.'/extend.func.php'); | ||||
@@ -56,7 +56,7 @@ class DedeSqlite | |||||
var $isInit = false; | var $isInit = false; | ||||
var $pconnect = false; | var $pconnect = false; | ||||
var $_fixObject; | var $_fixObject; | ||||
var $_fieldIdx = 1; //这里最好是数组,对应id,但由于用的地方不多,暂时先这样处理 | |||||
var $_fieldIdx = array(); //这里最好是数组,对应id,但由于用的地方不多,暂时先这样处理 | |||||
//用外部定义的变量初始类,并连接数据库 | //用外部定义的变量初始类,并连接数据库 | ||||
function __construct($pconnect = FALSE, $nconnect = FALSE) | function __construct($pconnect = FALSE, $nconnect = FALSE) | ||||
{ | { | ||||
@@ -253,7 +253,7 @@ class DedeSqlite | |||||
CheckSql($this->queryString); | CheckSql($this->queryString); | ||||
} | } | ||||
$t1 = ExecTime(); | $t1 = ExecTime(); | ||||
//var_dump($this->queryString); | |||||
// var_dump($this->queryString); | |||||
$this->result[$id] = $this->linkID->query($this->queryString); | $this->result[$id] = $this->linkID->query($this->queryString); | ||||
if (!$this->result[$id]) { | if (!$this->result[$id]) { | ||||
$this->DisplayError("执行SQL错误:{$this->linkID->lastErrorMsg()}"); | $this->DisplayError("执行SQL错误:{$this->linkID->lastErrorMsg()}"); | ||||
@@ -386,7 +386,7 @@ class DedeSqlite | |||||
return $sqlite_version; | return $sqlite_version; | ||||
} | } | ||||
//获取特定表的信息 | //获取特定表的信息 | ||||
function GetTableFields($tbname, $id = "me") | |||||
function GetTableFields($tbname) | |||||
{ | { | ||||
global $dsqlite; | global $dsqlite; | ||||
if (!$dsqlite->isInit) { | if (!$dsqlite->isInit) { | ||||
@@ -395,29 +395,27 @@ class DedeSqlite | |||||
$prefix = "#@__"; | $prefix = "#@__"; | ||||
$tbname = str_replace($prefix, $GLOBALS['cfg_dbprefix'], $tbname); | $tbname = str_replace($prefix, $GLOBALS['cfg_dbprefix'], $tbname); | ||||
$query = "SELECT * FROM {$tbname} LIMIT 1"; | $query = "SELECT * FROM {$tbname} LIMIT 1"; | ||||
$this->result[$id] = $this->linkID->query($query); | |||||
$result = $this->linkID->query($query); | |||||
$key = spl_object_hash($result); | |||||
$this->result[$key] = $result; | |||||
$this->_fieldIdx[$key] = 0; | |||||
return $key ; | |||||
} | } | ||||
//获取字段详细信息 | //获取字段详细信息 | ||||
function GetFieldObject($id = "me") | function GetFieldObject($id = "me") | ||||
{ | { | ||||
if (!$this->result[$id]) { | |||||
if (!isset($this->result[$id]) || !$this->result[$id]) { | |||||
return false; | return false; | ||||
} | } | ||||
$cols = $this->result[$id]->numColumns(); | $cols = $this->result[$id]->numColumns(); | ||||
if ($this->_fieldIdx >= $cols) { | |||||
$this->_fieldIdx = 1; | |||||
if ($this->_fieldIdx[$id] >= $cols) { | |||||
return false; | return false; | ||||
} | } | ||||
for ($i = 1; $i <= $cols; $i++) { | |||||
$field = new stdClass; | |||||
$n = $this->result[$id]->columnName($i); | |||||
$field->name = $n; | |||||
if ($this->_fieldIdx === $i) { | |||||
$this->_fieldIdx++; | |||||
return $field; | |||||
} | |||||
} | |||||
return false; | |||||
$n = $this->result[$id]->columnName($this->_fieldIdx[$id]); | |||||
$field = new stdClass(); | |||||
$field->name = $n; | |||||
$this->_fieldIdx[$id]++; | |||||
return $field; | |||||
} | } | ||||
//获得查询的总记录数 | //获得查询的总记录数 | ||||
function GetTotalRow($id = "me") | function GetTotalRow($id = "me") | ||||
@@ -460,41 +458,7 @@ class DedeSqlite | |||||
{ | { | ||||
$prefix = "#@__"; | $prefix = "#@__"; | ||||
$sql = str_replace($prefix, $GLOBALS['cfg_dbprefix'], $sql); | $sql = str_replace($prefix, $GLOBALS['cfg_dbprefix'], $sql); | ||||
$this->queryString = $sql; | |||||
//$this->queryString = preg_replace("/CONCAT\(',', arc.typeid2, ','\)/i","printf(',%s,', arc.typeid2)",$this->queryString); | |||||
if (preg_match("/CONCAT\(([^\)]*?)\)/i", $this->queryString, $matches)) { | |||||
$this->queryString = preg_replace("/CONCAT\(([^\)]*?)\)/i", str_replace(",", "||", $matches[1]), $this->queryString); | |||||
$this->queryString = str_replace("'||'", "','", $this->queryString); | |||||
} | |||||
$this->queryString = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)>0/i", "(',' || arc.flag || ',') LIKE '%,\\1,%'", $this->queryString); | |||||
$this->queryString = preg_replace("/FIND_IN_SET\('([\w]+)', arc.flag\)<1/i", "(',' || arc.flag || ',') NOT LIKE '%,\\1,%'", $this->queryString); | |||||
if (preg_match("/CREATE TABLE/i", $this->queryString)) { | |||||
$this->queryString = preg_replace("/[\r\n]/", '', $this->queryString); | |||||
$this->queryString = preg_replace('/character set (.*?) /i', '', $this->queryString); | |||||
$this->queryString = preg_replace('/unsigned/i', '', $this->queryString); | |||||
$this->queryString = str_replace('TYPE=MyISAM', '', $this->queryString); | |||||
$this->queryString = preg_replace('/TINYINT\(([\d]+)\)/i', 'INTEGER', $this->queryString); | |||||
$this->queryString = preg_replace('/mediumint\(([\d]+)\)/i', 'INTEGER', $this->queryString); | |||||
$this->queryString = preg_replace('/smallint\(([\d]+)\)/i', 'INTEGER', $this->queryString); | |||||
$this->queryString = preg_replace('/int\(([\d]+)\)/i', 'INTEGER', $this->queryString); | |||||
$this->queryString = preg_replace('/auto_increment/i', 'PRIMARY KEY AUTOINCREMENT', $this->queryString); | |||||
$this->queryString = preg_replace('/, KEY(.*?)MyISAM;/i', '', $this->queryString); | |||||
$this->queryString = preg_replace('/, KEY(.*?);/i', ');', $this->queryString); | |||||
$this->queryString = preg_replace('/, UNIQUE KEY(.*?);/i', ');', $this->queryString); | |||||
$this->queryString = preg_replace('/set\(([^\)]*?)\)/', 'varchar', $this->queryString); | |||||
$this->queryString = preg_replace('/enum\(([^\)]*?)\)/', 'varchar', $this->queryString); | |||||
if (preg_match("/PRIMARY KEY AUTOINCREMENT/", $this->queryString)) { | |||||
$this->queryString = preg_replace('/,([\t\s ]+)PRIMARY KEY \(`([0-9a-zA-Z]+)`\)/i', '', $this->queryString); | |||||
$this->queryString = str_replace(', PRIMARY KEY (`id`)', '', $this->queryString); | |||||
} | |||||
} | |||||
$this->queryString = preg_replace("/SHOW fields FROM `([\w]+)`/i", "PRAGMA table_info('\\1') ", $this->queryString); | |||||
$this->queryString = preg_replace("/SHOW CREATE TABLE .([\w]+)/i", "SELECT 0,sql FROM sqlite_master WHERE name='\\1'; ", $this->queryString); | |||||
//var_dump($this->queryString); | |||||
$this->queryString = preg_replace("/Show Tables/i", "SELECT name FROM sqlite_master WHERE type = \"table\"", $this->queryString); | |||||
$this->queryString = str_replace("\'", "\"", $this->queryString); | |||||
$this->queryString = str_replace('\t\n', "", $this->queryString); | |||||
$this->queryString = str_ireplace('rand', 'RANDOM', $this->queryString); | |||||
$this->queryString = ConvertMysqlToSqlite($sql); | |||||
//var_dump($this->queryString); | //var_dump($this->queryString); | ||||
} | } | ||||
function SetSql($sql) | function SetSql($sql) | ||||