Maneira alternativa de obter o parâmetro de saída do procedimento armazenado
Adoro usar o Dapper para minhas necessidades de ORM, mas sei que deve haver uma maneira melhor de inserir / atualizar meu banco de dados do servidor sql usando um procedimento armazenado e Listas fortemente tipadas.
Por exemplo:
Eu tenho uma música da classe:
public class Song
{
public int Id { get; set; }
public string title { get; set; }
public string genre { get; set; }
}
e alguém envia uma lista de músicas:
List<Song> songs = new List<Song> {
new Song { Id = 1, title = "Song 1" , genre="rock"},
new Song { Id = 2, title = "Song 2" , genre="disco"}};
Quero atualizar o banco de dados usando meu procedimento armazenado, que insere a nova música ou a atualiza, se a música já existir. Meu procedimento armazenado possui dois parâmetros de saída:@success_added int = 0 and @success_updated int = 0
meu sproc é o seguinte:
ALTER PROCEDURE [dbo].[UpdateSong]
@Id int = 0,
@title varchar(25) = NULL,
@genre varchar(25) = NULL,
@success_updated bit = 0 OUTPUT,
@success_added bit = 0 OUTPUT
AS
IF NOT EXISTS (SELECT Id FROM Songs WHERE Id = @Id)
BEGIN
INSERT INTO Songs
(
-- Id created by db
title,
genre
)
VALUES
(
@title,
@genre
)
SELECT @Success_Added = 1, @Success_Updated = 0
END
ELSE -- song already exists
BEGIN
UPDATE Songs
SET
title = @title,
@genre = @genre
WHERE Id = @Id
SELECT @Success_Added = 0, @Success_Updated = 1
END
RETURN
Eu sei que isso funciona:
dbConn.Open();
DynamicParameters p = new DynamicParameters();
foreach (var song in songs)
{
p.Add("@Id", song.Id);
p.Add("@title", song.title);
p.Add("@genre", song.genre);
p.Add("@success_updated", dbType: DbType.Boolean, direction: ParameterDirection.Output);
p.Add("@success_added", dbType: DbType.Boolean, direction: ParameterDirection.Output);
dbConn.Execute("Test_UpdateSong", p, commandType: CommandType.StoredProcedure);
Console.WriteLine("@success_added: " + p.Get<Boolean>("@success_added"));
Console.WriteLine("@success_updated: " + p.Get<Boolean>("@success_updated"));
}
dbConn.Close();
Mas isso requer a conversão manual de cada propriedade Song para um tipo anônimo DynamicParameter. Prefiro simplesmente fazer isso:
dbConn.Open();
foreach (var song in songs)
{
var updateResult = dbConn.Query<dynamic>("Test_UpdateSong", song, commandType: CommandType.StoredProcedure);
}
dbConn.Close();
O que também funciona. Mas agora como obtenho meus parâmetros de saída?