Skip to content

29. Exercício prático: versão 11

29.1. Introdução

Nas versões anteriores da aplicação cliente/servidor de cálculo de impostos, a camada [métier], que implementa as regras de negócio deste cálculo, estava do lado do servidor. Propõe-se agora transferi-la para o lado do cliente. Qual é a vantagem? Parte do trabalho que era realizado pelo servidor será transferido para o lado do cliente. Se pensarmos na situação de um servidor consultado por N clientes, N cálculos de imposto serão efetuados pelos clientes. Nas versões anteriores, o servidor realizava esses N cálculos de imposto. Como já não realiza o cálculo de imposto, o servidor irá responder mais rapidamente aos seus clientes e poderá, assim, servir um maior número de clientes simultaneamente.

A arquitetura cliente/servidor passa a ser a seguinte:

Image

  • a camada [métier] [10] foi duplicada como [12] no cliente;
  • foi adicionado um novo script [main2] [11] ao cliente;

O cliente web terá duas formas de calcular o imposto da lista de contribuintes encontrada em [3]:

  • utilizar o método da versão anterior. Este utiliza a camada [métier] [10] do servidor. O script [main] utilizará este método;
  • limitar-se a solicitar ao servidor os dados da administração fiscal [2-4] e, em seguida, utilizar a camada [métier] [12] local do cliente;

Iremos comparar o desempenho dos dois métodos.

29.2. O servidor web

A estrutura do servidor web será a seguinte:

Image

  • O ficheiro [http-servers/06] é inicialmente obtido através da cópia do ficheiro [http-servers/05]. De facto, vamos manter o que já existe na versão 10 anterior. Vamos simplesmente adicionar-lhe uma nova funcionalidade. Esta traduz-se na presença de um novo controlador [get_admindata_controller] [1]. O outro controlador, [calculate_tax_controller], não é outro senão o antigo controlador [index_controller], que foi renomeado;

29.3. Configuração

O servidor disponibilizará dois URL de serviço:

  • [/calculate-tax] para calcular o imposto de uma lista de contribuintes passada no corpo de um POST. Corresponde, portanto, ao URL [/] da versão 10 anterior;
  • A [/get-admindata] fornece a cadeia jSON com os dados da administração fiscal;

A configuração [config] associa cada um destes URL ao controlador que o processa:


    # dicionário dos controladores da aplicação web
    import calculate_tax_controller, get_admindata_controller
    controllers = {"calculate-tax": calculate_tax_controller, "get-admindata": get_admindata_controller}
    config['controllers'] = controllers

29.4. O script principal [main]

O script principal [main] reestrutura o script [main] da versão anterior:


# aguarda-se um parâmetro mysql ou pgres
import sys
syntaxe = f"{sys.argv[0]} mysql / pgres"
erreur = len(sys.argv) != 2
if not erreur:
    sgbd = sys.argv[1].lower()
    erreur = sgbd != "mysql" and sgbd != "pgres"
if erreur:
    print(f"syntaxe : {syntaxe}")
    sys.exit()

# configura-se a aplicação
import config
config = config.configure({'sgbd': sgbd})

# dependências


# recuperação de dados da administração fiscal
erreur = False
try:
    # «admindata» será um conjunto de dados de âmbito da aplicação, apenas para leitura
    config["admindata"] = config["layers"]["dao"].get_admindata()
    # registo de sucesso
    logger.write("[serveur] connexion à la base de données réussie\n")
except ImpôtsError as ex:
    # regista-se o erro
    …

# se tiver ocorrido um erro, a aplicação é interrompida
if erreur:
    sys.exit(2)

# a aplicação Flask pode iniciar
app = Flask(__name__)

