10.4 Testando MLflow

Agora que temos um servidor MLflow configurado e rodando, podemos testar todos os seus componentes, incluindo o registro de modelos, que não funciona com um sistema de arquivos somente. Faremos isso por meio do exemplo de classificador de produtos que estivemos trabalhando desde o início deste livro.

10.4.1 Criando um projeto MLflow

Crie uma pasta, chamada classificador-produtos-mlflow, e um ambiente virtual, definido no arquivo python_env.yaml:

  • (Obs: se estiver utilizando Windows, utilize WSL, pois os ambientes virtuais no mlflow não funcionam bem no Windows)

python: "3.12.3"
build_dependencies:
  - pip
dependencies:
  - mlflow==2.18.0
  - scikit-learn==1.5.2
  - pandas==2.2.3
  - nltk==3.9.1

Agora vamos criar o código de treinamento do modelo. É o mesmo código que já estivemos trabalhando até agora, apenas adaptado para rodar de uma forma mais simples, fora do notebook. Mas funcionaria dentro do notebook também, desde que seja utilizado o kernel do ambiente Conda.

Faça o download do dataset (produtos.csv) e salve-o na pasta também.

Crie um arquivo chamado train.py, que tem o mesmo conteúdo de antes, porém em um único script. Também acrescentamos alguns parâmetros simples de configuração, e um código de registro da execução no MLflow:

import sys

import pandas as pd
import nltk
from nltk.corpus import stopwords
from sklearn.feature_extraction.text import CountVectorizer
from sklearn.naive_bayes import MultinomialNB
from sklearn.model_selection import train_test_split
from sklearn.pipeline import Pipeline
from sklearn.metrics import accuracy_score
import mlflow

# Esta função tenta registrar o experimento no MLflow
def tentar_registrar_experimento(p_test_size, p_include_names, accuracy, dataset, model):
    with mlflow.start_run():
        # Vamos registrar as métricas
        mlflow.log_metric("acuracia", accuracy)
        # E o dataset (deve ser um caminho para um arquivo)
        mlflow.log_artifact(dataset)
        # E o modelo treinado
        mlflow.sklearn.log_model(model, "modelo")

if __name__ == "__main__":
    nltk.download('stopwords')

    # temos dois parâmetros agora
    # p_test_size: percentual de casos de teste, entre 0 e 1. Default é 0.2
    p_test_size = float(sys.argv[1]) if len(sys.argv) > 1 else 0.2
    # p_include_names: se nomes devem ser incluídos no treinamento ou não, apenas descrição. Valor deve ser sim ou não. Default é sim
    p_include_names = sys.argv[2] if len(sys.argv) > 2 else 'sim'

    print("Treinando classificador de modelos...")
    print(f"Tamanho de testes={p_test_size}")
    print(f"Incluir nomes={p_include_names}")

    dataset = 'produtos.csv'

    products_data = pd.read_csv(dataset, delimiter=';', encoding='utf-8')

    if p_include_names == 'sim':
        # concatenando as colunas nome e descricao
        products_data['informacao'] = products_data['nome'] + products_data['descricao']
    else:
        # apenas a descricao
        products_data['informacao'] = products_data['descricao']

    # excluindo linhas com valor de informacao ou categoria NaN
    products_data.dropna(subset=['informacao', 'categoria'], inplace=True)
    products_data.drop(columns=['nome', 'descricao'], inplace=True)

    stop_words=set(stopwords.words("portuguese"))
    # transforma a string em caixa baixa e remove stopwords
    products_data['sem_stopwords'] = products_data['informacao'].str.lower().apply(lambda x: ' '.join([word for word in x.split() if word not in (stop_words)]))
    tokenizer = nltk.RegexpTokenizer(r"\w+")
    products_data['tokens'] = products_data['sem_stopwords'].apply(tokenizer.tokenize) # aplica o regex tokenizer
    products_data.drop(columns=['sem_stopwords','informacao'],inplace=True) # Exclui as colunas antigas

    products_data["strings"]= products_data["tokens"].str.join(" ") # reunindo cada elemento da lista
    products_data.head()


    X_train,X_test,y_train,y_test = train_test_split( # Separação dos dados para teste e treino
        products_data["strings"], 
        products_data["categoria"], 
        test_size = p_test_size, 
        random_state = 10
    )
    pipe = Pipeline([('vetorizador', CountVectorizer()), ("classificador", MultinomialNB())]) # novo

    pipe.fit(X_train, y_train)

    y_prediction = pipe.predict(X_test)
    accuracy = accuracy_score(y_prediction, y_test)

    print(f"Acurácia={accuracy}")

    # Terminamos o treinamento, vamos tentar fazer o registro
    tentar_registrar_experimento(p_test_size, p_include_names, accuracy, dataset, pipe)

Os comentários no código explicam o seu funcionamento em detalhes.

Vamos configurar um projeto MLflow, para facilitar a reproducibilidade. Crie um arquivo chamado MLproject, com o seguinte conteúdo:

name: classificador-produtos

python_env: python_env.yaml

