Entradas

Comandos básicos de .NET CLI

Variables de Entorno

Deshabilitar Telemetría
DOTNET_CLI_TELEMETRY_OPTOUT = 1

Crear Solución

Crea un archivo .sln con el nombre proporcionado por el argumento --name.
dotnet new sln --name BlazorServerExample

Crear Proyecto a partir de Plantilla

Genera los archivos relacionados con la plantilla especificada, en una carpeta y proyecto cuyo nombre se define con el argumento -o.
dotnet new blazor -o BlazorApp
Para ver la lista de plantillas de proyectos, se puede usar el siguiente comando:
dotnet new list
Nombre de la plantilla Nombre corto Idioma Etiquetas
API web ASP.NET Core (native AOT) webapiaot [C#] Web/Web API/API/Service
Aplicación Blazor Server blazorserver [C#] Web/Blazor
Aplicación Blazor Server vacía blazorserver-empty [C#] Web/Blazor/Empty
... ... ... ...
Worker Service worker [C#],F# Common/Worker/Web

Agregar Proyecto a Solución

dotnet sln BlazorServerExample.sln add ./BlazorApp/BlazorApp.csproj

Referencias

Consulta de registros de IIS con Log Parser

Estamos llegando al final del ciclo de vida de los servidores actuales y nos estamos preparando para migrar de IIS a RedHat. Mientras se lleva a cabo la planificación de esta migración, es necesario obtener estadísticas sobre el tráfico en ciertas carpetas y los horarios de uso. Esto nos permitirá decidir qué se migrará y qué se conservará únicamente en el snapshot final.

Para esta tarea, recurrí a Log Parser, una herramienta lanzada en 2010 que permite ejecutar consultas sobre archivos de texto (en mi caso, archivos .log) utilizando una sintaxis similar a SQL. Sin embargo, me encontré con una sorpresa: aunque utiliza SQL, no se trata del estándar, ni siquiera del SQL usado por SQL Server. Su sintaxis es particular y algo de nicho, lo que hace que sea complicado obtener ayuda directa de herramientas de inteligencia artificial.

Probé generar consultas con ChatGPT y Gemini, pero ambas herramientas fallaron al dar una respuesta precisa debido a la naturaleza especializada, y no estandarizada del lenguaje que usa Log Parser. Por ello, aquí dejo algunos ejemplos de consultas que me han funcionado para generar estadísticas simples, por si a alguien más le resulta útil.

Configuración

Descarga Log Parser desde el sitio oficial https://www.microsoft.com/en-us/download/details.aspx?id=24659

Aparentemente, la ruta de Log Parser no se agrega a la ruta de variables de entorno, lo que significa que no se puede llamar simplemente desde la consola, por lo que se tiene especificar su ruta completa (C:\Program Files (x86)\Log Parser 2.2\LogParser.exe)

En mi caso hice un archivo.bat para que pueda escribir los archivos .sql aparte.
"C:\Program Files (x86)\Log Parser 2.2\logparser.exe" -i:IISW3C -o:CSV file:%~1

Documentación

Dependiendo de lo que se quiera consultar, Log Parser utiliza determinadas columnas que espera estén en los archivos de texto, mismas columnas se puede consultar con el parámetro -h (help) e -i: (input), donde especificamos el formato a consultar, en mi caso son los logs de IIS (IISW3C)
logparser -h -i:IISW3C
Lo que nos devuelve la lista de columnas con su respectivo tipo, en éste caso lo remuevo para mayor facilidad al copiar y pegar, adicionalmente de algunos ejemplos.
LogFilename , LogRow , date , time , c-ip , cs-username , s-sitename , s-computername , s-ip , s-port , cs-method , cs-uri-stem , cs-uri-query , sc-status , sc-substatus , sc-win32-status , sc-bytes , cs-bytes , time-taken , cs-version , cs-host , cs(User-Agent),  , cs(Cookie),  , cs(Referer),  , s-event , s-process-type , s-user-time , s-kernel-time , s-page-faults , s-total-procs , s-active-procs , s-stopped-procs

Consultas

Quiero hacer una lista de carpetas que se han usado en éste año (u_ex25*.log), y saber cúantas veces ha sido servido un recurso, (asociado a una carpeta, usando EXTRACT_TOKEN para obtener el primer folder), de manera exitosa (status 200).
SELECT 
	COUNT(*) AS Hits,
	EXTRACT_TOKEN(cs-uri-stem, 1, '/') AS Service
INTO servicios.csv	
FROM C:\inetpub\logs\LogFiles\W3SVC3\u_ex25*.log
WHERE 
	sc-status = 200 AND 
	cs-uri-stem LIKE '/%'
GROUP BY Service 
ORDER BY Hits DESC

Curiosamente, la herramienta ya estaba instalada en el servidor, lo que me hace pensar que ya había intentado usarla en el pasado… y que seguramente me desesperó. Conociéndome, es muy probable que en aquel momento incluso haya escrito alguna herramienta propia para obtener las estadísticas que necesitaba.

Generar entorno Portable de Python

Entorno de python portable

Últimamente me ha dado por hacer y usar algunas herramientas con Python, pero algunas veces, es preferible tener un entorno de Python aislado a la hora de instalar dependencias: ya van al menos 3 veces que he tenido que reinstalar tensor compilado para cuda, porque un script de requisitos lo reemmplaza por la versión para cpu. Winpython nos provee de una versión de Python portable, que corre en windows.

Enlace de descarga: https://github.com/winpython/winpython/releases/

Se descarga un archivo .ZIP, en mi caso es Winpython64-3.12.9.0dot.zip, donde copiamos las carpetas:

  • Notebooks
  • python
  • scripts
  • settings
y creamos un archivo bat para lanzar nuestra aplicación de Python. Lo único opcional es la línea python app.py
@echo off
cd /d ./scripts/
call activate.bat
python app.py

Cómo Reducir el Tamaño de una Base de Datos en SQL Server

Cuando trabajamos con una base de datos en entornos de desarrollo o pruebas, es común que crezca más de lo necesario; sin embargo, al momento publicarla o compartirla, es fundamental reducir su tamaño al mínimo posible.

En este artículo te mostraré los comandos esenciales de SQL Server que te permitirán reducir el tamaño físico de tu base de datos.

Obtener el nombre de archivos y su tamaño, se usa la siguiente consulta:

SELECT name, type_desc, size/128 AS SizeMB FROM sys.database_files;

Nos da el siguiente resultado:

name type_desc SizeMB
db_Prueba ROWS 9
db_Prueba_log LOG 28

Se usa SHRINKFILE para reducir el tamaño de archivo y sp_MSforeachtable para regenerar los índices. Si no usa RECOVERY SIMPLE, el tamaño del log no cambia.

ALTER DATABASE db_CorreosInstitucionales_UPIICSA SET RECOVERY SIMPLE;
DBCC SHRINKFILE ('db_Prueba_log', 1);
DBCC SHRINKFILE ('db_Prueba', 1);
EXEC sp_MSforeachtable 'ALTER INDEX ALL ON ? REBUILD';
ALTER DATABASE db_CorreosInstitucionales_UPIICSA SET RECOVERY FULL;

Volviendo a ejecutar la consulta para obtener el tamaño, obtenemos:

name type_desc SizeMB
db_Prueba ROWS 7
db_Prueba_log LOG 1

Lo que nos confirma que sí se redujo el tamaño del archivo.

Agregar o eliminar usuarios en Windows desde la consola (y quitar la expiración de contraseña)

Agregar usuario (y agregarlo a los administradores)

NET USER Usuario contraseña /add
NET LOCALGROUP administradores Usuario /add

Eliminar usuario (y quitarlo de los administradores)

NET LOCALGROUP administradores Usuario /delete
NET USER Usuario /delete

Deshabilitar la expiración de contraseña

Se necesita WMIC para hacer la configuración desde la línea de comando, pero fue depreciado en enero de 2024. se puede reinstalar la utilidad con DISM con el siguinte comando:
DISM /Online /Add-Capability /CapabilityName:WMIC~~~~
Deshabilitar la expiración de contraseña:
wmic UserAccount where Name="Usuario" set PasswordExpires=False

Inicio de sesión automático (Archivo.REG)

Windows Registry Editor Version 5.00

[HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon]
"AutoAdminLogon"="1"
"DefaultUserName"="Usuario"
"DefaultPassword"="contraseña"
Se ejecuta con:
regedit.exe /S Archivo.REG

Enlaces

Script en Python para renombrar múltiples archivos PNG fácilmente

Normalmente, el comando rename de DOS hace bien su trabajo. Pero en casos especiales, donde los archivos deben seguir un patrón específico, se queda corto. En mi caso, solo necesitaba renombrarlos de forma secuencial. Es la segunda vez en menos de un mes que me enfrento a esta necesidad, así que decidí dejar por aquí un pequeño script generado por ChatGPT.
import os

# Get all .png files in the current directory
png_files = sorted([f for f in os.listdir('.') if f.lower().endswith('.png')])

# Rename each file
for i, filename in enumerate(png_files, start=1):
    new_name = f"image_{i:05}.png"
    os.rename(filename, new_name)
    print(f'Renamed "{filename}" -> "{new_name}"')

Script en Python para exportar consultas de SQL Server a un archivo de texto

Requisitos

Windows

Microsoft ODBC Driver 18 for SQL Server - https://learn.microsoft.com/es-es/sql/connect/odbc/download-odbc-driver-for-sql-server?view=sql-server-ver16#download-for-windows

Alpine

Buscar la versión de Microsoft ODBC Driver a instalar y descargarla. https://learn.microsoft.com/es-es/sql/connect/odbc/download-odbc-driver-for-sql-server?view=sql-server-ver16#alpine
curl -O https://download.microsoft.com/download/3/5/5/355d7943-a338-41a7-858d-53b259ea33f5/msodbcsql18_18.3.3.1-1_amd64.apk
apk add --allow-untrusted msodbcsql18_18.3.3.1-1_amd64.apk
/etc/odbc.ini
[ODBC Driver 18 for SQL Server]
Description=Microsoft ODBC Driver 17 for SQL Server
Driver=/opt/microsoft/msodbcsql18/lib64/libmsodbcsql-18.3.so.3.1
UsageCount=1

Python

pip install --upgrade pip
pip install pyodbc

Código

import pyodbc

#https://learn.microsoft.com/es-es/sql/connect/odbc/download-odbc-driver-for-sql-server?view=sql-server-ver16#download-for-windows

connectionString = f'DRIVER={{ODBC Driver 18 for SQL Server}};SERVER=localhost;DATABASE=prueba;UID=sa;PWD=contraseña;TrustServerCertificate=yes;'

conn = pyodbc.connect(connectionString) 


SQL_QUERY = "SELECT  GETDATE() AS col1;"

cursor = conn.cursor()
cursor.execute(SQL_QUERY)

fout = open('query.txt','w', encoding="utf-8")

for r in cursor.fetchall():
    fout.write(f'{r.col1}\n')

fout.close()
cursor.close()
conn.close()