# controlador principal
def main_controller() -> tuple:
    # recuperamos a ação solicitada
    dummy, action=request.path.split('/')
    logger = None
    try:
        # registo
        logger = Logger(config["logsFilename"])
        # armazenamos-a numa configuração associada ao thread
        thread_config = {"logger": logger}
        thread_name = threading.current_thread().name
        config[thread_name] = {"config": thread_config}
        # regista-se o pedido
        logger.write(f"[index] requête : {request}\n")
        # interrompe-se o thread, caso tal tenha sido solicitado
        sleep_time = config["sleep_time"]
        if sleep_time != 0:
            # a pausa é aleatória, para que algumas threads sejam interrompidas e outras não
            aléa = randint(01)
            if aléa == 1:
                # registo antes da pausa
                logger.write(f"[index] mis en pause du thread pendant {sleep_time} seconde(s)\n")
                # pausa
                time.sleep(sleep_time)
        # a solicitação é executada pelo controlador de solicitações
        controller = config['controllers'][action]
        résultat, status_code = controller.execute(request, config)
        # ocorreu algum erro fatal?
        if status_code == status.HTTP_500_INTERNAL_SERVER_ERROR:
            # envia-se um e-mail ao administrador da aplicação
            config_mail = config["adminMail"]
            config_mail["logger"] = logger
            SendAdminMail.send(config_mail, json.dumps(résultat, ensure_ascii=False))
        # regista-se a resposta
        logger.write(f"[index] {résultat}\n")
        # envia-se a resposta
        print(résultat)
        return json_response(résultat, status_code)
    except BaseException as erreur:
        # regista-se o erro, se possível
        if logger:
            logger.write(f"[index] {erreur}")
        # prepara-se a resposta para o cliente
        résultat = {"réponse": {"erreurs"[f"{erreur}"]}}
        # envia-se a resposta
        return json_response(résultat, status.HTTP_500_INTERNAL_SERVER_ERROR)
    finally:
        # fecha-se o ficheiro de registos, caso tenha sido aberto
        if logger:
            logger.close()

# cálculo do imposto
@app.route('/calculate-tax', methods=['POST'])
@auth.login_required
def calculate_tax():
    # passa o controlo ao controlador principal
    return main_controller()

# obter os dados da administração fiscal
@app.route('/get-admindata', methods=['GET'])
@auth.login_required
def get_admindata():
    # passa-se o controlo ao controlador principal
    return main_controller()

# apenas controlo
if __name__ == '__main__':
    # iniciar o servidor
    app.config.update(ENV="development", DEBUG=True)
    app.run(threaded=True)
  • linhas 88-93: a função [calculate_tax] processa o URL e o [/calculate-tax];
  • linhas 95-100: a função [get_admindata] processa as funções URL e [/get-admindata];
  • estas duas funções não fazem nada por si próprias. Passam imediatamente o controlo para o controlador principal [main_controller] das linhas 37-86;
  • linhas 37-86: o controlador principal [main_controller] não é mais do que a função [index] da versão anterior, com uma única diferença: enquanto a função [index] processava apenas um único URL, aqui a função [main_controller] processa dois URL. Por isso, é necessário que estes sejam processados por um dos dois controladores [calculate_tax_controller, get_admin,data_controller];
  • linhas 39-40: recupera-se a ação solicitada [calculate_tax] ou [get_admindata]. Esta informação encontra-se no caminho do URL [request.path]. Consoante o caso, [request.path] corresponde a [/get-admindata] ou a [/calculate_tax]. A divisão da linha 40 resultará em dois elementos:
    • a cadeia vazia para a parte que precede o /;
    • o nome da ação solicitada para a parte que se segue ao /;
  • linhas 62-63: assim que a ação do URL for recuperada, sabemos qual o controlador a utilizar para processar o URL. Esta informação encontra-se na configuração [config];

29.5. Os controladores

O controlador [calculate_tax_controller] não é outro senão o controlador [index_controller] da versão anterior.

O controlador [get_admindata_controller] é o seguinte:


from flask_api import status
from werkzeug.local import LocalProxy

def execute(request: LocalProxy, config: dict) -> tuple:
    # envia-se a resposta
    return {"réponse": {"result": config["admindata"].asdict()}}, status.HTTP_200_OK
  • O URL [/get-admindata] deve gerar a cadeia jSON a partir dos dados da administração fiscal;
  • linha 6: estes foram recuperados pelo script principal [main] e colocados no dicionário [config] sob a forma de um objeto [AdminData]. Devolvemos o dicionário deste objeto;