entry_points:
  main:
    parameters:
      test_size: {type: float, default: 0.2}
      include_names: {type: string, default: 'sim'}
    command: "python train.py {test_size} {include_names}"

Podemos agora executar o projeto localmente e testá-lo. Para isso, precisamos criar um novo ambiente, de preferência em uma nova pasta (exemplo: ambiente-mlflow). Não é preciso instalar nada além do MLflow, pois ele irá criar/recriar o ambiente a cada execução, se necessário. Também precisamos indicar ao MLflow onde está o servidor, por meio de variáveis de ambiente:

python -m venv .venv
source .venv/bin/activate
pip install mlflow virtualenv
export MLFLOW_TRACKING_URI=http://localhost:5000
export MLFLOW_EXPERIMENT_NAME=classificador-produtos-sem-devops

Lembre-se, no Windows PowerShell a definição de variáveis de ambiente é diferente:

$env:MLFLOW_TRACKING_URI='http://localhost:5000'
$env:MLFLOW_EXPERIMENT_NAME='classificador-produtos-sem-devops'

Agora execute o comando mlflow run <pasta do projeto>, apontando para a pasta classificador-produtos-mlflow criada anteriormente. O comando irá procurar, na pasta indicada, pelo arquivo MLproject, e irá disparar a execução.

Poderíamos rodar com python train.py, mas aí precisaríamos ativar o ambiente definido em python_env.yaml manualmente. Também perderíamos as vantagens do empacotamento de projetos, que incluem, por exemplo, o rastreamento automático dos parâmetros definidos no arquivo MLproject.

Se não quiser executar agora, não tem problema. Faremos um ciclo DevOps completo a seguir, e a execução poderá ser feita daqui a pouco. Caso execute, você verá o resultado da execução aparecendo na interface do MLflow.

10.4.2 Subindo o projeto para o GitLab e executando experimentos

Vamos configurar um projeto no GitLab para poder controlar as versões cuidadosamente e testar nosso empacotamento. Crie um projeto no GitLab chamado classificador-produtos-mlflow e inicialize com o conteúdo da pasta com o projeto (lembre-se de excluir o conteúdo da pasta .venv, se tiver criado, pois ela é enorme e não deve ser versionada). Para facilitar os testes, crie o projeto com visibilidade pública.

Vamos rodar. Se não tiver feito ainda, abra um novo terminal e crie uma nova pasta/ambiente conforme as instruções no final da última seção. Se possível, para testar a reproducibilidade, crie tudo numa pasta limpa. Ou melhor ainda, tente em uma máquina diferente, se preferir, que tenha acesso de rede à máquina onde o MLflow está rodando. Neste caso, não esqueça de substituir o endereço definido na variável de ambiente pelo da máquina que está rodando o MLflow, e de verificar as conexões de firewall da rede.

Execute utilizando os seguintes comandos. Teste um, primeiro, para ver se deu certo, e os restantes depois, para criar várias execuções. Vamos também mudar o nome do experimento para diferenciar da execução anterior:

export MLFLOW_TRACKING_URI=http://localhost:5000
export MLFLOW_EXPERIMENT_NAME=classificador-produtos

mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.1 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.2 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.3 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.4 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.5 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.1 -P include_names=não
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.2 -P include_names=não
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.3 -P include_names=não
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.4 -P include_names=não
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.5 -P include_names=não

Não esqueça de substituir o endereço do repositório GitLab pelo que você criou, e o parâmetro main pelo nome do branch que você definiu. Vai demorar um tempinho até que o projeto seja transferido, o ambiente seja recriado, e o experimento, executado.

Podemos ver o resultado das execuções na interface visual. Abra o endereço no navegador: http://localhost:5000:

Experimente navegar pela interface e ver os resultados obtidos, as visualizações disponíveis. Veja como os commits no GitLab aparecem nessa interface também, para possibilitar a identificação da versão do código que originou cada experimento. Aparentemente, podemos retreinar com um tamanho de teste 0.25, pois o gráfico nos indica que há uma possibilidade de melhoria aqui. Experimente com esse parâmetro, se quiser.

10.4.3 Registrando modelos

Agora podemos registrar um modelo. Na tela de visualização dos experimentos, ordene pela métrica de acurácia, decrescentemente, e depois selecione o experimento clicando sobre o link indicado na figura.

Na visualização dos detalhes da execução, é possível conferir o comando executado, os parâmetros, as métricas, o dataset, e demais registros. É também possível registrar o modelo. Faça isso clicando no botão "Register Model".

Na janela aberta, escolha a opção "Create New Model" e escolha um nome para ele. No caso, escolhemos "classificador-produtos".

Quando uma execução gera um modelo, isso fica indicado na interface principal, por meio de um link.

Também é possível ver todos os modelos registrados clicando na aba "Models", no topo da página:

Note como, neste caso, só temos um modelo e uma versão. Em breve criaremos outra versão do modelo.

10.4.4 Subindo uma API HTTP para servir o modelo registrado

Obs: O conteúdo desta seção encontra-se desatualizado. Os exemplos podem não funcionar corretamente nas versões mais recentes das ferramentas

