Pular para o conteúdo principal

C# - Primeiros passos com Docker

Hoje veremos como disponibilizar uma aplicação console criada com .Net a partir de um contêiner, mas antes vamos entender melhor o que é o Docker, o que exatamente estamos criando e ai sim criar nosso primeiro exemplo.

O que é o Docker?

"Docker é uma plataforma de software que permite criar, testar e implantar aplicativos em contêineres. Os contêineres são ambientes isolados que contêm todo o software necessário para executar um aplicativo, incluindo código, bibliotecas, dependências e configurações. Eles permitem que os aplicativos sejam executados de forma consistente em diferentes ambientes, como desenvolvimento, teste e produção."

Em resumo, com ele conseguimos empacotar aplicativos e dependências em contêineres portáteis e executá-los em qualquer máquina que possua o Docker instalado.

Podemos dizer que temos os seguintes passos para esse processo:

  • Criar uma imagem: possui um conjunto de informações para dizer como um container deve funcionar;
  • Criar um contêiner: com base em uma imagem criamos um contêiner, que é com quem interagimos para disponibilizar as aplicações criadas;

As etapas envolvidas para a criação de um contêiner são:

  • Criar um arquivo "Dockerfile" contendo informações sobre o ambiente a ser criado, como imagem a ser usada, qual aplicativo estará contido na mesma dentre outros;
  • Criar a imagem a partir de nosso "Dockerfile";
  • Criar um contêiner a partir de nossa imagem;
  • Iniciar o contêiner criado para que a aplicação contida no mesmo entre em funcionamento;

Criando a aplicação .Net

Vamos criar uma aplicação console que tem como objetivo "printar" em tela um contador, segundo a segundo. Essa aplicação pode rodar de forma indefinida ou pode rodar uma quantidade de vezes definida por parâmetro no momento da execução do mesmo.

Podemos criar a aplicação com o seguinte comando:

	
	
    dotnet new console -o DotNetDockerApp -n DotNet.Docker
    

Na classe "Program" teremos o seguinte código:

	
Console.WriteLine("Hello, DotNetDockerApp!");

var counter = 0;
var max = args.Length is not 0 ? Convert.ToInt32(args[0]) : -1;
while (max is -1 || counter < max)
{
    Console.WriteLine($"Counter: {++counter}");
    await Task.Delay(TimeSpan.FromMilliseconds(1_000));
}
    

Por fim vamos gerar uma versão "publicável" de nosso aplicativo com o comando abaixo (os arquivos devem ser gerados em "bin\Release\net7.0\publish"):

	
	
    dotnet publish -c Release
    

Dockerfile

Esse arquivo não possui extensão, deve ser criado no mesmo diretório do ".csproj", ele é usado pelo Docker para a criação da imagem. Para nosso exemplo esse arquivo fará o seguinte:

  • Baixar e usar uma imagem do sdk do .Net;
  • Copiar os arquivos do nosso projeto;
  • Restaurar pacotes e gerar uma versão release;
  • Baixar uma imagem para o runtime e fazer a cópia final dos artefatos compilados;

	
	
FROM mcr.microsoft.com/dotnet/sdk:7.0 AS build-env
WORKDIR /App

# Copy everything
COPY . ./
# Restore as distinct layers
RUN dotnet restore
# Build and publish a release
RUN dotnet publish -c Release -o out

# Build runtime image
FROM mcr.microsoft.com/dotnet/aspnet:7.0
WORKDIR /App
COPY --from=build-env /App/out .
ENTRYPOINT ["dotnet", "DotNet.Docker.dll"]
    

Criando nossa imagem e contêiner