29.6. Testes no Postman

Iniciamos o servidor web, o SGBD e o servidor de e-mail [hMailServer]. Em seguida, utilizando um cliente Postman, calculamos o imposto de vários contribuintes:

Image

Na consola do Postman, o diálogo cliente/servidor é o seguinte:


POST /calculate-tax HTTP/1.1
Authorization: Basic YWRtaW46YWRtaW4=
Content-Type: application/json
User-Agent: PostmanRuntime/7.26.2
Accept: */*
Cache-Control: no-cache
Postman-Token: 5e71461a-fec8-4315-85e8-41721de939e5
Host: localhost:5000
Accept-Encoding: gzip, deflate, br
Connection: keep-alive
Content-Length: 824

[ 
{ 
"marié": "oui", 
"enfants": 2, 
"salaire": 55555 
}, 

{ 
"marié": "oui", 
"enfants": 3, 
"salaire": 200000 
} 
]

HTTP/1.0 200 OK
Content-Type: application/json; charset=utf-8
Content-Length: 1461
Server: Werkzeug/1.0.1 Python/3.8.1
Date: Wed, 29 Jul 2020 07:02:07 GMT

{"réponse": {"results": [{"marié": "oui", "enfants": 2, "salaire": 55555, "impôt": 2814, "surcôte": 0, "taux": 0.14, "décôte": 0, "réduction": 0}, {"marié": "oui", "enfants": 2, "salaire": 50000, "impôt": 1384, "surcôte": 0, "taux": 0.14, "décôte": 384, "réduction": 347…]}}

Agora, vamos solicitar o URL e o [/get-admindata], juntamente com um GET:

Image

O diálogo cliente/servidor na consola do Postman é o seguinte:


GET /get-admindata HTTP/1.1
Authorization: Basic YWRtaW46YWRtaW4=
User-Agent: PostmanRuntime/7.26.2
Accept: */*
Cache-Control: no-cache
Postman-Token: 4af342c4-7ecb-4ab2-9e12-d653f81da424
Host: localhost:5000
Accept-Encoding: gzip, deflate, br
Connection: keep-alive

HTTP/1.0 200 OK
Content-Type: application/json; charset=utf-8
Content-Length: 596
Server: Werkzeug/1.0.1 Python/3.8.1
Date: Wed, 29 Jul 2020 07:07:24 GMT

{"réponse": {"result": {"limites": [9964.0, 27519.0, 73779.0, 156244.0, 93749.0], "coeffr": [0.0, 0.14, 0.3, 0.41, 0.45], "coeffn": [0.0, 1394.96, 5798.0, 13913.7, 20163.4], "plafond_decote_couple": 1970.0, "valeur_reduc_demi_part": 3797.0, "plafond_revenus_celibataire_pour_reduction": 21037.0, "plafond_qf_demi_part": 1551.0, "abattement_dixpourcent_max": 12502.0, "plafond_impot_celibataire_pour_decote": 1595.0, "plafond_decote_celibataire": 1196.0, "plafond_revenus_couple_pour_reduction": 42074.0, "id": 1, "abattement_dixpourcent_min": 437.0, "plafond_impot_couple_pour_decote": 2627.0}}}

29.7. O cliente web

Image

Image

A pasta [http-clients/06] é inicialmente obtida por cópia da pasta [http-clients/05]. O trabalho de modificação consiste essencialmente em:

  • modificar a configuração [config_layers] para que passe a incluir uma camada [métier]. Anteriormente, tinha apenas uma camada [dao];
  • adicionar um novo método à camada [dao];
  • escrever um script [main2] que se baseie na camada [métier] do cliente para calcular o imposto dos contribuintes;

29.7.1. Configuração das camadas do cliente

A configuração das camadas ocorre em dois pontos:

  • na configuração [config], que deve incluir nas dependências do cliente a pasta que contém a implementação da camada [métier]. Esta pasta já estava incluída nas dependências:

