@indexName附近的语法不正确。短小精悍的

问题描述 投票:2回答:1

这似乎是一个奇怪的行为,但以下代码抛出一个错误:

public async Task RebuildIndex(string tableName, string indexName)
{
    await _dbConnection.ExecuteAsync($@"
        alter index @indexName on @tableName rebuild;", 
    new
    {
        indexName = indexName,
        tableName = tableName
    });
}

错误信息:

System.Data.SqlClient.SqlException (0x80131904): Неправильный синтаксис около конструкции "@indexName".
   at System.Data.SqlClient.SqlConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
   at System.Data.SqlClient.SqlInternalConnection.OnError(SqlException exception, Boolean breakConnection, Action`1 wrapCloseInAction)
   at System.Data.SqlClient.TdsParser.ThrowExceptionAndWarning(TdsParserStateObject stateObj, Boolean callerHasConnectionLock, Boolean asyncClose)
   at System.Data.SqlClient.TdsParser.TryRun(RunBehavior runBehavior, SqlCommand cmdHandler, SqlDataReader dataStream, BulkCopySimpleResultSet bulkCopyHandler, TdsParserStateObject stateObj, Boolean& dataReady)
   at System.Data.SqlClient.SqlCommand.FinishExecuteReader(SqlDataReader ds, RunBehavior runBehavior, String resetOptionsString)
   at System.Data.SqlClient.SqlCommand.CompleteAsyncExecuteReader()
   at System.Data.SqlClient.SqlCommand.EndExecuteNonQueryInternal(IAsyncResult asyncResult)
   at System.Data.SqlClient.SqlCommand.EndExecuteNonQuery(IAsyncResult asyncResult)
   at System.Threading.Tasks.TaskFactory`1.FromAsyncCoreLogic(IAsyncResult iar, Func`2 endFunction, Action`1 endAction, Task`1 promise, Boolean requiresSynchronization)
--- End of stack trace from previous location where exception was thrown ---
   at Dapper.SqlMapper.ExecuteImplAsync(IDbConnection cnn, CommandDefinition command, Object param) in C:\projects\dapper\Dapper\SqlMapper.Async.cs:line 678
   at Railways.DbManager.Repositories.DbmIndexDefragmentationRepository.RebuildIndex(String tableName, String indexName) in C:\Railways\asu-itk-core\Railways\Railways.DbManager.Repositories\DbmIndexDefragmentationRepository.cs:line 48
   at Railways.DBManager.Services.IndexDefragmentationService.DefragIndexes() in C:\Railways\asu-itk-core\Railways\Railways.DBManager.Services\IndexDefragmentationService.cs:line 41
ClientConnectionId:7db4e7f6-e01d-4671-9fe3-e22dcb388cd4
Error Number:102,State:1,Class:15

这可能有什么不对?

我只是传递参数,这应该工作。

也许我错过了什么?有人可以建议一个解决方案?

c# sql-server dapper
1个回答
3
投票

您将无法参数化SQL语句中使用的目标对象名称,即无法参数化表名和索引名。这不仅仅是一个Dapper限制 - 例如ADO.Net和sp_executesql同样不能参数化表。 (推测性地,参数化不仅可以防止SQL注入攻击,还可以根据确切的数据类型改进执行计划缓存 - 如果目标表是动态的,则计划缓存显然是不可能的)。

在您的情况下,您将需要进行自己的SQL注入检查(如果需要 - 索引名称和表名来自不受信任的用户是非常罕见的),然后只需将名称替换为SQL语句,例如:通过string interpolationString.Format

如果您真的接受来自不受信任的用户的表或索引名称并且需要防范SQL注入攻击,则可以使用Aaron Betrand's advice并在执行SQL语句之前首先检查sys.tables中表的存在以及sys.indexes中的索引。

编辑

您可以在systemsys.tables之类的DMVs表的查询中参数化表名的原因是因为这里,表或其他对象的名称是列中的值。但同样,您将无法参数化系统表或DMV本身的名称。

© www.soinside.com 2019 - 2024. All rights reserved.