Agora vamos servir esse modelo como uma API HTTP, utilizando o servidor embutido do MLflow. Faremos isso usando Docker.

Na pasta mlflow-server, que criamos na seção anterior, crie uma pasta chamada classificador-produtos. Dentro dela, crie um Dockerfile:

FROM continuumio/miniconda3:latest

RUN pip install mlflow

ADD . /app
WORKDIR /app

COPY wait-for-it.sh wait-for-it.sh 
RUN chmod +x wait-for-it.sh

Essa imagem servirá para rodarmos apenas o MLflow para subir o modelo como uma API, então precisamos apenas prepará-la. Veja como não há nada exceto a instalação do mlflow em uma imagem miniconda, e o script wait-for-it.sh (o mesmo que já utilizamos antes). Agora modifique o arquivo compose.yml:

services:
  s3:
    image: minio/minio:RELEASE.2022-05-26T05-48-41Z.hotfix.15f13935a
    restart: unless-stopped
    ports:
      - "9000:9000"
      - "9001:9001"
    environment:
      - MINIO_ROOT_USER=${AWS_ACCESS_KEY_ID}
      - MINIO_ROOT_PASSWORD=${AWS_SECRET_ACCESS_KEY}
    command: server /data --console-address ":9001"
    networks:
      - internal
      - public
    volumes:
      - minio_volume:/data
  create_s3_buckets:
    image: minio/mc
    depends_on:
      - "s3"
    entrypoint: >
      /bin/sh -c "
      until (/usr/bin/mc alias set minio http://s3:9000 '${AWS_ACCESS_KEY_ID}' '${AWS_SECRET_ACCESS_KEY}') do echo '...waiting...' && sleep 1; done;
      /usr/bin/mc mb minio/${AWS_BUCKET_NAME};
      exit 0;
      "
    networks:
      - internal
  db:
    image: mysql/mysql-server:5.7.38
    restart: unless-stopped
    container_name: mlflow_db
    expose:
      - "3306"
    environment:
      - MYSQL_DATABASE=${MYSQL_DATABASE}
      - MYSQL_USER=${MYSQL_USER}
      - MYSQL_PASSWORD=${MYSQL_PASSWORD}
      - MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD}
    volumes:
      - db_volume:/var/lib/mysql
    networks:
      - internal
  mlflow:
    container_name: mlflow-server-container
    image: mlflow-server
    restart: unless-stopped
    build:
      context: .
      dockerfile: Dockerfile
    ports:
      - "5000:5000"
    environment:
      - AWS_ACCESS_KEY_ID=${AWS_ACCESS_KEY_ID}
      - AWS_SECRET_ACCESS_KEY=${AWS_SECRET_ACCESS_KEY}
      - AWS_DEFAULT_REGION=${AWS_REGION}
      - MLFLOW_S3_ENDPOINT_URL=http://s3:9000
    networks:
      - public
      - internal
    entrypoint: bash ./wait-for-it.sh db:3306 -t 90 -- mlflow server --backend-store-uri mysql+pymysql://${MYSQL_USER}:${MYSQL_PASSWORD}@db:3306/${MYSQL_DATABASE} --artifacts-destination s3://${AWS_BUCKET_NAME}/ --serve-artifacts -h 0.0.0.0
+  classificador-produtos:
+    build:
+      context: .
+      dockerfile: classificador-produtos/Dockerfile
+    environment:
+      - MLFLOW_TRACKING_URI=http://mlflow:5000
+    ports:
+      - "8080:8080"
+    networks:
+      - public
+      - internal
+    entrypoint: bash ./wait-for-it.sh mlflow:5000 -t 90 -- mlflow models serve -m 'models:/classificador-produtos/1' -p 8080 -h 0.0.0.0
networks:
  internal:
  public:
    driver: bridge
volumes:
  db_volume:
  minio_volume:

Iniciaremos um novo serviço, responsável por subir a API com o modelo registrado. Veja como definimos o endereço do servidor do MLflow na variável de ambiente MLFLOW_TRACKING_URI. Isso é a única coisa necessária para que possamos recuperar o modelo.

Veja agora o conteúdo de entrypoint:

  • bash ./wait-for-it.sh mlflow:5000 -t 90: espera que o MLflow esteja funcionando para subir a API

  • mlflow models serve: comando do MLflow para servir um modelo

  • -m 'models:/classificador-produtos/1': indica ao MLflow que é para recuperar o modelo chamado classificador-produtos em sua versão 1 (a única que temos até agora, que registramos há pouco)

  • -p 8080 -h 0.0.0.0: escuta na porta 8080 requisições de qualquer origem

Interrompa a execução do docker compose e rode-o novamente. Lembre-se de incluir a opção --build para forçar a reconstrução das imagens, se necessário.

docker compose up --build

Acompanhe enquanto os serviços são executados um a um. Veja como o serviço da API demora para subir. Isso acontece pois ele irá criar todo o ambiente Conda novamente para hospedar o modelo, agora dentro do contêiner Docker. Para isso, irá instalar as dependências, entre as tarefas de criação do ambiente.