absolute_dependencies = [
        # pastas do projeto
        # BaseEntity, MyException
        f"{root_dir}/classes/02/entities",
        # InterfaceImpôtsDao, InterfaceImpôtsMétier, InterfaceImpôtsUi
        f"{root_dir}/impots/v04/interfaces",
        # AbstractImpôtsdao, ImpôtsConsole, ImpôtsMétier
        f"{root_dir}/impots/v04/services",
        # ImpotsDaoWithAdminDataInDatabase
        f"{root_dir}/impots/v05/services",
        # AdminData, ImpôtsError, TaxPayer
        f"{root_dir}/impots/v04/entities",
        # Constantes, intervalos
        f"{root_dir}/impots/v05/entities",
        # ImpôtsDaoWithHttpClient
        f"{script_dir}/../services",
        # scripts de configuração
        script_dir,
        # Registador
        f"{root_dir}/impots/http-servers/02/utilities",
    ]

Em seguida, o ficheiro [config_layers] deve ser alterado:


def configure(config: dict) -> dict:
    # instanciação das camadas da aplicação

    # camada [métier]
    from ImpôtsMétier import ImpôtsMétier
    métier = ImpôtsMétier()

    # camada DAO
    from ImpôtsDaoWithHttpClient import ImpôtsDaoWithHttpClient
    dao = ImpôtsDaoWithHttpClient(config)

    # efetua-se a configuração das camadas
    return {
        "dao": dao,
        "métier": métier
    }
  • linhas 4-6: instanciação da camada [métier];
  • linhas 13-16: a camada [métier] é incluída no dicionário de camadas;

29.7.2. Implementação da camada [dao]

Image

A camada [dao] apresentará a seguinte interface [InterfaceImpôtsDaoWithHttpClient]:


from abc import abstractmethod

from AbstractImpôtsDao import AbstractImpôtsDao

class InterfaceImpôtsDaoWithHttpClient(AbstractImpôtsDao):

    # cálculo do imposto
    @abstractmethod
    def calculate_tax_in_bulk_mode(self, taxpayers: list):
        pass
  • linha 5: a interface [InterfaceImpôtsDaoWithHttpClient] herda da classe abstrata [AbstractImpôtsDao], que gere o acesso ao sistema de ficheiros do cliente. Recorde-se que esta possui um método abstrato [get_admindata];
  • linhas 7-10: o método [calculate_tax_in_bulk_mode] que definimos na versão anterior permite o cálculo do imposto de uma lista de contribuintes;

Esta interface é implementada pela seguinte classe [ImpôtsDaoWithHttpClient]:


# importações

import json

import requests
from flask_api import status

from AbstractImpôtsDao import AbstractImpôtsDao
from AdminData import AdminData
from ImpôtsError import ImpôtsError
from InterfaceImpôtsDaoWithHttpClient import InterfaceImpôtsDaoWithHttpClient