Agora que nosso projeto está OK e já temos nosso Dockerfile basta executarmos os comandos necessários (devem ser executados no diretório onde o Dockerfile está):

	
	
    # Criar a imagem "counter-image" com base em nosso Dockerfile
    docker build -t counter-image -f Dockerfile .
    
    # Para ver as imagens disponíveis
    docker images
    
    # Agora que temos a imagem podemos criar nosso contêiner "core-counter" 
    docker create --name core-counter counter-image
    
    # Para ver todos os contêineres
    docker ps -a
    
    # Para iniciar nosso contêiner
    docker start core-counter
    
    # Para ver os contêineres em execução
    docker ps
    
    # Para ver o que nosso contêiner está fazendo podemos nos conectar a ele
    docker attach --sig-proxy=false core-counter
    
    # Se necessário podemos parar nosso contêiner
    docker stop core-counter
    
    # Se necessário podemos apagar nosso contêiner
    docker rm core-counter
    

Conclusão

Como podemos ver em nosso exemplo é bem simples trabalhar com contêineres e nossas aplicações .Net, ele torna a implantação de aplicativos muito mais fácil e rápida, pois não é necessário configurar manualmente cada máquina, sendo uma das principais ferramentas usadas em DevOps e implantação contínua.


Comentários

Mais visitadas

Funções de data Oracle

  Com o Oracle é possível obter uma série de resultados a partir de uma ou mais datas, como por exemplo verificar o último dia do mês ou verificar a quantidade de meses entre duas datas, então vamos a alguns exemplos:   Data atual do sistema: SYSDATE Remover meses de uma data: ADD_MONTHS(SYSDATE, -1) Adicionar meses de uma data: ADD_MONTHS(SYSDATE, +1) Buscar o último dia do mês: LAST_DAY(SYSDATE) Primeiro dia do mês: TRUNC(SYSDATE, ‘MONTH’) Quantidade de meses entre duas datas: MONTHS_BETWEEN(SYSDATE, ‘27/07/1982’) Primeiro dia do ano: TRUNC(SYSDATE, ‘YEAR’) Dias da semana: DECODE( TO_NUMBER( TO_CHAR          (SYSDATE, ‘D’) ) ,1, ‘domingo’ ,2, ‘segunda-feira’ ,3, ‘terça-feira’ ,4, ‘quarta-feira’ ,5, ‘quinta-feira’ ,6, ‘sexta-feira’ ,7,’sábado’ )

Funções de Data e Hora (Delphi)

É muito comum nos depararmos no dia a dia com a necessidade de manipular datas e horas, seja para um calculo de permanência, dias de atraso enfim, é praticamente escapar de alguma situação que necessite desse tipo de controle. Assim como a necessidade e se utilizar algum recurso para manipular as datas e horas de alguma maneira e freqüente, as duvidas de como o faze-lo também é, basta um breve olhar em qualquer fórum especializado e lá está, alguma duvida relacionada, por isso decidi falar um pouco sobre uma unit muito poderosa chamada DateUtils para a manipulação de data e hora, com um grande numero de métodos e classes que facilitam a vida de qualquer um. Alguns exemplos: CompareDate(constA, B: TDateTime): TValueRelationship; Compara apenas a data de dois valores (do tipo TDateTime) retornando: LessThanValue O primeiro valor é menor que o segundo EqualsValue Os valores são iguais GreaterThanValue O primeiro valor é maior que o segundo CompareDateTime(const A, B: TD

Como Verificar se um Objeto Existe (Delphi)

Em alguns momentos surge a necessidade de verificar se um determinado objeto existe, ou seja se já foi criado, principalmente quando se trabalha com criação dinâmica em tempo de execução, então vamos ao exemplo: - Vamos criar uma variável, um vetor do tipo caixa de texto: var Minha_caixa : array of TEdit; - Em seguida definir o tamanho desse vetor, no caso será dez: setLength(Minha_caixa, 10) - Agora iremos criar nossa caixa de texto: // lembrando que o vetor inicia em zero // logo o índice final é o tamanho total - 1 for vl_i := 0 to Length(Minha_caixa) -1 do begin Minha_caixa[vl_i] := TEdit.Create(self); with Minha_caixa[vl_i] do begin Parent := Self; Name := 'Caixa_N'+IntToStr(vl_i); Text := 'Esta é a '+IntToStr(vl_i)+' º caixa !'; ReadOnly := true; Height := 21; Width :=