Quando estiver tudo rodando, podemos fazer um teste, enviando uma requisição POST:

Fazemos uma requisição com conteúdo text/csv pois estamos enviando texto. O MLflow tem suporte a esse formato.

10.4.5 Utilizando modelos em tarefas offline

No mesmo ambiente onde o mlflow foi instalado anteriormente, crie um arquivo chamado batch.py:

import mlflow.pyfunc
import os

model_name = "classificador-produtos"
model_version = 1
os.environ['MLFLOW_TRACKING_URI']='http://localhost:5000'

model = mlflow.pyfunc.load_model(
    model_uri=f"models:/{model_name}/{model_version}"
)

data = ["boneca de pano", "halo", "senhor dos anéis", "senhor dos aneis"]

categorias = model.predict(data)

print(categorias)

O código é extremamente simples, e ilustra como o registro de modelos facilita a busca e uso de modelos. Execute-o e veja as predições sendo produzidas em lote (batch). Poderíamos fazer o mesmo que fizemos na Seção 2.1, salvando as predições em um banco de dados.

10.4.6 Utilizando versões Staging e Production (está sendo descontinuado)

Também é possível trabalhar com versões preparadas como Staging e Production. Dessa forma, os usuários dos modelos não precisam se preocupar com qual é a melhor versão. Basta saber que é a versão em produção. Fica a cargo dos desenvolvedores e avaliadores dos modelos decidir qual é a melhor versão. Vamos fazer isso agora. Mas antes, vamos criar uma nova versão. Vamos fazer a já famigerada mudança para ignorar acentos no treinamento. No projeto classificador-produtos, que é onde fazemos o treinamento, vamos fazer as modificações necessárias. Se tiver apagado a pasta, faça uma clonagem do repositório do GitLab novamente.

Primeiro, vamos ajustar o arquivo python_env.yaml:

python: "3.12.3"
build_dependencies:
  - pip
dependencies:
  - mlflow==2.18.0
  - scikit-learn==1.5.2
  - pandas==2.2.3
  - nltk==3.9.1
+  - unidecode==1.3.8

Agora vamos modificar o código de train.py:

import sys

import pandas as pd
import nltk
from nltk.corpus import stopwords
from sklearn.feature_extraction.text import CountVectorizer
from sklearn.naive_bayes import MultinomialNB
from sklearn.model_selection import train_test_split
from sklearn.pipeline import Pipeline
from sklearn.metrics import accuracy_score
import mlflow
+from unidecode import unidecode

...

stop_words=set(stopwords.words("portuguese"))
# transforma a string em caixa baixa e remove stopwords

-products_data['sem_stopwords'] = products_data['informacao'].str.lower().apply(lambda x: ' '.join([word for word in x.split() if word not in (stop_words)]))
+products_data['sem_stopwords'] = products_data['informacao'].str.lower().apply(lambda x: ' '.join([unidecode(word) for word in x.split() if word not in (stop_words)]))

...

Para treinar novamente, basta fazer um novo commit:

git commit -am "Corrigindo modelo para tratamento de acentos" 
git push

Agora, no terminal com o ambiente do mlflow, execute o comando que faz mais seis treinamentos com a nova versão (nem testaremos com a opção sem nomes, pois já vimos que são todas piores). Caso necessário, reative o mesmo ambiente de antes e exporte as variáveis de ambiente necessárias:

export MLFLOW_TRACKING_URI=http://localhost:5000
export MLFLOW_EXPERIMENT_NAME=classificador-produtos
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.1 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.2 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.25 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.3 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.4 -P include_names=sim
mlflow run -v main https://gitlab.com/daniel.lucredio/classificador-produtos-mlflow.git -P test_size=0.5 -P include_names=sim

A nova execução irá demorar pois, lembre-se, modificamos o ambiente (python_env.yaml), que precisará ser recriado antes da execução. Ao término da execução, teremos seis novas execuções para analisar. Veja como elas tem uma versão de código diferente, pois fizemos a alteração no GitLab, e isso fica registrado no MLflow.

Aparentemente, nossa alteração não alterou a acurácia dos modelos, portanto podemos mais uma vez apenas selecionar a opção com maior acurácia:

Veja como a outra execução já está registrada como modelo, na versão 1. Agora vamos registrar essa nova execução como a versão 2 desse mesmo modelo:

Ao escolher o mesmo modelo que antes, será criada a versão 2. Clique na aba "Models" no topo da página, e depois em "classificador-produtos" para ver as duas versões:

Como discutido anteriormente, poderíamos simplesmente nos referir aos números das versões para utilizar esses modelos, seja ao subir uma API ou ao realizar uma tarefa em batch, como já fizemos até agora. Mas podemos também usar o conceito de Stage. Vamos colocar a versão 1 como Production e a versão 2 como Staging. Isso pode ser feito clicando-se na versão e escolhendo a opção "Transition to":

Obs: pode ser necessário desabilitar a opção "New model registry UI", caso esteja habilitada, pois esta função está sendo depreciada. Pode ser que, no momento da leitura, esta opção nem exista mais. Neste caso, vá para a seção seguinte.