class ImpôtsDaoWithHttpClient(InterfaceImpôtsDaoWithHttpClient):

    # construtor
    def __init__(self, config: dict):
        # inicialização do pai
        AbstractImpôtsDao.__init__(self, config)
        # armazenamento dos elementos de configuração
        # configuração geral
        self.__config = config
        # servidor
        self.__config_server = config["server"]
        # modo de depuração
        self.__debug = config["debug"]
        # registo
        self.__logger = None

    def get_admindata(self) -> AdminData:
        # permitir que as exceções sejam reportadas

        # URL do serviço
        config_server = self.__config_server
        config_services = config_server['url_services']
        url_service = f"{config_server['urlServer']}{config_services['get-admindata']}"

        # ligação
        if config_server['authBasic']:
            response = requests.get(url_service,
                                    auth=(
                                        config_server["user"]["login"],
                                        config_server["user"]["password"]))
        else:
            response = requests.get(url_service)

        # modo de depuração?
        if self.__debug:
            # registador
            if not self.__logger:
                self.__logger = self.__config['logger']
            # registo
            self.__logger.write(f"{response.text}\n")

        # código de estado
        status_code = response.status_code
        # resultado na forma de um dicionário
        résultat = json.loads(response.text)
        # se o código de estado for diferente de 200 OK
        if status_code != status.HTTP_200_OK:
            raise ImpôtsError(58, résultat['réponse']['erreurs'])
        # retorna o resultado (um dicionário)
        return AdminData().fromdict(résultat["réponse"]["result"])

    # cálculo do imposto em modo em massa
    def calculate_tax_in_bulk_mode(self, taxpayers: list):
        # deixa-se que as exceções sejam reportadas

        
  • linha 13: a classe [ImpôtsDaoWithHttpClient] implementa a interface [InterfaceImpôtsDaoWithHttpClient]. Por conseguinte, deriva da classe [AbstractImpôtsDao];
  • linhas 65-66: o método [calculate_tax_in_bulk_mode] analisado na versão anterior;
  • linhas 29-62: o método [get_admindata] que a classe pai [AbstractImpôtsDao] declarou como abstrato. Por conseguinte, é implementado na classe filha;
  • linhas 33-35: determina-se o URL do serviço web que o método [get-admindata] deve consultar. Estes URL do serviço estão definidos na configuração [config] do cliente:

       # o servidor de cálculo do imposto
        "server": {
            "urlServer""http://127.0.0.1:5000",
            "authBasic"True,
            "user": {
                "login""admin",
                "password""admin"
            },
            "url_services": {
                "calculate-tax""/calculate-tax",
                "get-admindata""/get-admindata"
            }
        },
  • (continuação)
    • linhas 9-12: os dois URL do servidor web;
  • linhas 37-44: o URL de serviço é consultado de forma síncrona;
  • linhas 46-42: se a configuração assim o exigir, a resposta do servidor é registada;
  • linha 57: sabe-se que o servidor enviou uma cadeia jSON de um dicionário;
  • linhas 58-60: se o estado HTTP da resposta não for 200, então é lançada uma exceção;
  • linhas 61-62: devolve-se o objeto [AdminData] que encapsula os dados da administração fiscal enviados pelo servidor;

29.8. Os scripts [main, main2]

O script [main] é o da versão anterior. Utiliza o método [calculate_tax_in_bulk_mode] da camada [dao] e, por conseguinte, utiliza a camada [métier] do servidor;

O script [main2] faz o mesmo que o script [main], mas utilizando a camada [métier] do cliente:


# configuração da aplicação

import config
config = config.configure({})

# dependências
from ImpôtsError import ImpôtsError
from Logger import Logger

logger = None
# código
try:
    # registo
    logger = Logger(config["logsFilename"])
    # armazená-lo na configuração
    config["logger"] = logger
    # registo de início
    logger.write("début du calcul de l'impôt des contribuables\n")
    # recuperamos a camada [dao]
    dao = config["layers"]["dao"]
    # recuperam-se os contribuintes
    taxpayers = dao.get_taxpayers_data()["taxpayers"]
    # dos contribuintes?
    if not taxpayers:
        raise ImpôtsError(36f"Pas de contribuables valides dans le fichier {config['taxpayersFilename']}")
    # recuperam-se os dados da administração fiscal
    admindata = dao.get_admindata()
    # cálculo do imposto dos contribuintes pela camada [métier]
    métier = config['layers']['métier']
    for taxpayer in taxpayers:
        métier.calculate_tax(taxpayer, admindata)
    # os resultados são gravados no ficheiro jSON
    dao.write_taxpayers_results(taxpayers)
# exceto BaseException como erro:
# # exibição do erro
# print(f"Ocorreu o seguinte erro: {erro}")
finally:
    # encerramos o registador
    if logger:
        # registo de fim
        logger.write("fin du calcul de l'impôt des contribuables\n")
        # encerramento do registador
        logger.close()
    # concluído
    print("Travail terminé...")
  • linhas 26-27: recuperam-se os dados da administração fiscal a partir do servidor;
  • linhas 28-31: em seguida, o cálculo do imposto dos contribuintes é efetuado localmente;

29.9. Testes do cliente

