Subido por oscar piñones

CARGAR Y DESCARGAR MEDIA DESDE SQL SERVER MVC

Anuncio
2019
FILESTREAM: Descargar y
cargar documentos desde
SQL Server a través de
ASP.Net MVC
ISGCAS: ADMINISTRACIÓN DOCUMENTAL EN SQL SERVER
OSCAR PIÑONES SAAVEDRA
OPENDAT S.A. | Uribe 636, Oficina 601
Descargar y cargar medios de SQL Server a través de ASP.Net MVC
Una pregunta frecuente que aparece en los foros de discusión es cómo servir un medio que se almacena en la tabla
de SQL Server desde una aplicación ASP.Net. Desafortunadamente, la respuesta casi siempre es errónea, ya que la
solución predominante consiste en copiar el medio completo. Esta solución funciona bien cuando se prueba con
una carga ligera y devuelve pocos medios pequeños. Pero en el entorno de producción, la memoria requerida por
todos los archivos almacenados como matrices de bytes en la memoria causa una grave degradación del
rendimiento. Una buena solución debe utilizar la semántica de transmisión, transfiriendo los datos en pequeños
fragmentos del servidor SQL al resultado devuelto de HTTP.
Los componentes de SqlClient ofrecen semánticas de transmisión para grandes conjuntos de resultados, incluidos
grandes campos BLOB, pero el cliente tiene que solicitarlo. El 'ingrediente secreto' es el paso en el indicador
CommandBehavior.SequentialAccess hacia SqlCommand.ExecuteReader:
Proporciona una forma para que el DataReader maneje las filas que contienen columnas con
grandes valores binarios. En lugar de cargar toda la fila, SequentialAccess permite al DataReader
cargar datos como una secuencia. Luego puede usar el método GetBytes o GetChars para
especificar una ubicación de byte para iniciar la operación de lectura y un tamaño de búfer
limitado para los datos que se devuelven.
Una carpeta ASP.Net Virtual MVC Media respaldada por SQL Server
Digamos que queremos tener una carpeta de medios virtual en un sitio MVC de ASP.Net, que sirva los archivos de
una base de datos de SQL Server. Una solicitud GET para una URL como "http: //site/Media/IMG0042.JPG" debe
devolver el contenido del archivo llamado IMG0042.JPG de la base de datos. Una solicitud POST a la URL "http: //
site / Media" que contiene un archivo incrustado debe insertar este nuevo archivo en la base de datos y redirigir la
respuesta a la ruta virtual del archivo recién agregado. Así es como subimos el formulario HTML:
<form method="post" action="/Media" enctype="multipart/form-data">
<input type="file" name="file" id="file"/>
<input type="submit" name="Submit" value="Submit"/>
</form>
routes.MapRoute(
"Media",
"Media/{filename}",
new { controller = "Media", action = "GetFile" },
new { filename = @"[^/?*:;{}\\]+" });
26-5-2019
Comenzaremos agregando una ruta especial en Global.asax.cs que actuará como una carpeta virtual para las
solicitudes de descarga de archivos:
1
Tenga en cuenta que la carga será manejada por la ruta MVC predeterminada si agregamos un método Index() a
nuestro controlador que manejará el POST.
Para nuestras solicitudes GET necesitamos una clase FileDownloadModel para representar las propiedades del
archivo solicitado. Para este ejemplo, no necesitamos un modelo POST ya que solo tendremos un único campo de
entrada, el archivo cargado. Vamos a utilizar una interfaz de repositorio que declara dos métodos: GetFileByName
devuelve desde el repositorio un FileDownloadModel dado un nombre de archivo, y PutFile aceptará un archivo
cargado y lo colocará en el repositorio.
public class FileDownloadModel
{
public
public
public
public
public
string FileName {get; internal set;}
string ContentType {get; internal set;}
string ContentCoding {get; internal set;}
long ContentLength {get; internal set;}
Stream Content {get; internal set;}
}
public interface IMediaRepository
{
bool GetFileByName(
string fileName,
out FileDownloadModel file);
void PostFile(HttpPostedFileBase file, out string fileName);
}
Con esta interfaz de Repository podemos codificar nuestra clase MediaController:
public class MediaController: Controller
{
public IMediaRepository Repository {get; set;}
public MediaController()
{
Repository = new SqlMediaRepository();
}
[HttpGet]
public ActionResult GetFile(string fileName)
{
FileDownloadModel model;
if (false == Repository.GetFileByName(
fileName,
out model))
{
return new HttpNotFoundResult
{
StatusDescription = String.Format(
"File {0} not found",
fileName)
26-5-2019
[HttpPost]
public ActionResult Index()
{
string fileName;
Repository.PostFile(Request.Files[0], out fileName);
return new RedirectResult("/Media/" + fileName);
}
2
};
}
if ( null != model.ContentCoding)
{
Response.AddHeader(
"Content-Encoding",
model.ContentCoding);
}
Response.AddHeader(
"Content-Length",
model.ContentLength.ToString ());
Response.BufferOutput = false;
return new FileStreamResult(
model.Content,
model.ContentType);
}
}
Hemos codificado la implementación del Repositorio en SqlMediaRepository, una clase que crearemos en breve.
Un proyecto real probablemente usaría los patrones de inyección de dependencia o inversión de control, tal vez
usando Castle Windsor, por ejemplo. Para mayor brevedad, omitiremos estos detalles, hay muchos blogs y artículos
que describen cómo hacerlo.
Tenga en cuenta el uso de la devolución FileStreamResult, que es una acción suministrada por MVC para devolver
una descarga de un objeto Stream arbitrario. Lo que también nos lleva al siguiente punto, necesitamos
implementar un flujo que lea el contenido de un SqlDataReader.
Una secuencia basada en SqlDataReader
Ahora necesitamos una implementación de la clase Stream abstracta que pueda transmitir una columna BLOB
desde un SqlDataReader. Espera, dices, ¿SqlBytes ya tiene una propiedad Stream que lee un BLOB de un resultado
como un Stream? Desafortunadamente, este pequeño comentario hace que esta clase sea inútil para nuestros
propósitos:
Obtener o configurar la propiedad Stream carga todos los datos en la memoria. Su uso con datos
de gran tamaño puede provocar una excepción OutOfMemoryException.
public class SqlReaderStream: Stream
{
private SqlDataReader reader;
private int columnIndex;
26-5-2019
Así que nos quedamos con la implementación de un Stream basado en un campo BLOB SqlDataReader, un Stream
que devuelve el contenido del BLOB utilizando las llamadas GetBytes adecuadas y no carga todo el BLOB en la
memoria. Afortunadamente, esto es bastante simple ya que solo necesitamos implementar un puñado de
métodos:
3
private long position;
public SqlReaderStream(
SqlDataReader reader,
int columnIndex)
{
this.reader = reader;
this.columnIndex = columnIndex;
}
public override long Position
{
get {return position;}
set {throw new NotImplementedException();}
}
public override int Read(byte[] buffer, int offset, int count)
{
long bytesRead = reader.GetBytes(columnIndex, position, buffer, offset, count);
position += bytesRead;
return (int)bytesRead;
}
public override bool CanRead
{
get {return true;}
}
public override bool CanSeek
{
get {return false;}
}
public override bool CanWrite
{
get {return false;}
}
public override void Flush()
{
throw new NotImplementedException();
}
public override long Length
{
get {throw new NotImplementedException();}
}
public override long Seek(long offset, SeekOrigin origin)
{
throw new NotImplementedException();
}
public override void SetLength(long value)
{
throw new NotImplementedException();
}
protected override void Dispose(bool disposing)
{
if (disposing && null != reader)
{
reader.Dispose();
reader = null;
}
base.Dispose(disposing);
}
26-5-2019
public override void Write(byte[] buffer, int offset, int count)
{
throw new NotImplementedException();
}
4
}
Como puede ver, solo necesitamos devolver las respuestas correctas a CanRead (sí), CanWrite (no) y CanSeek
(también no), realizar un seguimiento de nuestra posición actual y necesitamos implementar la lectura para
obtener más bytes del lector, usando GetReader.
También estamos anulando el método Dispose (bool disposing). Esto se debe a que tendremos que cerrar el
SqlDataReader cuando se complete la transferencia de contenido. Si es la primera vez que ve esta firma del método
de Disposición, entonces debe leer Implementación de un método de Disposición
Streaming de carga de datos BLOB
Al igual que recuperar datos BLOB grandes de SQL Server presenta desafíos para evitar la creación de copias
completas en memoria de todo el BLOB, surgen problemas similares cuando se intenta insertar un BLOB. La mejor
solución es en realidad bastante complicada. Implica enviar los datos al servidor en trozos y usar la sintaxis de BLOB
UPDATE en el lugar. La MSDN tiene esto que decir en la sección de Comentarios:
Use la cláusula .WRITE (expresión, @Offset, @Length) para realizar una actualización parcial o completa de los
tipos de datos varchar (max), nvarchar (max) y varbinary (max). Por ejemplo, una actualización parcial de una
columna varchar (max) podría eliminar o modificar solo los primeros 200 caracteres de la columna, mientras que
una actualización completa eliminaría o modificaría todos los datos de la columna. Para obtener el mejor
rendimiento, recomendamos que los datos se inserten o actualicen en tamaños de trozos que sean múltiplos de
8040 bytes.
Para implementar dicha semántica, escribiremos una segunda implementación de Stream, esta vez para las
subidas:
public class SqlStreamUpload: Stream
{
public SqlCommand InsertCommand {get; set;}
public SqlCommand UpdateCommand {get; set;}
public SqlParameter InsertDataParam {get; set;}
public SqlParameter UpdateDataParam {get; set;}
public override bool CanRead
{
get {return false;}
}
public override bool CanSeek
{
get {return false;}
}
public override void Flush()
{
}
public override long Length
{
get {throw new NotImplementedException();}
26-5-2019
public override bool CanWrite
{
get {return true;}
}
5
}
public override long Position
{
get; set;
}
public override int Read(byte[] buffer, int offset, int count)
{
throw new NotImplementedException();
}
public override long Seek(long offset, SeekOrigin origin)
{
throw new NotImplementedException();
}
public override void SetLength(long value)
{
throw new NotImplementedException();
}
public override void Write(byte[] buffer, int offset, int count)
{
byte[] data = buffer;
if (offset != 0 ||
count != buffer.Length)
{
data = new byte[count];
Array.Copy(buffer, offset, data, 0, count);
}
if (0 == Position &&
null != InsertCommand)
{
InsertDataParam.Value = data;
InsertCommand.ExecuteNonQuery();
}
else
{
UpdateDataParam.Value = data;
UpdateCommand.ExecuteNonQuery();
}
Position += count;
}
}
Esta implementación de Stream utiliza dos objetos SqlCommand: un InsertCommand para guardar el primer
fragmento, y un UpdateCommand para guardar los fragmentos subsiguientes. Tenga en cuenta que el tamaño de
los fragmentos (los 8040 bytes óptimos) no se especifica en ninguna parte, lo que se logra fácilmente envolviendo
el SqlStreamUpload en una instancia de BufferedStream.
create table media (
[media_id] int not null identity(1,1),
[file_name] varchar(256),
[content_type] varchar(256),
[content_coding] varchar(256),
[content] varbinary(max),
constraint pk_media_id primary key([media_id]),
constraint unique_file_name unique ([file_name]));
26-5-2019
La tabla MEDIA
6
Esta tabla contiene los archivos multimedia descargables. Los archivos se identifican por su nombre, por lo que los
nombres tienen una restricción única. He agregado una clave primaria de IDENTIDAD, porque en un CMS estos
archivos a menudo son referencias de otras partes de la aplicación e INT es una clave de referencia más corta que
un nombre de archivo. El campo tipo de contenido es necesario para saber el tipo de archivo: "imagen / png",
"imagen / jpg", etc. (consulte los tipos de medios de la IANA registrados oficialmente. El campo de codificación de
contenido es necesario si los archivos se almacenan comprimidos y un HTTP de codificación de contenido El
encabezado con la etiqueta "gzip" o "desinflar" se debe agregar a la respuesta. Tenga en cuenta que la mayoría de
los tipos de imágenes (JPG, PNG) no se comprimen bien, ya que estos formatos de archivo ya incluyen un algoritmo
de compresión y cuando se comprimen nuevamente con el los algoritmos de transferencia HTTP comunes (gzip,
deflate, comprimir) generalmente aumentan de tamaño.
El SqlMediaRepository
La última pieza del rompecabezas: una implementación de la interfaz IMediaRepository que utiliza un servidor de
SQL Server para el almacenamiento de archivos:
public class SqlMediaRepository: IMediaRepository
{
private SqlConnection GetConnection()
{
SqlConnectionStringBuilder scsb = new SqlConnectionStringBuilder(
ConfigurationManager.ConnectionStrings["Images"].ConnectionString);
scsb.Pooling = true;
SqlConnection conn = new SqlConnection(scsb.ConnectionString);
conn.Open();
return conn;
}
public void PostFile(
HttpPostedFileBase file,
out string fileName)
{
fileName = Path.GetFileName(file.FileName);
SqlCommand cmdUpdate = new SqlCommand(
@"UPDATE media
SET content.write (@data, NULL, NULL)
WHERE file_name = @content_disposition;", conn, trn);
cmdUpdate.Parameters.Add("@data", SqlDbType.VarBinary, -1);
26-5-2019
using (SqlConnection conn = GetConnection())
{
using (SqlTransaction trn = conn.BeginTransaction())
{
SqlCommand cmdInsert = new SqlCommand(
@"INSERT INTO media (
file_name,
content_type,
content_coding,
content)
values (
@content_disposition,
@content_type,
@content_coding,
@data);", conn, trn);
cmdInsert.Parameters.Add("@data", SqlDbType.VarBinary, -1);
cmdInsert.Parameters.Add("@content_disposition", SqlDbType.VarChar, 256);
cmdInsert.Parameters["@content_disposition"].Value = fileName;
cmdInsert.Parameters.Add("@content_type", SqlDbType.VarChar, 256);
cmdInsert.Parameters["@content_type"].Value = file.ContentType;
cmdInsert.Parameters.Add("@content_coding", SqlDbType.VarChar, 256);
cmdInsert.Parameters["@content_coding"].Value = DBNull.Value;
7
cmdUpdate.Parameters.Add("@content_disposition", SqlDbType.VarChar, 256);
cmdUpdate.Parameters["@content_disposition"].Value = fileName;
using (Stream uploadStream = new BufferedStream(
new SqlStreamUpload
{
InsertCommand = cmdInsert,
UpdateCommand = cmdUpdate,
InsertDataParam = cmdInsert.Parameters["@data"],
UpdateDataParam = cmdUpdate.Parameters["@data"]
}, 8040))
{
file.InputStream.CopyTo(uploadStream);
}
trn.Commit();
}
}
}
public bool GetFileByName(string fileName, out FileDownloadModel file)
{
SqlConnection conn = GetConnection();
try
{
SqlCommand cmd = new SqlCommand(
@"SELECT file_name,
content_type,
content_coding,
DATALENGTH (content) as content_length,
content
FROM media
WHERE file_name = @fileName;", conn);
SqlParameter paramFilename = new SqlParameter(
@"fileName", SqlDbType.VarChar, 256);
paramFilename.Value = fileName;
cmd.Parameters.Add(paramFilename);
SqlDataReader reader = cmd.ExecuteReader(
CommandBehavior.SequentialAccess |
CommandBehavior.SingleResult |
CommandBehavior.SingleRow |
CommandBehavior.CloseConnection);
if (false == reader.Read())
{
reader.Dispose();
conn = null;
file = null;
return false;
}
string contentDisposition = reader.GetString(0);
string contentType = reader.GetString(1);
string contentCoding = reader.IsDBNull(2) ? null : reader.GetString(2);
long contentLength = reader.GetInt64(3);
Stream content = new SqlReaderStream(reader, 4);
}
finally
{
if (null != conn)
{
conn.Dispose();
}
26-5-2019
file = new FileDownloadModel
{
FileName = contentDisposition,
ContentCoding = contentCoding,
ContentType = contentType,
ContentLength = contentLength,
Content = content
};
conn = null; // ownership transfered to the reader/stream
return true;
8
}
}
}
Normalmente, las aplicaciones MVC tienden a usar un repositorio basado en LINQ. En este caso, aunque no pude
aprovechar LINQ debido a los requisitos peculiares de la implementación de transmisión eficiente para BLOB
grandes. Por lo tanto, este Repositorio usa código SqlClient simple de vainilla.
El método GetFileByName obtiene una fila de la tabla de medios y devuelve un FileDownloadModel con un objeto
SqlReaderStream que está envolviendo el resultado del comando SELECT. Tenga en cuenta que no pude
implementar el patrón típico de "uso" para la SqlConnection desechable porque la conexión debe permanecer
abierta hasta que el comando SqlDataReader termine la transmisión en el BLOB, y la secuencia será controlada por
el marco MVC que ejecuta ActionResult después de que GetFileByName sea terminado. La conexión se cerrará
cuando se elimine el SqlReaderStream, debido a la bandera CommandBehavior.CloseConnection. La secuencia será
eliminada por el método FileStreamResult.WriteFile.
El método PostFile crea dos instrucciones SqlCommand, una para insertar la primera parte de los datos junto con
las otras columnas relevantes, y un comando de actualización que usa actualizaciones parciales de BLOB para
escribir las partes posteriores. Un objeto SqlStreamUpload está utilizando estos dos SqlCommand para transmitir
de manera eficiente en el archivo cargado. El BufferedStream intermedio se utiliza para crear fragmentos de carga
del tamaño crítico de 8040 bytes (ver más arriba). Si hubiera que comprimir el contenido, aquí es donde sucedería,
se colocaría un GZipStream frente a Bufferedstream para comprimir el archivo cargado, y el parámetro
"@content_coding" tendría que configurarse de modo "gzip”.
Almacenamiento en caché HTTP
He dejado fuera de esta implementación el control de almacenamiento en caché HTTP apropiado. El
almacenamiento en caché de HTTP es extremadamente importante en los sitios de alto volumen de tráfico, no hay
mejor manera de optimizar el procesamiento de una solicitud HTTP que nunca reciba dicha solicitud y que la
memoria caché de usuario-agente o algún proxy intermedio sirvan una respuesta. Nuestra aplicación tendría que
agregar los encabezados HTTP apropiados de ETag y Cache-Control y el MediaController tendría que tener una
acción para las solicitudes HEAD. La interfaz de IMediaRepository necesitaría un nuevo método para obtener todas
las propiedades del archivo sin el contenido real. Por el momento, dejaré esto como un ejercicio para el lector ...
Este artículo no trata de abordar la pregunta fundamental de si debe almacenar las imágenes en la base de datos o
en el sistema de archivos. Los argumentos pueden ser a favor y en contra de esto. Russel Sears, Catherine van
Ingen y Jim Gray publicaron un trabajo de investigación en 2006 sobre su análisis de comparación de rendimiento
entre el almacenamiento de archivos en la base de datos y el almacenamiento en el sistema de archivos: A BLOB or
Not a BLOB: Almacenamiento de objetos grandes en una Base de datos o Sistema de archivos ?. Llegaron a la
conclusión de que:
26-5-2019
BLOB o no BLOB
9
El estudio indica que si los objetos son más grandes que un megabyte en promedio, NTFS tiene
una clara ventaja sobre SQL Server. Si los objetos tienen menos de 256 kilobytes, la base de datos
tiene una clara ventaja. Dentro de este rango, depende de la intensidad de escritura de la carga
de trabajo y de la antigüedad de almacenamiento de una réplica típica en el sistema.
26-5-2019
Sin embargo, su estudio no comparaba cómo servir los archivos como respuestas HTTP. El hecho de que el servidor
web pueda servir un archivo de manera eficiente directamente desde el sistema de archivos sin que se ejecute
ningún código en la aplicación web cambia la ecuación bastante, e inclina el rendimiento fuertemente a favor del
sistema de archivos. Pero si ya ha considerado los pros y los contras y ha decidido que las ventajas de una copia de
seguridad / restauración consistente y una integridad referencial sólida ameritan BLOB almacenados en la base de
datos, espero que este artículo resalte una manera eficiente de devolver esos BLOB como respuestas HTTP.
10
FILESTREAM MVC: descarga y carga multimedial de SQL Server
En el artículo anterior, se ha mostrado cómo es posible usar una semántica de transmisión eficiente al descargar y
cargar imágenes desde SQL Server a través de ASP.Net MVC. En este artículo repasaremos un enfoque alternativo
que se basa en los tipos de columna FILESTREAM introducidos en SQL Server 2008.
¿Qué es FILESTREAM?
El almacenamiento FILESTREAM es una nueva opción disponible en SQL Server 2008 y versiones posteriores que
permite que las columnas BLOB se almacenen directamente en el sistema de archivos como archivos individuales.
Como archivos, los datos son accesibles a través de la API de acceso a archivos Win32 como ReadFile y WriteFile.
Pero al mismo tiempo, los mismos datos están disponibles a través de las operaciones normales de T-SQL como
SELECT o UPDATE. No solo eso, sino que los datos están contenidos de forma lógica en la base de datos, por lo que
se incluirán en una copia de seguridad de la base de datos, están sujetos a un comportamiento de confirmación y
retrotracción de transacciones ordinarios, se buscan en los índices de texto completo de SQL Server y siguen la
seguridad normal de SQL Server. Reglas de acceso: si se le otorga el permiso SELECCIONAR en la tabla, puede abrir
el archivo para leer. Hay algunas restricciones, por ejemplo. una base de datos con FILESTREAM no puede ser
duplicada. Para obtener una lista completa de restricciones y limitaciones, consulte Uso de FILESTREAM con otras
características de SQL Server. Tenga en cuenta que la edición de SQL Server Express no admite el almacenamiento
de FILESTREAM.
Otro atributo del almacenamiento de FILESTREAM es la limitación de tamaño: los valores de columna BLOB
normales tienen un tamaño máximo de 2Gb. Las columnas FILESTREAM están limitadas solo por el límite de
tamaño de volumen del sistema de archivos. 2Gb puede parecer un gran valor, pero tenga en cuenta que un
archivo multimedia como una secuencia de películas HD puede subir fácilmente a un tamaño de 5Gb.
Utilizando FILESTREAM
Una forma de utilizar las columnas FILESTREAM es tratarlas como valores BLOB ordinarios y manipularlos a través
de T-SQL. La única restricción es que no se admite la sintaxis de actualización parcial eficiente para BLOB. Es decir,
no se puede ejecutar UPDATE tabla SET columna. WRITE(…) WHERE … en una columna FILESTREAM. Pero cuando el
almacenamiento de FILESTREAM comienza a brillar es cuando se accede a través de la API del sistema de archivos.
Esto permite que la aplicación lea, escriba y busque de manera eficiente en un valor BLOB grande, tal como lo haría
en un archivo. De hecho, la aplicación sí lee, escribe y busca en un archivo :).
El requisito de proporcionar un contexto de transacción al manipular un valor de FILESTREAM a través del archivo
IO API destaca otro aspecto del trabajo con este nuevo tipo: una transacción T-SQL debe iniciarse y mantenerse
abierta mientras se manipula el archivo, y luego debe estar comprometido Si lo piensa, tal requisito es de esperar,
ya que dijimos que las columnas FILESTREAM están sujetas a la semántica normal de confirmación y retrotracción
26-5-2019
Las aplicaciones nativas de Win32 usan una nueva función de API OpenSqlFilestream que abre un HANDLE que
luego se puede usar con las funciones de la API de IO del archivo. Las aplicaciones administradas utilizan la nueva
clase SqlFileStream que expone una secuencia basada en el valor de FILESTREAM subyacente. Tanto la API nativa
como la administrada requieren como entrada dos valores especiales, un Nombre de ruta y un Contexto de
transacción que deben obtenerse previamente del servidor SQL utilizando T-SQL.
11
de transacciones de T-SQL, incluso cuando se manipulan mediante la API de lectura / escritura de archivos de
Windows.
Tabla de multimedios basada en FILESTREAM
Para tener la columna FILESTREAM, debemos tener un grupo de archivos especial en nuestra base de datos, un
grupo de archivos FILESTREAM. Puede agregar un nuevo grupo de archivos a su base de datos existente o puede
crear la base de datos con un grupo de archivos FILESTREAM desde cero. Además, la función FILESTREAM debe
estar habilitada en la instancia de SQL Server. Para todos los detalles, consulte Cómo comenzar con el
almacenamiento de FILESTREAM. Para mi proyecto, simplemente voy a crear una nueva base de datos con un
grupo de archivos FILESTREAM:
create database images
on (name='images_data', filename='c:\temp\images.mdf')
, filegroup FS contains FILESTREAM
(name = 'images_files', filename='c:\temp\images_files')
log on (name='images_log', filename='c:\temp\images.ldf');
go
La tabla de medios utilizada por nuestro proyecto MVC será similar a la utilizada en el artículo anterior, pero a la
columna de contenido se le agregará el atributo FILESTREAM. Es necesario que una tabla que tenga columnas
FILESTREAM tenga una columna ROWGUIDCOL, por lo que también agregaremos una de ellas:
create table media (
[media_id] int not null identity(1,1),
[file_name] varchar(256),
[content_type] varchar(256),
[content_coding] varchar(256),
[media_rowguid] uniqueidentifier not null
ROWGUIDCOL UNIQUE default newsequentialid(),
[content] varbinary(max) filestream,
constraint pk_media_id primary key([media_id]),
constraint unique_file_name unique ([file_name]));
go
Repositorio basado en FILESTREAM
/// <summary>
/// SQL Server FILESTREAM based implementation of the IMediaRepository
/// </summary>
public class FileStreamMediaRepository: IMediaRepository
{
/// <summary>
/// Gets an open connection to the SQL Server back end
/// </summary>
/// <returns>the SqlConneciton object, ready to use</returns>
26-5-2019
Si aún no ha leído el artículo anterior Descargar y cargar imágenes de SQL Server a través de ASP.Net MVC, ahora es
un buen momento para hacerlo. Vamos a reutilizar el mismo código y simplemente proporcionaremos una nueva
implementación para la interfaz IMediaRepository, una implementación que funciona con el almacenamiento de
FILESTREAM:
12
private SqlConnection GetConnection()
{
SqlConnectionStringBuilder scsb = new SqlConnectionStringBuilder(
ConfigurationManager.ConnectionStrings["Images"].ConnectionString);
scsb.Pooling = true;
scsb.AsynchronousProcessing = true;
SqlConnection conn = new SqlConnection(scsb.ConnectionString);
conn.Open();
return conn;
}
/// <summary>
/// Gets a file from the SQL repository
/// </summary>
/// <param name="fileName">filename to retrieve</param>
/// <param name="file">Output, the model for the file if found</param>
/// <returns>True if the file is found, False if not</returns>
public bool GetFileByName(string fileName, out FileDownloadModel file)
{
SqlConnection conn = GetConnection();
SqlTransaction trn = conn.BeginTransaction();
try
{
SqlCommand cmd = new SqlCommand(
@"SELECT file_name,
content_type,
content_coding,
DATALENGTH (content) as content_length,
content.PathName() as path,
GET_FILESTREAM_TRANSACTION_CONTEXT ()
FROM media
WHERE file_name = @fileName;", conn, trn);
SqlParameter paramFilename = new SqlParameter(
@"fileName", SqlDbType.VarChar, 256);
paramFilename.Value = fileName;
cmd.Parameters.Add(paramFilename);
using (SqlDataReader reader = cmd.ExecuteReader())
{
if (false == reader.Read())
{
reader.Close();
trn.Dispose();
conn.Dispose();
trn = null;
conn = null;
file = null;
return false;
}
file = new FileDownloadModel
{
FileName = contentDisposition,
ContentCoding = contentCoding,
ContentType = contentType,
ContentLength = contentLength,
Content = new MvcResultSqlFileStream
{
SqlStream = new SqlFileStream(path, context, FileAccess.Read),
Connection = conn,
Transaction = trn
}
};
conn = null; // ownership transfered to the stream
trn = null;
26-5-2019
string contentDisposition = reader.GetString(0);
string contentType = reader.GetString(1);
string contentCoding = reader.IsDBNull(2) ? null: reader.GetString(2);
long contentLength = reader.GetInt64(3);
string path = reader.GetString(4);
byte[] context = reader.GetSqlBytes(5).Buffer;
13
return true;
}
}
finally
{
if (null != trn)
{
trn.Dispose();
}
if (null != conn)
{
conn.Dispose();
}
}
}
/// <summary>
/// Adds a file to the SQL repository
/// </summary>
/// <param name="file">POST-ed file to be added</param>
/// <param name="fileName">The filename part of the uploaded file
public void PostFile(HttpPostedFileBase file, out string fileName)
{
fileName = Path.GetFileName(file.FileName);
path</param>
using (SqlConnection conn = GetConnection ())
{
using (SqlTransaction trn = conn.BeginTransaction ())
{
SqlCommand cmdInsert = new SqlCommand(
@"insert into media
(file_name, content_type, content_coding, content)
output
INSERTED.content.PathName(),
GET_FILESTREAM_TRANSACTION_CONTEXT ()
values
(@content_disposition, @content_type, @content_coding, 0x)", conn, trn);
cmdInsert.Parameters.Add("@content_disposition", SqlDbType.VarChar, 256);
cmdInsert.Parameters["@content_disposition"].Value = fileName;
cmdInsert.Parameters.Add("@content_type", SqlDbType.VarChar, 256);
cmdInsert.Parameters["@content_type"].Value = file.ContentType;
cmdInsert.Parameters.Add("@content_coding", SqlDbType.VarChar, 256);
cmdInsert.Parameters["@content_coding"].Value = DBNull.Value;
string path = null;
byte[] context = null;
// cmdInsert is an INSERT command that uses the OUTPUT clause
// Thus we use the ExecuteReader to get the
// result set from the output columns
//
using (SqlDataReader rdr = cmdInsert.ExecuteReader())
{
rdr.Read();
path = rdr.GetString(0);
context = rdr.GetSqlBytes(1).Buffer;
}
using (SqlFileStream sfs = new SqlFileStream(
path, context, FileAccess.Write))
{
file.InputStream.CopyTo(sfs);
}
trn.Commit ();
}
}
}
26-5-2019
}
14
La implementación es bastante sencilla. El método PostFile inicia una transacción, inserta una fila en la tabla de
imágenes y obtiene el contexto de transacción PahtName y FILESTREAM en la fila recién insertada, y luego abre un
SqlFileStream y copia en el archivo cargado directamente en el archivo que respalda la columna FILESTREAM. El
método GetFileByName obtiene el contexto de transacción PathName y FILESTREAM del archivo deseado y luego
devuelve un SqlFileStream que accede directamente al archivo en el que se almacena el valor de FILESTREAM.
Lo único que requiere explicación es la clase MvcResultSqlFileStream. Debido a que SqlFileStream usa un contexto
de transacción, se requiere que mantenga SqlConnection y SqlTransaction abiertos hasta que SqlFileStream
termine de acceder al contenido. En el método PostFile, este requisito se logra fácilmente porque todo el uso de
SqlFileStream se completa dentro del marco de la pila, pero el método GetFileByName tiene que devolver un
Stream que a su vez se pasa a un FileStreamResult por la clase MediaController y no hay control sobre cómo El
marco MVC utilizará este ActionResult devuelto. Mi solución es envolver el SqlFileStream en una nueva clase
derivada de Stream (ya que SqlFileStream está sellado, no se puede heredar ...) y luego manejar la administración
de recursos en esta clase derivada ’
/// <summary>
/// Implementation of System.IO.Stream based on a SqlFileStream
/// Disposes the connection, transaction and SqlFileStream objects
/// </summary>
public class MvcResultSqlFileStream: Stream
{
public SqlFileStream SqlStream { get; set; }
public SqlConnection Connection { get; set; }
public SqlTransaction Transaction { get; set; }
public override bool CanRead
{
get { return SqlStream.CanRead; }
}
public override bool CanSeek
{
get { return SqlStream.CanSeek; }
}
public override bool CanWrite
{
get { return SqlStream.CanWrite; }
}
public override void Flush()
{
SqlStream.Flush ();
}
public override long Position
{
get
{
return SqlStream.Position;
}
set
{
SqlStream.Position = value;
}
}
public override int Read(byte[] buffer, int offset, int count)
{
26-5-2019
public override long Length
{
get { return SqlStream.Length; }
}
15
return SqlStream.Read (buffer, offset, count);
}
public override long Seek(long offset, SeekOrigin origin)
{
return SqlStream.Seek(offset, origin);
}
public override void SetLength(long value)
{
SqlStream.SetLength(value);
}
public override void Write(byte[] buffer, int offset, int count)
{
SqlStream.Write(buffer, offset, count);
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
SqlStream.Dispose();
Transaction.Dispose();
Connection.Dispose();
}
base.Dispose(disposing);
}
}
Esta clase adicional se siente más bien innecesaria, pero desafortunadamente es necesaria porque el marco MVC
no tiene un manejo adecuado de los recursos que pasan del controlador. Si nos preguntara, ActionResult debería
implementar IDisposable, pero, por supuesto, nadie nos ha preguntado 😉
Conclusión
El manejo del contenido multimedia almacenado en la base de datos mediante el almacenamiento de FILESTREAM
hace que la experiencia del desarrollador sea más agradable. Se puede acceder al contenido de FILESTREAM
utilizando la semántica de transmisión basada en la clase SqlFileStream o utilizando la API de E / S del archivo
Win32 para aplicaciones nativas. En comparación, el artículo anterior que se basó en T-SQL directo para manipular
los valores BLOB tuvo que hacer algunos trucos muy poco intuitivos para lograr la semántica de transmisión,
especialmente para cargar un archivo. El almacenamiento FILESTREAM simplifica el desarrollo y es compatible con
todas las ediciones de SQL Server, incluido Express.
Si tiene la intención de implementar FILESTREAM en su entorno, se recomienda que revise primero algunos
artículos de la Base de conocimientos y artículos técnicos:
FILESTREAM Storage in SQL Server 2008
How to: Use FILESTREAM in a Firewall-Protected Environment
Recommendations and guidelines for improving SQL Server FileStream performance
26-5-2019



16
FILESTREAM permite a las aplicaciones basadas en SQL Server almacenar datos no estructurados, como
documentos e imágenes, en el sistema de archivos. FILESTREAM habilita a las aplicaciones basadas en SQL Server
almacenar datos no estructurados, tales como documentos e imágenes, en el sistema de archivos. Las aplicaciones
pueden aprovechar las API de transmisión de datos enriquecidas y el rendimiento del sistema de archivos al mismo
tiempo que mantienen la coherencia transaccional entre los datos no estructurados y los datos estructurados
correspondientes.
FILESTREAM integra Motor de base de datos de SQL Server con sistemas de archivos NTFS o ReFS almacenando
datos de objetos binarios grandes (BLOB) varbinary(max) como archivos en el sistema de archivos. Transact-SQL
puede insertar, actualizar, consultar, buscar y realizar copias de seguridad de los datos FILESTREAM. Las interfaces
del sistema de archivos de Win32 proporcionan el acceso de la transmisión por secuencias a los datos.
FILESTREAM usa la memoria caché del sistema NT para almacenar en memoria caché los datos de archivos. Esto
ayuda a reducir cualquier efecto que los datos FILESTREAM podrían tener en el rendimiento de motor de base de
datos. No se usa el grupo de búferes de SQL Server; por consiguiente, esta memoria está disponible para el
procesamiento de consultas.
FILESTREAM no se habilita automáticamente al instalar o actualizar SQL Server. Debe habilitar FILESTREAM
utilizando el Administrador de configuración de SQL Server y SQL Server Management Studio. Para utilizar
FILESTREAM, debe crear o modificar una base de datos que contenga un tipo especial de grupo de archivos. Luego,
debe crear o modificar una tabla de modo que contenga una columna varbinary(max) con el atributo FILESTREAM.
Después de completar estas tareas, puede usar Transact-SQL y Win32 para administrar los datos FILESTREAM.
Cuando se usa FILESTREAM
En SQL Server, los BLOB pueden ser datos de varbinary(max) estándar que almacena los datos en tablas u objetos
FILESTREAM varbinary(max) que almacenan los datos en el sistema de archivos. El tamaño y el uso de los datos
determinan si debería usar el almacenamiento de base de datos o el almacenamiento del sistema de archivos. Si las
condiciones siguientes son verdaderas, debería pensar en usar FILESTREAM:



Los objetos que se están almacenando son, por término medio, mayores de 1 MB.
El acceso de lectura rápido es importante.
Está desarrollando aplicaciones que usan un nivel intermedio para la lógica de la aplicación.
Para objetos de menor tamaño, el almacenamiento de BLOB varbinary(max) en la base de datos a menudo
proporciona un mejor rendimiento de la transmisión de datos.
Almacenamiento de FILESTREAM
Para especificar que una columna debería almacenar datos en el sistema de archivos, especifique el atributo
FILESTREAM en una columna varbinary(max). Esto hace que motor de base de datos almacene todos los datos para
esa columna en el sistema de archivos pero no en el archivo de base de datos.
26-5-2019
El almacenamiento de FILESTREAM se implementa como una columna varbinary(max) en la que los datos están
almacenados como BLOB en el sistema de archivos. Los tamaños de los BLOB están limitados solo por el tamaño del
volumen del sistema de archivos. La limitación varbinary(max) estándar de tamaños de archivo de 2 GB no se aplica
a BLOB que están almacenados en el sistema de archivos.
17
Los datos de FILESTREAM deben estar almacenados en los grupos de archivos FILESTREAM. Un grupo de archivos
FILESTREAM es un grupo de archivos especial que contiene los directorios de sistema de archivos en lugar de los
propios archivos. Estos directorios del sistema de archivos se denominan contenedores de datos. Los contenedores
de datos son la interfaz entre el almacenamiento del motor de base de datos y el almacenamiento del sistema de
archivos.
Cuando use el almacenamiento FILESTREAM, piense en lo siguiente:





Cuando una tabla contiene una columna FILESTREAM, cada fila debe tener un identificador de fila único
distinto de NULL.
Se pueden agregar varios contenedores de datos a un grupo de archivos FILESTREAM.
Los contenedores de datos FILESTREAM no pueden estar anidados.
Cuando se usan clústeres de conmutación por error, los grupos de archivos FILESTREAM deben estar en
recursos de disco compartido.
Los grupos de archivos FILESTREAM pueden estar en volúmenes comprimidos.
Administración integrada
Debido a que FILESTREAM se implementa como columna varbinary(max) y se integra directamente en el motor de
base de datos, la mayoría de las funciones y de las herramientas de administración de SQL Server funcionan sin la
modificación de los datos FILESTREAM. Por ejemplo, puede usar todos los modelos de recuperación y copia de
seguridad con datos FILESTREAM y se realizan copias de seguridad de los datos FILESTREAM con los datos
estructurados de la base de datos. Si no desea realizar una copia de seguridad de los datos FILESTREAM con datos
relacionales, puede usar una copia de seguridad parcial para excluir los grupos de archivos FILESTREAM.
Seguridad integrada
En SQL Server, los datos de FILESTREAM se protegen de la misma manera que los demás datos: concediendo
permisos en el nivel de tabla o columna. Si un usuario tiene permiso para la columna FILESTREAM de una tabla, el
usuario puede abrir los archivos asociados.
Nota
El cifrado no se admite en los datos FILESTREAM.
Solo la cuenta con la que la cuenta de servicio de SQL Server se ejecuta permiten los permisos al contenedor
FILESTREAM. Recomendamos que no se concedan permisos a ninguna otra cuenta en el contenedor de datos.
Nota
26-5-2019
Los inicios de sesión de SQL no funcionarán con contenedores FILESTREAM.SQL Solo la autenticación NTFS o ReFS
funcionará con contenedores FILESTREAM.
18
Acceso a datos BLOB con Transact-SQL y acceso de transmisión de datos del sistema de
archivos
Después de almacenar los datos en una columna FILESTREAM, puede tener acceso a los archivos usando las
transacciones de Transact-SQL o usando las API de Win32.
Acceso a Transact-SQL
Usando Transact-SQL, puede insertar, actualizar y eliminar los datos de FILESTREAM:
Puede usar una operación de inserción para rellenar previamente un campo FILESTREAM con un valor nulo, un
valor vacío o un dato insertado relativamente corto. Sin embargo, se envía una gran cantidad de datos de manera
más eficaz en un archivo que usa interfaces de Win32.
Al actualizar un campo FILESTREAM, modifica los datos de BLOB subyacentes en el sistema de archivos. Cuando un
campo FILESTREAM está establecido en NULL, se eliminan los datos de BLOB asociados al campo. No puede usar
ninguna actualización fragmentada de Transact-SQL, implementada como UPDATE. Write(), para realizar
actualizaciones parciales en los datos.
Al eliminar una fila, o eliminar o truncar una tabla que contiene datos FILESTREAM, elimina los datos de BLOB
subyacentes del sistema de archivos.
Acceso a la transmisión por secuencias del sistema de archivos
La compatibilidad de transmisión por secuencias de Win32 funciona en el contexto de una transacción de SQL
Server. Dentro de una transacción, puede usar las funciones FILESTREAM para obtener una ruta de acceso al
sistema de archivos de UNC lógica de un archivo. Tras ello, use la API OpenSqlFilestream para obtener un
identificador de archivos. Después, este identificador lo pueden usar las interfaces de transmisión por secuencias
de archivo de Win32, como ReadFile() y WriteFile(), para obtener acceso y actualizar el archivo a través del sistema
de archivos.
Dado que las operaciones de archivo son transaccionales, no puede eliminar ni cambiar el nombre de los archivos
FILESTREAM a través del sistema de archivos.
Modelo de la instrucción
El acceso del sistema de archivos de FILESTREAM modela una instrucción de Transact-SQL usando la apertura y el
cierre de archivo. La instrucción se inicia cuando un identificador de archivos se abre y finaliza cuando se cierra el
identificador. Por ejemplo, cuando se cierra un identificador de escritura, cualquier posible desencadenador de
AFTER que esté registrado en la tabla se desencadena como si la instrucción UPDATE estuviera
En FILESTREAM, el motor de base de datos controla el espacio de nombres del sistema de archivos físico de BLOB.
Una nueva función intrínseca, PathName, proporciona la ruta UNC lógica del BLOB que se corresponde con cada
celda de FILESTREAM de la tabla. La aplicación usa esta ruta de acceso lógica para obtener el identificador de
Win32 y funcionar en los datos de BLOB usando las interfaces del sistema de archivos de Win32 normales. La
función devuelve NULL si el valor de la columna FILESTREAM es NULL.
26-5-2019
Espacio de nombres de almacenamiento
19
Acceso al sistema de archivos transaccionales
Una nueva función intrínseca, GET_FILESTREAM_TRANSACTION_CONTEXT(), proporciona el token que representa
la transacción actual a la que la sesión está asociada. Se debe haber iniciado la transacción y no haberse anulado ni
confirmado todavía. Al obtener un token, la aplicación enlaza las operaciones de transmisión por secuencias del
sistema de archivos FILESTREAM con una transacción iniciada. La función devuelve NULL en caso de no haber
ninguna transacción explícitamente iniciada.
Se deben cerrar todos los identificadores de archivo antes de que la transacción se confirme o se anule. Si un
identificador se deja abierto más allá del ámbito de transacción, las lecturas adicionales frente al identificador
producirán un error; las escrituras adicionales frente al identificador tendrán éxito pero los datos reales no se
escribirán en el disco. De igual forma, si la base de datos o la instancia del Motor de base de datos se cierra, se
invalidan todos los identificadores abiertos.
Durabilidad transaccional
Con FILESTREAM, al confirmar la transacción, el Motor de base de datos asegura la durabilidad de la transacción
para los datos de BLOB FILESTREAM que se modifican del acceso a la transmisión por secuencias del sistema de
archivos.
Semántica de aislamiento
La semántica de aislamiento se rige por los niveles de aislamiento de transacción del Motor de base de datos. Se
admite el nivel de aislamiento de lectura confirmada para Transact-SQLTransact-SQL y el acceso al sistema de
archivos. Se admiten operaciones de lectura repetibles, así como serializables y aislamientos de instantáneas. No se
admite la lectura de datos sucios.
Las operaciones de apertura de acceso al sistema de archivos no esperan ningún bloqueo. En su lugar, se produce
un error inmediato de las operaciones de apertura si no pueden obtener acceso a los datos debido al aislamiento
de transacción. Se produce un error en las llamadas de API de transmisión por secuencias con
ERROR_SHARING_VIOLATION si la operación de apertura no puede continuar debido a la infracción de aislamiento.
Para permitir que se realicen actualizaciones parciales, la aplicación puede emitir un control FS de dispositivo
(FSCTL_SQL_FILESTREAM_FETCH_OLD_CONTENT) para capturar el contenido anterior en el archivo al que hace
referencia el identificador abierto. Esto desencadenará una copia de contenido antiguo de servidor. Para un mejor
rendimiento de la aplicación, y para evitar encontrarse con posibles tiempos de espera mientras trabaja con
archivos muy grandes, recomendamos que use E/S asincrónica.
Si se emite FSCTL una vez que se haya escrito en el identificador, se conservará la última operación de escritura y se
perderán las escrituras anteriores realizadas en el identificador.
Cuando una API del sistema de archivos no puede abrir un archivo a causa de una infracción de aislamiento, se
devuelve una excepción ERROR_SHARING_VIOLATION. Esta infracción de aislamiento se produce cuando dos
transacciones intentan acceder al mismo archivo. El resultado de la operación de acceso depende del modo en el
que se abrió el archivo y de la versión de SQL Server en la que se ejecute la transacción. En la tabla siguiente se
explican resumidamente los posibles resultados de dos transacciones que están accediendo al mismo archivo.
26-5-2019
API del sistema de archivos y niveles de aislamiento admitidos
20
Transacción 2 Resultado en SQL Server 2008
Resultado en SQL Server 2008 R2 y versiones
posteriores
Abrir para
lectura.
Abrir para
lectura.
Ambas son correctas.
Ambas son correctas.
Abrir para
lectura.
Abrir para
escritura.
Ambas son correctas. Las operaciones de escritura
de la transacción 2 no influyen en las operaciones
de lectura realizadas en la transacción 1.
Ambas son correctas. Las operaciones de escritura
de la transacción 2 no influyen en las operaciones
de lectura realizadas en la transacción 1.
Abrir para
escritura.
Abrir para
lectura.
Se produce un error en la operación de apertura de
la transacción 2 con una excepción
Ambas son correctas.
ERROR_SHARING_VIOLATION.
Abrir para
escritura.
Abrir para
escritura.
Se produce un error en la operación de apertura de Se produce un error en la operación de apertura de
la transacción 2 con una excepción
la transacción 2 con una excepción
ERROR_SHARING_VIOLATION.
ERROR_SHARING_VIOLATION.
Abrir para
lectura.
Abrir para
SELECT.
Ambas son correctas.
Ambas son correctas.
Abrir para
lectura.
Abrir para
UPDATE o
DELETE.
Ambas son correctas. Las operaciones de escritura
de la transacción 2 no influyen en las operaciones
de lectura realizadas en la transacción 1.
Ambas son correctas. Las operaciones de escritura
de la transacción 2 no influyen en las operaciones
de lectura realizadas en la transacción 1.
Abrir para
escritura.
Abrir para
SELECT.
La transacción 2 se bloquea hasta que la transacción
1 se confirme o finalice la transacción. O bien, se
Ambas son correctas.
agota el tiempo de espera de bloqueo de la
transacción.
Abrir para
escritura.
Abrir para
UPDATE o
DELETE.
La transacción 2 se bloquea hasta que la transacción
1 se confirme o finalice la transacción. O bien, se
agota el tiempo de espera de bloqueo de la
transacción.
La transacción 2 se bloquea hasta que la transacción
1 se confirme o finalice la transacción. O bien, se
agota el tiempo de espera de bloqueo de la
transacción.
Abrir para
SELECT.
Abrir para
lectura.
Ambas son correctas.
Ambas son correctas.
Abrir para
SELECT.
Abrir para
escritura.
Ambas son correctas. Las operaciones de escritura
de la transacción 2 no influyen en la transacción 1.
Ambas son correctas. Las operaciones de escritura
de la transacción 2 no influyen en la transacción 1.
Abrir para
UPDATE o
DELETE.
Abrir para
lectura.
Se produce un error en la operación de apertura de
la transacción 2 con una excepción
Ambas son correctas.
ERROR_SHARING_VIOLATION.
Abrir para
UPDATE o
DELETE.
Abrir para
escritura.
Se produce un error en la operación de apertura de Se produce un error en la operación de apertura de
la transacción 2 con una excepción
la transacción 2 con una excepción
ERROR_SHARING_VIOLATION.
ERROR_SHARING_VIOLATION.
Abrir para
SELECT con
REPEATABLE
READ.
Abrir para
lectura.
Ambas son correctas.
Ambas son correctas.
26-5-2019
Transacción 1
21
Transacción 1
Transacción 2 Resultado en SQL Server 2008
Abrir para
SELECT con
REPEATABLE
READ.
Abrir para
escritura.
Resultado en SQL Server 2008 R2 y versiones
posteriores
Se produce un error en la operación de apertura de Se produce un error en la operación de apertura de
la transacción 2 con una excepción
la transacción 2 con una excepción
ERROR_SHARING_VIOLATION.
ERROR_SHARING_VIOLATION.
Escritura continua desde clientes remotos
El acceso del sistema de archivos remoto a los datos FILESTREAM está habilitado por el protocolo Bloque de
mensajes de servidor (SMB). Si el cliente es remoto, no se almacena en caché ninguna operación de escritura del
lado cliente. Las operaciones de escritura siempre se enviarán al servidor. Los datos pueden se pueden almacenar
en memoria caché en el servidor. Recomendamos que las aplicaciones que se están ejecutando en clientes remotos
consoliden pequeñas operaciones de escritura para realizar menos operaciones de escritura mediante un tamaño
de datos mayor.
No se admite la creación de vistas asignadas de memoria (E/S asignada de memoria) usando un identificador
FILESTREAM. Si la asignación de memoria se usa para los datos FILESTREAM, el motor de base de datos no puede
garantizar la coherencia y la durabilidad de los datos o la integridad de la base de datos.
Por defecto, ASP.NET permite que la carga del tamaño máximo de archivo sea de 4 MB. Si utiliza una carga de
archivos ASP.NET de terceros para cargar archivos grandes y recibe una notificación de que la carga ha fallado
debido al límite máximo de tamaño de archivo, el componente correspondiente no admite la carga de archivos
grandes.
Para anular la restricción del tamaño máximo de archivo, debe modificar el archivo web.config y habilitar el
proceso de carga de archivos para manejar archivos más grandes. Con RadAsyncUpload (Telerik) puede subir
rápidamente archivos de más de 4 MB. RadAsyncUpload lo ayuda a superar la limitación de carga de 4 MB en
ASP.NET dividiendo los archivos grandes en trozos más pequeños y subiéndolos posteriormente. Puede controlar el
tamaño de los trozos y, por lo tanto, la cantidad de solicitudes al servidor necesarias para cargar el archivo, lo que
puede mejorar el rendimiento de su aplicación.
Use las Herramientas de desarrollo de su navegador para monitorear el cambio en el número de solicitudes al
servidor, cuando cambia el tamaño de los fragmentos, AsyncUpload divide los archivos grandes en fragmentos más
26-5-2019
pequeños para superar la limitación de ASP.NET de 4MB.
22
Vea la demostración para ver cómo el control de carga de ASP.NET de Telerik le permite cargar fácilmente archivos
grandes en ASP.NET sin modificar el valor en el archivo web.config. Nuestra documentación en línea proporciona
más detalles sobre cómo cargar archivos grandes con RadAsyncUpload.
Tareas Relacionadas










Habilitar y configurar FILESTREAM
Crear una base de datos habilitada para FILESTREAM
Crear una tabla para almacenar datos FILESTREAM
Obtener acceso a datos FILESTREAM con Transact-SQL Crear aplicaciones cliente para datos FILESTREAM
Obtener acceso a los datos FILESTREAM con OpenSqlFilestream
Realizar actualizaciones parciales de los datos FILESTREAM
Evitar conflictos con operaciones de base de datos en aplicaciones FILESTREAM
Mover una base de datos habilitada para FILESTREAM
Configurar FILESTREAM en un clúster de conmutación por error
Configurar un Firewall para el acceso de FILESTREAM
Contenido relacionado
Compatibilidad de FILESTREAM con otras características de SQL Server
Vistas de administración dinámica de secuencia de archivo y FileTable (Transact-SQL)
Vistas de catálogo de secuencia de archivo y FileTable (Transact-SQL)
Procedimientos almacenados del sistema de Filestream y FileTable (Transact-SQL)
26-5-2019




23
PRINCIPIO DE RESPONSABILIDAD UNICA (SINGLE RESPONSIBILITY)
PRINCIPIO OPEN/CLOSED
PRINCIPIO DE SUSTITUCION DE LISKOV (LISKOV SUBSTITUTION)
PRINCIPIO DE SEGREGACION DE INTERFACES (INTERFACE SEGREGATION)
PRINCIPIO DE INVERSION DE DEPENDENCIAS (DEPENDENCY INVERSION)
26-5-2019
REFACTORIZAR
CLASE BASE
CLASE DERIVADA
CLASE ABSTRACTA
CLASE SELLADA
METODO ABSTRACTO
ALTA COHESION
BAJO ACOPLAMIENTO
INYECCION DE DEPENDENCIA
SERVICE LOCATOR
HERENCIA
POLIMORFISMO
INMUTABILIDAD
INSTANCIA
24
En DevExpress ofrecemos dos bibliotecas principales que admiten el
marco MVC ASP.NET de Microsoft:
Controles de DevExpress ASP.NET MVC
DevExtreme MVC Controls
En esta publicación, nos sumergiremos en las dos bibliotecas
aparentemente similares que ofrecen controles para ASP.NET MVC. Veremos
las diferencias, similitudes, ventajas, desventajas y consecuencias.
Comencemos observando el desarrollo web del lado del servidor y del
lado del cliente, porque esta elección será fundamental en qué
biblioteca funcionará para usted.
Una de las preocupaciones más importantes para la elección de una
plataforma de desarrollo web es la arquitectura: la representación
tradicional de la interfaz de usuario del lado del servidor o del lado
del cliente. Es importante porque tiene muchas consecuencias.
Recomiendo leer esta sección, pero para aquellos que no tienen tiempo,
aquí hay un breve tldr:
26-5-2019
Elecciones arquitectónicas y consecuencias
25
La elección de la interfaz de usuario representada de servidor
contra cliente influye en la arquitectura general de un sistema de
software hasta el punto de que es recomendable analizar este asunto
antes de tomar decisiones sobre bibliotecas de componentes
específicos.
Lado del servidor
Si usa la IU representada por el servidor, esto implica que necesita un
servidor lo suficientemente potente como para representar las páginas
dinámicamente (por solicitud). Por lo general, esto funciona utilizando
archivos de plantilla en el servidor que describen las páginas, con un
modelo de extensión para insertar el contenido dinámico. Ambos marcos
usan este enfoque con ASP.NET WebForms que usan páginas ASPX y ASP.NET
MVC que usa archivos Razor como plantillas.
Si bien los modelos de componentes son diferentes entre WebForms y MVC,
ambos se representan en el entorno .NET del lado del servidor. Esto
significa que se pueden usar los métodos de acceso a datos basados en
.NET, la lógica de negocios escrita en C# y el estado en el lado del
servidor. Incluso la representación dinámica del navegador o específica
del usuario se puede realizar en el servidor, quizás dependiendo de la
configuración regional del usuario o los permisos consultados desde un
sistema de seguridad.
También hay una brecha para ser superada con este modelo: por ejemplo,
podría ser necesario escribir ciertos fragmentos de código dos veces,
en C# y JavaScript, como los requeridos para realizar tareas de
26-5-2019
Este modelo permite a los programadores que se sientan como en casa en
el entorno del servidor. Sin embargo, puede ser costoso con ASP.NET
WebForms debido a los numerosos viajes de ida y vuelta del servidor
requeridos para propósitos de representación. También hace que sea casi
imposible implementar la aplicación web resultante en entornos donde un
servidor podría no estar disponible, como las aplicaciones de
escritorio basadas en Electron o las aplicaciones móviles basadas en
Cordova. Se requiere mucho cuidado para evitar problemas de
escalabilidad porque los recursos del servidor son preciosos. Además,
no siempre es fácil agregar infraestructura, por ejemplo, cuando la
administración de estado del lado del servidor no se ha estructurado
correctamente.
26
validación de datos tanto para comentarios inmediatos del usuario en el
lado del cliente como para fines de lógica de negocios del lado del
servidor.
Sin embargo, si está creando un sitio web público, la representación
del lado del servidor es ideal para SEO (optimización de motores de
búsqueda). Debido a que el contenido HTML se genera en el servidor, los
motores de búsqueda como Google pueden rastrearlo, indexarlo y
almacenarlo en caché. La representación del lado del cliente no puede
hacer esto fácilmente sin trabajo adicional.
Lado del cliente
Con la IU representada por el cliente, un servidor no es estrictamente
necesario. El contenido HTML y JavaScript se abre camino en el cliente
de alguna manera: se puede descargar como contenido estático desde un
servidor, pero también se puede implementar como un paquete de
instalación para destinos de escritorio o móviles.
Los componentes utilizados por el cliente para representar la IU
compleja se escriben en JavaScript (o un lenguaje que se compila con
JavaScript, como TypeScript), y los desarrolladores generalmente eligen
los marcos de aplicación como Angular o React que los ayudan a
estructurar la aplicación del lado del cliente, pero también Requieren
integración con bibliotecas de componentes. Este modelo es una
reminiscencia de las aplicaciones cliente/servidor simples, y solo
utiliza protocolos web estándar para la comunicación de servicio y el
acceso a datos. Es posible crear sistemas de aplicación basados en
JavaScript de pila completa si los servicios están escritos en
JavaScript (probablemente para Node.js), y debido a la naturaleza sin
estado inherente de los servicios, generalmente es fácil escalar los
servidores según sea necesario. En general, los requisitos de
infraestructura del lado del servidor son menos complejos con esta
arquitectura, por lo que las implementaciones en la nube son fáciles y
permiten configuraciones de escalado automáticas.
26-5-2019
Debido a que el navegador del cliente ahora manejará el HTML y la
representación de esos elementos de la interfaz de usuario, la
aplicación del lado del cliente en su mayoría necesita los datos del
servidor. Por lo general, deberá crear servicios web para proporcionar
estos datos. Se puede utilizar la WebApi de ASP.NET MVC para
implementar dichos servicios, lo que nuevamente brinda al desarrollador
la oportunidad de usar mecanismos de acceso a datos basados en .NET,
pero se requieren protocolos adicionales, por ejemplo, para consultar
datos dinámicamente desde los servicios y transferirlos de vuelta al
cliente.
27
La IU representada por el cliente es el modelo arquitectónico de
elección para la mayoría de las aplicaciones web / móviles modernas y
un número creciente de aplicaciones de escritorio, que incluyen Office
365, Skype y Microsoft Visual Studio Code. La mejor recomendación para
tomar esta decisión es sopesar los pros y los contras cuidadosamente,
ya que se aplican a sus planes, la experiencia de los desarrolladores
en su equipo, su base de clientes y otros factores.
Esto nos lleva a la gran diferencia entre nuestros dos controles de
MVC. Mientras que ambos son controles de ASP.NET MVC del lado del
servidor, los Controles de MVC de DevExtreme representarán la IU del
lado del cliente debido a que envuelven los controles de JavaScript de
DevExtreme del lado del cliente en los controles de ASP.NET MVC. Por
otro lado, los controles DevExpress ASP.NET MVC representarán el HTML
del lado del servidor y luego lo entregarán al cliente.
Veamos ahora cómo estas dos bibliotecas comenzaron a comprender
nuestras motivaciones para crearlas.
Historia
1. DevExpress ASP.NET MVC Controls
En abril de 2009, Microsoft lanzó ASP.NET MVC y muchos desarrolladores
se interesaron rápidamente en este nuevo e interesante marco web.
Muchos de nuestros clientes solicitaron que proporcionáramos controles
de UI para ASP.NET MVC.
En ese momento, nuestra biblioteca ASP.NET WebForms proporcionaba
controles sólidos, estables y ricos en funciones. Así que tomamos la
decisión de tomar la API, las características y la representación de
nuestros controles de formularios web de ASP.NET y proporcionarlos como
controles MVC nativos. Este enfoque tuvo la ventaja de ahorrar tiempo y
ofrecerle un conjunto sólido de controles con muchas funciones.
Casi al mismo tiempo, JavaScript también estaba ganando popularidad
entre los desarrolladores web...
26-5-2019
En 2010, presentamos los controles DevExpress ASP.NET MVC como beta con
solo cinco controles. Luego, durante los siguientes siete años, con
cada lanzamiento continuamos aumentando los controles, las funciones y
el rendimiento. Hoy en día, el conjunto de controles DevExpress MVC es
sólido y tiene más de 65 controles.
28
2. Controles DevExtreme MVC
Desde finales de la década de 2000, la popularidad de JavaScript ha
seguido aumentando. Y hace muchos años, el desarrollo web cambió
decididamente hacia tecnologías basadas en JavaScript del lado del
cliente como jQuery, Node.js, Angular, Backbone, React, etc. sin razón,
porque el modelo de desarrollo del lado del cliente ofrece beneficios.
Muchos de nuestros clientes web existentes nos pidieron que
proporcionáramos controles de JavaScript que se representaban
exclusivamente en el lado del cliente.
Por lo tanto, en 2011, asignamos a un grupo independiente de nuestros
desarrolladores la tarea de crear un conjunto de controles de
JavaScript totalmente del lado del cliente que tuvieran características
similares a las de nuestros controles del lado del servidor. Después de
un año de desarrollo, la biblioteca DevExtreme nació en 2012. Hemos
continuado desarrollando esta biblioteca del lado del cliente desde
entonces para proporcionar más controles, características, estabilidad
y soporte para otros marcos del lado del cliente. Luego, alrededor de
2014, algunos de nuestros clientes web expresaron la necesidad de un
conjunto de controles MVC que utilizaron el enfoque de representación
del cliente. Así que decidimos envolver los controles de JavaScript
DevExtreme del lado del cliente como controles MVC del lado del
servidor.
En junio de 2016, los controles DevExtreme MVC se introdujeron como un
avance tecnológico comunitario (CTP). Ahora, después de varias
versiones en las que el equipo ha trabajado duro para mejorar los
controles DevExtreme MVC, proporcionan más de 65 controles,
características y una gran integración con ASP.NET MVC (y ASP.NET
Core).
¿Qué es lo correcto para ti?
Si bien las consideraciones arquitectónicas deberían ayudar a impulsar
la mayoría de los proyectos en su dirección preferida, también es
posible comparar nuestras bibliotecas en base a méritos técnicos
específicos. Es posible que la siguiente descripción general no sea
exhaustiva. No dude en preguntar sobre los temas que le faltan y
ampliaremos la publicación.
¿Qué recomiendas para mi escenario?
¿Cuáles son las principales diferencias entre los dos productos?
26-5-2019
Recientemente, hemos recibido preguntas de desarrolladores que
preguntan:
29
Para ayudarlo a decidir, lea las siguientes cuatro secciones que
comparan las fortalezas y debilidades de estas dos bibliotecas:
1. ¿Qué está incluido y qué falta?
A primera vista, parece que ambas bibliotecas tienen más de 65 juegos
de controles MVC y proporcionan controles clave como Cuadrícula de
datos, Gráficos, Editores, Navegación, etc. Sin embargo, no todos los
controles están disponibles en ambas bibliotecas y esto puede ser Un
factor decisivo clave. Por ejemplo, si necesita controles de hoja de
cálculo, cinta de opciones o RichEdit, estos solo estarán disponibles
en la suite de Controles DevExpress ASP.NET MVC.
Aquí hay un desglose de qué controles están disponibles en cada
biblioteca:
DevExpress ASP.NET MVC Controls (Server-Side)
DevExtreme MVC Controls (Client-Side)
Grid
PivotGrid
TreeList, TreeView
Scheduler
TreeList, TreeView
Scheduler
Navigation Controls
Editor Controls
Charts & Gauges
Forms
Reporting Suite
RichEdit
HTML Editor
Ribbon
26-5-2019
Spreadsheet
30
FileManager
CardView
Vertical Grid
Docking Suite
Spell Checker
ActionSheet, ScrollView, SlideOut
Bar, Circular, Linear Gauge
Map, TreeMap, VectorMap
RangeSelector
Bullet
Sparkline
FilterBuilder
Integrated into other
controls
Box, ResponsiveBox
Toast
2. Representación Pros / Contras
Los controles DevExpress ASP.NET MVC proporcionan representación del lado del
servidor.
Pros: Sólido con (muchos más) años de lanzamientos. Controles más potentes
disponibles.
Contras: Windows IIS servidores de alojamiento. Las devoluciones de llamada, sin
embargo, los datos se envían a través de formularios nativos, formularios MVC o
formularios Ajax MVC y se puede acceder a ellos mediante la acción de un controlador.
Pros: emiten un marcado semántico que es representado por el navegador del
cliente. La interacción del usuario es más rápida. Integración más estrecha con el
marco MVC de ASP.NET.
Contras: para la interactividad y la personalización en tiempo de ejecución,
tendrás que lidiar con los widgets principales de JavaScript.
A. Desempeño
26-5-2019
Los controles DevExtreme MVC proporcionan una representación del lado del cliente.
31
Los controles DevExtreme MVC tienen algunas ventajas menores en términos de
rendimiento debido a su representación del lado del cliente. Principalmente, la
interacción del usuario se sentirá más receptiva.
Por ejemplo, al agrupar filas en la cuadrícula de datos de DevExtreme MVC, encontrará
que solo recuperará los datos de las filas que necesita actualizar para mostrar el
diseño agrupado. En la imagen de las herramientas del navegador a continuación,
podemos ver que DevExtreme MVC Grid solo obtiene JSON para las filas de datos del
servidor y Grid mostrará y actualizará la interfaz de usuario en consecuencia:
26-5-2019
Sin embargo, DevExpress ASP.NET MVC GridView realizará una actualización parcial de la
página. Lo que significa que recuperará el HTML para el nuevo diseño de agrupación del
servidor y luego los entregará al cliente y los procesará en su lugar. Una
actualización parcial de la página es similar a la siguiente cuando se envía a través
de la cadena de solicitud / respuesta:
32
Las devoluciones de llamada requieren un esfuerzo adicional del servidor para volver a
representar estos fragmentos de código HTML para (potencialmente) un gran número de
clientes y, por lo tanto, aumentar el volumen de datos. Sin embargo, en una conexión
rápida, es probable que no note una gran diferencia en la velocidad.
B. Procesamiento semántico de HTML
26-5-2019
Como se mencionó, los controles DevExtreme MVC proporcionan un marcado semántico
cuando se procesan (haga clic en la imagen para una versión más grande):
33
26-5-2019
Los controles de DevExpress ASP.NET MVC más grandes, como el Grid, utilizan un diseño
basado en tabla para la representación. Sin embargo, los controles de navegación
(paginador, menú, barra de navegación, etc.) tienen una representación semántica que
se basa en listas, divs, etc.
34
Ambos enfoques de representación cumplen con los estándares HTML5.
3. Enlace de datos
Las Extensiones DevExpress ASP.NET MVC son controles del lado del servidor y, por lo
tanto, pueden enlazarse fácilmente a cualquier capa de datos .NET del lado del
servidor. Sin embargo, el enlace del lado del cliente no es compatible con estos
controles del lado del servidor.
Los controles DevExtreme MVC proporcionan varias formas de enlazar controles
conscientes de los datos en el lado del cliente. Estos controles también proporcionan
una capa de datos del lado del servidor que le permite enlazar en el lado del servidor
también. Este mecanismo no cambia las consideraciones arquitectónicas sobre el enlace
de datos, simplemente proporciona servicios web para el acceso a los datos de forma
automática, según los orígenes de datos existentes del lado del servidor.
Ambas bibliotecas son compatibles con .NET Framework 4+ y el mismo conjunto de
navegadores.
26-5-2019
4. Soporte de .NET Core
35
Sin embargo, los controles DevExtreme MVC también son compatibles con ASP.NET Core
1.0.1 y versiones posteriores (incluyendo 1.1, 2.0+), mientras que los controles
DevExpress ASP.NET MVC no lo hacen.
Si bien no es parte de esta comparación, los nuevos controles DevExpress ASP.NET
Bootstrap Core son compatibles exclusivamente con el marco Core de ASP.NET.
Utilizar juntos
Tenga en cuenta que si elige el marco completo clásico ASP.NET MVC (no ASP.NET Core),
puede usar ambos controles MVC juntos para complementarse entre sí. Por ejemplo, puede
usar DevExpress ASP.NET MVC Spreadsheet y DevExtreme MVC Grid en el mismo proyecto
ASP.NET MVC.
¡Estas dos bibliotecas MVC también se ofrecen en la Suscripción DevExpress ASP.NET!
La decisión es tuya
Para resumir, las dos suites de control MVC le ofrecen muchos controles de interfaz de
usuario excelentes y también se pueden usar juntos.
Sin embargo, si necesita elegir entre ellos, le ofreceré mis observaciones como
administrador del programa web que interactuó con muchos desarrolladores:
Los desarrolladores que prefieren la representación del lado del cliente suelen
elegir los controles DevExtreme MVC
Los fans de C # y la representación del lado del servidor normalmente eligen los
controles DevExpress ASP.NET MVC
Con suerte, la información anterior le ayudará a elegir. Si no es así, deje un
comentario a continuación sobre cómo puedo ayudar.
Autor:
26-5-2019
Email: mharry@devexpress.com
36
Descargar