Faça isso agora. Coloque a versão 1 em estágio de Production e a versão 2 em estágio de Staging.

Agora vamos modificar o arquivo batch.py para recuperar os modelos utilizando fases ao invés de números:

import mlflow.pyfunc
import os

model_name = "classificador-produtos"
-model_version = 1
os.environ['MLFLOW_TRACKING_URI']='http://localhost:5000'

-model = mlflow.pyfunc.load_model(
+staging_model = mlflow.pyfunc.load_model(
+    model_uri=f"models:/{model_name}/Staging"
+)
+production_model = mlflow.pyfunc.load_model(
+    model_uri=f"models:/{model_name}/Production"
+)

data = ["boneca de pano", "halo", "senhor dos anéis", "senhor dos aneis"]

-categorias = model.predict(data)
+categorias_staging = staging_model.predict(data)
+categorias_production = production_model.predict(data)

-print(categorias)
+print("Modelo Staging: ")
+print(categorias_staging)
+print("Modelo Production: ")
+print(categorias_production)

O resultado irá mostrar os resultados de cada versão. Veja como a versão de Staging prediz corretamente a categoria sem acentos. Vamos trocar as versões para ver como é fácil? Na interface do MLflow, troque as versões de Staging para Production e vice-versa. Cuidado para não arquivar automaticamente as versões ao fazer as alterações (será exibida uma opção para fazer isso, desmarque). Caso tenha arquivado, porém, basta resgatá-las e reativá-las com as transições desejadas.

Re-execute o script e veja que as mudanças se refletiram sem a necessidade de mudar o código. Ou seja, os usuários dos modelos não precisam sempre ficar sabendo das escolhas dos melhores modelos, podem apenas utilizar a versão marcada como Production ou Staging, confiando nos desenvolvedores dos modelos.

10.4.7 Utiliando aliases (substitui as versões Staging e Production)

Ao invés de utilizar os estágios predefinidos pelo MLFlow (Nenhum, Staging, Production e Archived), atualmente é possível definir seus próprios rótulos para cada modelo. Além de trazer a possibilidade de adicionar mais estágios, caso necessário, é possível dar mais do que um rótulo para uma mesma versão de modelo. Obviamente, não é possível que um mesmo rótulo seja atribuído a mais de uma versão, já que o objetivo é oferecer uma forma mais amigável de se encontrar uma versão de um modelo, dentre as várias registradas.

Para ilustrar, acesse a interface do MLFlow e habilite a opção "New model registry UI", caso ainda não esteja habilitada.

Em seguida, crie alguns rótulos para as duas versões dos modelos. Neste exemplo, utilizaremos testes e produção para marcar dois estágios diferentes de evolução de um modelo, mas também utilizaremos rótulos para marcar o que quisermos, no caso treinamento-com-acentos e treinamento-sem-acentos, para indicar a diferença entre essas versões:

Agora volte ao código batch.py para mudar a forma como os modelos são encontrados, utilizando os rótulos recém-criados. Note como o padrão da URL muda, ao invés de /, utilizamos @ para separar o nome da versão:

import mlflow.pyfunc
import os

model_name = "classificador-produtos"
os.environ['MLFLOW_TRACKING_URI']='http://localhost:5000'


staging_model = mlflow.pyfunc.load_model(
-    model_uri=f"models:/{model_name}/Staging"
+    model_uri=f"models:/{model_name}@testes"
)
production_model = mlflow.pyfunc.load_model(
-    model_uri=f"models:/{model_name}/Production"
+    model_uri=f"models:/{model_name}@producao"
)
+treinamento_com_acentos_model = mlflow.pyfunc.load_model(
+    model_uri=f"models:/{model_name}@treinamento-com-acentos"
+)
+treinamento_sem_acentos_model = mlflow.pyfunc.load_model(
+    model_uri=f"models:/{model_name}@treinamento-sem-acentos"
+)

data = ["boneca de pano", "halo", "senhor dos anéis", "senhor dos aneis"]

categorias_staging = staging_model.predict(data)
categorias_production = production_model.predict(data)
+categorias_com_acentos = treinamento_com_acentos_model.predict(data)
+categorias_sem_acentos = treinamento_sem_acentos_model.predict(data)

print("Modelo Staging: ")
print(categorias_staging)
print("Modelo Production: ")
print(categorias_production)
+print("Modelo treinado com acentos: ")
+print(categorias_com_acentos)
+print("Modelo treinado sem acentos: ")
+print(categorias_sem_acentos)

Veja como é possível recuperar a mesma versão utilizando dois rótulos diferentes.

Esse exemplo não segue exatamente a recomendação da documentação do MLFlow, que sugere o registro de modelos diferentes para produção e testes, e não apenas diferentes versões. Dessa forma, é possível ganhar ainda mais variedade no versionamento.

Por exemplo, ao invés de termos um modelo e duas versões para produção e testes:

  • models:/classificador-produtos/Staging

  • models:/classificador-produtos/Production

Sugere-se dois modelos com NOMES diferentes (ambientes diferentes, versões de códigos diferentes, etc), e rótulos para indicar um grau a mais de refinamento:

  • models:/prod.classificador-produtos@champion

  • models:/stag.classificador-produtos@treinado-sem-acentos

Definir uma política consistente de nomes e rótulos traz grande flexibilidade ao ciclo de vida MLOps, facilitando, por exemplo, a nomeação de versões, testes A/B, e até mesmo o controle de acesso separado por versões, algo que não foi abordado aqui mas que está detalhado na documentação da ferramenta.

10.4.8 Suporte para IA generativa

O MLflow vem evoluindo constantemente. Uma das adições mais (relativamente) recentes é o suporte a tarefas utilizando LLMs, como o ChatGPT e o Gemini. Conforme a documentação oficial, existem várias funcionalidades referentes a esse tipo de solução. Aqui iremos demonstrar como a mesma filosofia de experimentação e tracking pode ser realizada com o apoio do MLflow para a tarefa de engenharia de prompt. Para isso, veremos duas configurações adicionais: a API Gateway para LLMs e a interface para Engenharia de Prompt. Faremos isso por meio de exemplos, modificando a infraestrutura que estamos construindo até agora.

A API Gateway para LLMs consiste em uma camada intermediária que funciona como um ponto único de acesso a diferentes soluções. Imagine que sua empresa tem diferentes planos contratados em diferentes provedores, como OpenAI, Azure e AWS. Com a API Gateway, todos podem ser acessados a partir de um único ponto, facilitando a configuração e promovendo uma melhor separação de conceitos. As aplicações passam a depender apenas dessa API, possibilitando, por exemplo, que o desenvolvedor troque de um modelo para outro sem causar impacto em seu código.

Para configurar a API Gateway do MLflow, acesse a pasta onde estivemos trabalhando com o servidor MLflow (é a pasta que tem o arquivo compose.yml). Primeiro crie um arquivo de configuração chamado config.yaml:

endpoints:
  - name: chat
    endpoint_type: llm/v1/chat
    model:
      provider: openai
      name: gpt-4o-mini
      config:
        openai_api_key: $OPENAI_API_KEY

Esse arquivo cria um endpoint, ou seja, um ponto de acesso a algum serviço de LLM. No caso, estamos configurando um ponto para o modelo gpt-4o-mini da OpenAI. Note como é necessário configurar uma chave de acesso. No caso, deixamos essa configuração para o arquivo .env, por meio da chave $OPENAI_API_KEY, mas isso pode ser feito de outras formas, a depender do ambiente de execução. Altere o arquivo .env:

AWS_ACCESS_KEY_ID=admin
AWS_SECRET_ACCESS_KEY=senhasenha
AWS_BUCKET_NAME=mlflow
AWS_REGION=us-east-1
MYSQL_DATABASE=mlflow
MYSQL_USER=mlflow_user
MYSQL_PASSWORD=mlflow_password
MYSQL_ROOT_PASSWORD=senhasenha
+OPENAI_API_KEY=<cole aqui sua chave de acesso da OpenAI>

Detalhes de como obter uma chave de acesso para a OpenAI podem ser encontrados na documentação oficial.

Crie agora o arquivo Dockerfile-AI-gateway, com o seguinte conteúdo:

FROM python

RUN pip install mlflow[genai]

ADD . /app
WORKDIR /app

COPY config.yaml .

Esse arquivo será utilizado para criar uma imagem com base na instalação padrão do MLflow, com suporte às IAs generativas. Note como ela copia o arquivo config.yaml para que o mesmo possa ser acessado ao se iniciar o servidor.

Sim, precisaremos subir uma nova instância do MLflow. Já temos uma rodando, que está servindo para registrar os experimentos e modelos. Ela acessa o banco de dados (MySQL) e o serviço S3 (MinIO) para armazenar seus dados e metadados. Agora precisaremos de outra, cujo propósito é servir como API gateway, unificando o acesso aos endpoints configurados.

Para isso, basta modificar o arquivo compose.yml:

services:
  minio:
    image: minio/minio
    restart: unless-stopped
    ports:
      - "9000:9000"
      - "9001:9001"
    expose:
      - 53
    environment:
      - MINIO_ROOT_USER=${AWS_ACCESS_KEY_ID}
      - MINIO_ROOT_PASSWORD=${AWS_SECRET_ACCESS_KEY}
    healthcheck:
      test: timeout 5s bash -c ':> /dev/tcp/127.0.0.1/9000' || exit 1
      interval: 1s
      timeout: 10s
      retries: 5
    command: server /data --console-address ":9001"
    networks:
      - internal
      - public
    volumes:
      - minio_volume:/data
  create_s3_buckets:
    image: minio/mc
    depends_on:
      minio:
        condition: service_healthy
    entrypoint: >
      bash -c "
      mc alias set minio http://minio:9000 '${AWS_ACCESS_KEY_ID}' '${AWS_SECRET_ACCESS_KEY}' &&
      mc mb minio/${AWS_BUCKET_NAME}
      "
    networks:
      - internal
  db:
    image: mysql
    restart: unless-stopped
    container_name: mlflow_db
    expose:
      - "3306"
    environment:
      - MYSQL_DATABASE=${MYSQL_DATABASE}
      - MYSQL_USER=${MYSQL_USER}
      - MYSQL_PASSWORD=${MYSQL_PASSWORD}
      - MYSQL_ROOT_PASSWORD=${MYSQL_ROOT_PASSWORD}
    volumes:
      - db_volume:/var/lib/mysql
    networks:
      - internal
    healthcheck:
      test: ["CMD", "mysqladmin", "ping", "-h", "localhost", "-u", "{MYSQL_USER}", "-p{MYSQL_PASSWORD}"]
      interval: 30s
      timeout: 10s
      retries: 5
  mlflow:
    container_name: mlflow-server-container
    image: mlflow-server
    restart: unless-stopped
    depends_on:
      db:
        condition: service_healthy
    build:
      context: .
      dockerfile: Dockerfile
    ports:
      - "5000:5000"
    environment:
      - AWS_ACCESS_KEY_ID=${AWS_ACCESS_KEY_ID}
      - AWS_SECRET_ACCESS_KEY=${AWS_SECRET_ACCESS_KEY}
      - AWS_DEFAULT_REGION=${AWS_REGION}
      - MLFLOW_S3_ENDPOINT_URL=http://minio:9000
+      - MLFLOW_DEPLOYMENTS_TARGET=http://mlflow-ai-gateway:7000
    networks:
      - public
      - internal
    entrypoint: mlflow server --backend-store-uri mysql+pymysql://${MYSQL_USER}:${MYSQL_PASSWORD}@db:3306/${MYSQL_DATABASE} --artifacts-destination s3://${AWS_BUCKET_NAME}/ --serve-artifacts -h 0.0.0.0
+  mlflow-ai-gateway:
+    container_name: mlflow-ai-gateway-container
+    image: mlflow-ai-gateway
+    restart: unless-stopped
+    build:
+      context: .
+      dockerfile: Dockerfile-AI-gateway
+    ports:
+      - "7000:7000"
+    environment:
+      - OPENAI_API_KEY=${OPENAI_API_KEY}
+    networks:
+      - public
+      - internal
+    entrypoint: mlflow deployments start-server --config-path config.yaml --port 7000 --host 0.0.0.0
networks:
  internal:
  public:
    driver: bridge
volumes:
  db_volume:
  minio_volume:

Note como essa nova entrada cria uma imagem baseada no Dockerfile-AI-gateway que acabamos de criar. Ela também abre a porta 7000 para ser utilizada no host. Também define as variáveis de ambiente que serão lidas pelos servidores, o que inclui o novo que estamos criando mas também o atual, para que o tracking funcione com a nova opção. Por último, no entrypoint, essa entrada sobe uma instância do MLflow que servirá como API gateway.

Para testar, execute o comando docker compose up, abra um navegador e entre no endereço http://localhost:7000, e veja como é possível acessar os serviços configurados e fazer requisições ao ChatGPT. Por exemplo, tente enviar o seguinte conteúdo ao endpoint chat/invocations:

{
  "temperature": 0,
  "n": 1,
  "max_tokens": 25,
  "messages": [
    {
      "role": "user",
      "content": "Faça uma piada sobre LLMs"
    }
  ]
}

Uma vez que a API gateway esteja configurada, é possível acessar uma nova área no MLflow. Crie um novo experimento, cujo nome é gerador-descricoes, e depois clique em "New run", conforme a imagem a seguir.

Em seguida, escolha a opção "using Prompt Engineering":

Na tela que se abrir, escolha o modelo "chat". Foi o modelo que criamos no arquivo config.yaml agora há pouco. Se tiver escolhido outro nome, ele aparecerá aqui. Em seguida escolha os parâmetros do modelo, como temperatura e máximo de tokens. Escolha também um nome para esta execução (run), se quiser.

No lado direito será exibida uma área para a criação de um template de prompt. É possível criar variáveis para customizar o prompt. Siga o exemplo da figura, clique em "Evaluate" e veja o resultado. Aproveite para experimentar com outros prompts. Quando estiver satisfeito, clique em "Create run".

Será criada uma nova execução (run), cujos detalhes podem ser vistos clicando sobre "Evaluation". Também será exibida uma tabela com os valores testados. É possível acrescentar outros valores para experimentar com o resultado do prompt. Veja na imagem a seguir essas opções. Depois de acrescentar valores, não esqueça de avaliar todos (botão "Evaluate all") e depois salvar.

Caso queira experimentar e modificar o prompt, é preciso criar uma nova execução. Clique em "duplicate run":

Depois modifique os parâmetros que desejar, incluindo o nome da run, se quiser, e clique em "Create run" novamente.

Ao retornar à tela anterior, veja como será acrescentada uma nova coluna com os novos parâmetros. Clique em "Evaluate all" e salve tudo novamente para deixar registrado.

Assim como os modelos de IA convencionais, também é possível registrar os modelos de IA generativa no MLflow. Clique em uma das runs (por exemplo, "Primeiro teste (1)") e em seguida "Register model":

Escolha "Create new model" e dê um nome, por exemplo, "gerador-descricoes". É possível ver o modelo registrado na área "Models" no topo da página. E assim como vimos anteriormente, é possível atribuir um apelido (alias) para facilitar seu acesso por meio do nome, possibilitando a troca de versões diretamente no MLflow. Acrescente o apelido "producao" a este modelo que gera um slogan, e o apelido "testes" ao modelo que não gera slogan.

Assim como outros modelos do MLflow, é possível recuperar e utilizar este modelo programaticamente. Para isso, vamos criar um novo ambiente virtual Python e instalar o mlflow com suporte a GenAI. Crie uma pasta chamada ambiente-mlflow-genai e execute os seguintes comandos (use Linux ou WSL):

python -m venv .venv
source .venv/bin/activate
pip install mlflow[gateway]

Agora crie um arquivo teste.py:

import mlflow.models
import mlflow.pyfunc
import os

model_name = "gerador-descricoes"
os.environ['MLFLOW_TRACKING_URI']='http://localhost:5000'
os.environ['MLFLOW_GATEWAY_URI']='http://localhost:7000'

model = mlflow.pyfunc.load_model(
    model_uri=f"models:/{model_name}@producao"
)

result = model.predict({"titulo_produto":"videogame", "publico_alvo": "adolescente"})

print(result)

Este código deveria, em tese, recuperar o modelo pelo nome e executar uma predição, ou seja, gerar uma descrição para um produto. No entanto, no momento da escrita deste exemplo, existe um bug ainda não resolvido e que impede que esse exemplo seja bem sucedido.

O que ele deveria fazer:

  1. Recuperar os metadados do modelo com base em seu nome

  2. Ler os parâmetros do modelo, o que inclui detalhes como temperatura, tokens, o template de prompt e suas variáveis

  3. Substituir as variáveis pelos valores especificados

  4. Enviar a requisição ao API Gateway (o bug está aqui)

Sabendo desse passo-a-passo, e enquanto o bug não é resolvido, podemos replicar esse comportamento manualmente e contornar o bug. Crie o arquivo testeContornandoBug.py:

import mlflow.models
import mlflow.pyfunc
import os
import re

import mlflow.runs
from mlflow.deployments import get_deploy_client

def replace(template: str, replacements: dict) -> str:
    def replacer(match):
        key = match.group(1)
        return replacements.get(key, match.group(0))
    return re.sub(r"\{\{(\w+)\}\}", replacer, template)

def custom_predict(model: mlflow.pyfunc.PyFuncModel, variables: dict):
    run_info = mlflow.get_run(model.metadata.run_id)
    params = run_info.to_dictionary()["data"]["params"]

    max_tokens = params["max_tokens"]
    model_route = params["model_route"]
    prompt_template = params["prompt_template"]
    temperature = params["temperature"]

    prompt = replace(prompt_template, variables)

    response = client.predict(
        endpoint=model_route,
        inputs={
            "temperature": temperature,
            "max_tokens": max_tokens,
            "messages": [{"role": "user", "content": prompt}]},
    )

    return response["choices"][0]["message"]["content"]

model_name = "gerador-descricoes"
client = get_deploy_client("http://localhost:7000")
os.environ['MLFLOW_TRACKING_URI']='http://localhost:5000'

model = mlflow.pyfunc.load_model(f"models:/{model_name}@producao")

result = custom_predict(model=model, variables={
        "titulo_produto": "Smartphone",
        "publico_alvo": "jovem adulto"
    })

print(result)

Note como o novo código reproduz os passos anteriormente definidos. Primeiro ele recupera o modelo marcado com o apelido desejado. Depois inspeciona os detalhes do modelo em busca do run_id, que é o identificador da execução que gerou o modelo. Em seguida, ele recupera os detalhes da execução, incluindo os parâmetros especificados. Depois ele substitui as variáveis no template de prompt. Por fim, ele utiliza a API gateway, por meio de um cliente, para fazer a requisição, seguindo o formato de entrada do endpoint.

Experimente mudar, no MLflow, os apelidos definidos para as diferentes versões dos modelos, e execute novamente o código acima. Veja como o novo resultado aparece sem a necessidade de se alterar o código. Esse processo facilita muito os testes e atualização dos modelos.

Além desse suporte à engenharia de prompt, o MLflow traz outras features relacionadas à IA generativa e LLMs. Acesse a documentação oficial para aprender mais.

10.4.9 Considerações finais

O MLflow é uma ferramenta que pode ser muito útil para diferentes cenários, desde pequenas equipes até grandes empresas. Não é uma ferramenta que resolve todos os problemas de MLOps, mas ela ajuda em diferentes momentos. Além de ser de fácil instalação e configuração, o MLflow pode ser utilizado em partes. Caso queira usar somente o componente de rastreamento de experimentos, é possível. Caso queira usar o componente de projetos, também é possível. O registro de modelos também facilita bastante a vida quando a quantidade de modelos e versões é muito grande.

Há muitas outras funções e possibilidades de configuração e uso do MLflow. A documentação oficial é bastante rica e cheia de exemplos. Vale a pena estudar!

Last updated