Em cada um dos scripts [main, main2], registamos o início e o fim do script. Desta forma, poderemos calcular o tempo de execução do script. Vamos fazer algumas previsões:

  • o script [main] da versão anterior:
    • cria N threads que são executadas simultaneamente;
    • cada thread processa um lote de contribuintes, cujo imposto é calculado através de uma única consulta ao servidor;
    • como as N threads são executadas simultaneamente, a consulta N+1 é lançada antes de a consulta N ter recebido a sua resposta. Assim, as N consultas têm um custo superior ao de uma única consulta, mas provavelmente não muito superior. Além disso, há 11 (o número de contribuintes) cálculos de negócio no servidor;
  • o script [main2] desta versão:
    • efetua uma única consulta ao servidor;
    • realiza 11 cálculos de negócio localmente no cliente;

Os cálculos de negócio terão a mesma duração, quer sejam realizados no servidor ou no cliente. A diferença irá, portanto, residir nas consultas. É de esperar, assim, que o tempo de execução do [main] seja ligeiramente superior ao do [main2].

Iniciamos o servidor da versão 11, o SGBD e o servidor de e-mail [hMailServer]. No lado do servidor, definimos o parâmetro [sleep_time] como zero para que os dois testes sejam executados nas mesmas condições.

Execução 1 [main]

A execução do [main] gera os seguintes registos:


2020-07-29 14:35:50.016079, MainThread : début du calcul de l'impôt des contribuables
2020-07-29 14:35:50.016079, Thread-1 : début du calcul de l'impôt des 1 contribuables
2020-07-29 14:35:50.016079, Thread-2 : début du calcul de l'impôt des 4 contribuables
2020-07-29 14:35:50.016079, Thread-3 : début du calcul de l'impôt des 2 contribuables
2020-07-29 14:35:50.016079, Thread-4 : début du calcul de l'impôt des 2 contribuables
2020-07-29 14:35:50.024426, Thread-5 : début du calcul de l'impôt des 2 contribuables
2020-07-29 14:35:50.050473, Thread-1 : {"réponse": {"results": [{"marié": "oui", "enfants": 2, "salaire": 55555, "impôt": 2814, "surcôte": 0, "taux": 0.14, "décôte": 0, "réduction": 0}]}}
2020-07-29 14:35:50.050473, Thread-1 : fin du calcul de l'impôt des 1 contribuables
2020-07-29 14:35:50.050473, Thread-3 : {"réponse": {"results": [{"marié": "oui", "enfants": 3, "salaire": 100000, "impôt": 9200, "surcôte": 2180, "taux": 0.3, "décôte": 0, "réduction": 0}, {"marié": "oui", "enfants": 5, "salaire": 100000, "impôt": 4230, "surcôte": 0, "taux": 0.14, "décôte": 0, "réduction": 0}]}}
2020-07-29 14:35:50.051214, Thread-3 : fin du calcul de l'impôt des 2 contribuables
2020-07-29 14:35:50.051214, Thread-5 : {"réponse": {"results": [{"marié": "non", "enfants": 0, "salaire": 200000, "impôt": 64210, "surcôte": 7498, "taux": 0.45, "décôte": 0, "réduction": 0}, {"marié": "oui", "enfants": 3, "salaire": 200000, "impôt": 42842, "surcôte": 17283, "taux": 0.41, "décôte": 0, "réduction": 0}]}}
2020-07-29 14:35:50.051214, Thread-5 : fin du calcul de l'impôt des 2 contribuables
2020-07-29 14:35:50.051214, Thread-2 : {"réponse": {"results": [{"marié": "oui", "enfants": 2, "salaire": 50000, "impôt": 1384, "surcôte": 0, "taux": 0.14, "décôte": 384, "réduction": 347}, {"marié": "oui", "enfants": 3, "salaire": 50000, "impôt": 0, "surcôte": 0, "taux": 0.14, "décôte": 720, "réduction": 0}, {"marié": "non", "enfants": 2, "salaire": 100000, "impôt": 19884, "surcôte": 4480, "taux": 0.41, "décôte": 0, "réduction": 0}, {"marié": "non", "enfants": 3, "salaire": 100000, "impôt": 16782, "surcôte": 7176, "taux": 0.41, "décôte": 0, "réduction": 0}]}}
2020-07-29 14:35:50.051214, Thread-2 : fin du calcul de l'impôt des 4 contribuables
2020-07-29 14:35:50.051214, Thread-4 : {"réponse": {"results": [{"marié": "non", "enfants": 0, "salaire": 100000, "impôt": 22986, "surcôte": 0, "taux": 0.41, "décôte": 0, "réduction": 0}, {"marié": "oui", "enfants": 2, "salaire": 30000, "impôt": 0, "surcôte": 0, "taux": 0.0, "décôte": 0, "réduction": 0}]}}
2020-07-29 14:35:50.051214, Thread-4 : fin du calcul de l'impôt des 2 contribuables
2020-07-29 14:35:50.051214, MainThread : fin du calcul de l'impôt des contribuables

A duração da execução foi de [051214-016079] nanossegundos (linha 17 – linha 1), ou seja, 35 milissegundos e 135 nanossegundos.

Verifica-se que, entre o primeiro pedido feito ao servidor e a última resposta recebida pelo cliente, existe o mesmo intervalo de tempo [051214-016079] (linha 15 – linha 1), ou seja, 35 milissegundos e 135 nanossegundos.

Execução 2 [main2]

A execução de [main2] produz os seguintes registos:


2020-07-29 14:41:03.303520, MainThread : début du calcul de l'impôt des contribuables
2020-07-29 14:41:03.345084, MainThread : {"réponse": {"result": {"limites": [9964.0, 27519.0, 73779.0, 156244.0, 13500.0], "coeffr": [0.0, 0.14, 0.3, 0.41, 0.45], "coeffn": [0.0, 1394.96, 5798.0, 13913.7, 20163.4], "plafond_decote_couple": 1970.0, "valeur_reduc_demi_part": 3797.0, "plafond_revenus_celibataire_pour_reduction": 21037.0, "plafond_qf_demi_part": 1551.0, "abattement_dixpourcent_max": 12502.0, "plafond_impot_celibataire_pour_decote": 1595.0, "plafond_decote_celibataire": 1196.0, "plafond_revenus_couple_pour_reduction": 42074.0, "id": 1, "abattement_dixpourcent_min": 437.0, "plafond_impot_couple_pour_decote": 2627.0}}}
2020-07-29 14:41:03.349975, MainThread : fin du calcul de l'impôt des contribuables

A duração da execução foi de [349975-303520] nanossegundos (linha 3 – linha 1), ou seja, 46 milissegundos e 455 nanossegundos. De forma totalmente inesperada, o [main] é mais rápido do que o [main2].

Verifica-se que a única consulta de [main2] demorou [345084-303520] (linha 2 – linha 1), ou seja, 41 milissegundos e 564 nanossegundos. O cálculo do imposto demorou, em seguida, [349975-345084] (linha 3 – linha 2), ou seja, 4 milissegundos e 91 nanossegundos. É a consulta HTTP que determina o tempo de execução. Surpreendentemente, verifica-se aqui que a única consulta de [main2] demorou mais tempo ([41 millisecondes]) do que as quatro consultas simultâneas de [main] e [35 millisecondes].

Do lado do servidor, os registos são os seguintes:


2020-07-29 14:35:27.047721, MainThread : [serveur] démarrage du serveur
2020-07-29 14:35:27.140927, MainThread : [serveur] connexion à la base de données réussie
2020-07-29 14:35:28.790716, MainThread : [serveur] démarrage du serveur
2020-07-29 14:35:28.847518, MainThread : [serveur] connexion à la base de données réussie
2020-07-29 14:35:50.039178, Thread-2 : [index] requête : <Request 'http://127.0.0.1:5000/calculate-tax' [POST]>
2020-07-29 14:35:50.039178, Thread-3 : [index] requête : <Request 'http://127.0.0.1:5000/calculate-tax' [POST]>
2020-07-29 14:35:50.043220, Thread-4 : [index] requête : <Request 'http://127.0.0.1:5000/calculate-tax' [POST]>
2020-07-29 14:35:50.044307, Thread-5 : [index] requête : <Request 'http://127.0.0.1:5000/calculate-tax' [POST]>
2020-07-29 14:35:50.045796, Thread-2 : [index] {'réponse': {'results': [{'marié': 'oui', 'enfants': 3, 'salaire': 100000, 'impôt': 9200, 'surcôte': 2180, 'taux': 0.3, 'décôte': 0, 'réduction': 0}, {'marié': 'oui', 'enfants': 5, 'salaire': 100000, 'impôt': 4230, 'surcôte': 0, 'taux': 0.14, 'décôte': 0, 'réduction': 0}]}}
2020-07-29 14:35:50.045796, Thread-3 : [index] {'réponse': {'results': [{'marié': 'oui', 'enfants': 2, 'salaire': 55555, 'impôt': 2814, 'surcôte': 0, 'taux': 0.14, 'décôte': 0, 'réduction': 0}]}}
2020-07-29 14:35:50.046825, Thread-6 : [index] requête : <Request 'http://127.0.0.1:5000/calculate-tax' [POST]>
2020-07-29 14:35:50.046825, Thread-6 : [index] {'réponse': {'results': [{'marié': 'oui', 'enfants': 2, 'salaire': 50000, 'impôt': 1384, 'surcôte': 0, 'taux': 0.14, 'décôte': 384, 'réduction': 347}, {'marié': 'oui', 'enfants': 3, 'salaire': 50000, 'impôt': 0, 'surcôte': 0, 'taux': 0.14, 'décôte': 720, 'réduction': 0}, {'marié': 'non', 'enfants': 2, 'salaire': 100000, 'impôt': 19884, 'surcôte': 4480, 'taux': 0.41, 'décôte': 0, 'réduction': 0}, {'marié': 'non', 'enfants': 3, 'salaire': 100000, 'impôt': 16782, 'surcôte': 7176, 'taux': 0.41, 'décôte': 0, 'réduction': 0}]}}
2020-07-29 14:35:50.046825, Thread-4 : [index] {'réponse': {'results': [{'marié': 'non', 'enfants': 0, 'salaire': 200000, 'impôt': 64210, 'surcôte': 7498, 'taux': 0.45, 'décôte': 0, 'réduction': 0}, {'marié': 'oui', 'enfants': 3, 'salaire': 200000, 'impôt': 42842, 'surcôte': 17283, 'taux': 0.41, 'décôte': 0, 'réduction': 0}]}}
2020-07-29 14:35:50.046825, Thread-5 : [index] {'réponse': {'results': [{'marié': 'non', 'enfants': 0, 'salaire': 100000, 'impôt': 22986, 'surcôte': 0, 'taux': 0.41, 'décôte': 0, 'réduction': 0}, {'marié': 'oui', 'enfants': 2, 'salaire': 30000, 'impôt': 0, 'surcôte': 0, 'taux': 0.0, 'décôte': 0, 'réduction': 0}]}}
2020-07-29 14:41:03.341582, Thread-7 : [index] requête : <Request 'http://127.0.0.1:5000/get-admindata' [GET]>
2020-07-29 14:41:03.341582, Thread-7 : [index] {'réponse': {'result': {'limites': [9964.0, 27519.0, 73779.0, 156244.0, 13500.0], 'coeffr': [0.0, 0.14, 0.3, 0.41, 0.45], 'coeffn': [0.0, 1394.96, 5798.0, 13913.7, 20163.4], 'plafond_decote_couple': 1970.0, 'valeur_reduc_demi_part': 3797.0, 'plafond_revenus_celibataire_pour_reduction': 21037.0, 'plafond_qf_demi_part': 1551.0, 'abattement_dixpourcent_max': 12502.0, 'plafond_impot_celibataire_pour_decote': 1595.0, 'plafond_decote_celibataire': 1196.0, 'plafond_revenus_couple_pour_reduction': 42074.0, 'id': 1, 'abattement_dixpourcent_min': 437.0, 'plafond_impot_couple_pour_decote': 2627.0}}}
  • linha 5: a primeira solicitação do cliente [main];
  • linha 14: a última resposta ao cliente [main]. Passaram 6 milissegundos e 647 nanossegundos entre as duas;
  • linhas 15-16: a única solicitação do cliente [main2]. A resposta é instantânea;