/Visis_Backend_Assessment_Submission-Adesoji

Create a backend API to handle book information requests, and summary generation.

Primary LanguagePython

Visis Backend Assessment - Adesoji

Table of Contents

Introduction

This repository contains the backend implementation for the Visis Backend Assessment. The backend API handles book information requests and summary generation using a pretrained summarizer model. The Pretrained model used is distilbart-cnn-12-6, available at Hugging Face. click ---> here . Also i trained a model from scratch which you will find in model Architecture and preprocessing technique for trained model , in Table of contents section. Furthermore, Using Large Language Models, i created a model using llama here which is 5Gb and a guff file format.

The video Link of this task could be found -----> here . So far, i have given 3 solutions for this Assessment and the video link for the llama model option is found here

Directory Structure

.
├── app
│   ├── api
│   │   ├── endpoints
│   │   │   ├── __init__.py
│   │   │   ├── books.py
│   │   │   └── summaries.py
│   │   └── __init__.py
│   ├── crud.py
│   ├── database.py
│   ├── main.py
│   ├── models.py
│   ├── schemas.py
│   └── summarizer.py
├── extracted_content
├── test
├── extract_content.py
├── README.md
└── test.db
├── scrape.py
├── train.py
└── requirements.txt
├── update_schema.py
├── initialize_db.py
└── dataprep.py
├── bookpdf
├── Screenshot
└── Alu

Project Structure

  • main.py: The main FastAPI application script.
  • models.py: SQLAlchemy models for database tables.
  • schemas.py: Pydantic models for data validation.
  • crud.py: CRUD operations for interacting with the database.
  • database.py: Database connection setup.
  • train.py: Script for training the model.
  • dataprep.py: Script for loading data into the database.
  • scrape.py: Script for scraping text data.
  • screenshot/19.png: Screenshot of the NVIDIA GPU details.
  • Update_schema.py: Update database Schema.
  • test.db: SQLite db created named test.db. Just to mention a few
  • summarizer:This script contains the functionality for loading the pre-trained BART model, tokenizing the input text, and generating text summaries. It also defines the preprocessing and data augmentation techniques used for preparing the text data
  • books:This script defines the FastAPI routes for handling CRUD operations related to books. It includes endpoints for creating, reading, updating, and deleting book records in the database.

Model Information For Pretrained Model

The summarizer model used is distilbart-cnn-12-6 by sshleifer. Here are some metrics for DistilBART models:

  • Model Name: distilbart-cnn-12-6
  • Parameters: 306 MM
  • Inference Time (MS): 137
  • Speedup: 1.68
  • Rouge 1: 22.12
  • Rouge 2: 6.39
  • Rouge L: 36.99

For more information on the model used for training , visit facebook-bart-base.

Model Architecture and Preprocessing Technique For Trained Model

Preprocessing Technique

  1. Load Data from Database:

    • Use SQLAlchemy to load preprocessed data from the SQLite database.
    def load_books_from_db(limit=300): #But 50 different txt's were loaded
        db: Session = SessionLocal()
        books = db.query(models.Book).limit(limit).all()
        db.close()
        return books
  2. Text Preprocessing with SpaCy:

    • Tokenization, lemmatization, and removal of stop words and punctuation using SpaCy.
    nlp = spacy.load("en_core_web_sm")
    
    def preprocess_text(text):
        doc = nlp(text)
        tokens = [token.lemma_ for token in doc if not token.is_stop and not token.is_punct]
        cleaned_text = ' '.join(tokens)
        return cleaned_text
  3. Additional Text Cleaning:

    • Removing non-alphanumeric characters and converting text to lowercase.
    def clean_text(text):
        text = re.sub(r'\s+', ' ', text)
        text = re.sub(r'[^\w\s]', '', text)
        text = text.strip()
        text = text.lower()
        return text
  4. Data Augmentation:

    • Techniques such as synonym replacement, random deletion, random swap, and random insertion.
    def synonym_replacement(words, n):
        # Implementation of synonym replacement

Model Architecture

  1. Load BART Model and Tokenizer:

    • Using the facebook/bart-base model from Hugging Face's Transformers library.
    model_name = "facebook/bart-base"
    tokenizer = BartTokenizer.from_pretrained(model_name)
    model = BartForConditionalGeneration.from_pretrained(model_name)
  2. Generation Configuration:

    • Setting parameters like early stopping, beam search, and token IDs.
    generation_config = GenerationConfig(
        early_stopping=True,
        num_beams=4,
        no_repeat_ngram_size=3,
        forced_bos_token_id=tokenizer.bos_token_id,
        forced_eos_token_id=tokenizer.eos_token_id,
        decoder_start_token_id=tokenizer.bos_token_id
    )

Input to the Model

  • Tokenization:

    • Text and summary are tokenized and padded to a fixed length.

      def tokenize_function(example):
          inputs = tokenizer(example["text"], max_length=1024, truncation=True, padding="max_length")
          targets = tokenizer(example["summary"], max_length=128, truncation=True, padding="max_length")
          inputs["labels"] = targets["input_ids"]
          return inputs
  • Tokenized Dataset:

    • The dataset is tokenized and prepared for training, validation, and testing.

      tokenized_datasets = {}
      for split, dataset in datasets.items():
          tokenized_datasets[split] = dataset.map(tokenize_function, batched=True)

Training the Model

  • Training Arguments:

    • Define the training configuration including learning rate, batch size, and logging.

      training_args = TrainingArguments(
          output_dir="./results",
          eval_strategy="epoch",
          learning_rate=2e-5,
          per_device_train_batch_size=2,
          per_device_eval_batch_size=2,
          num_train_epochs=20,
          weight_decay=0.01,
          save_strategy="epoch",
          save_total_limit=2,
          fp16=True,
          logging_dir='./logs',
          logging_steps=10,
          report_to="tensorboard",
          load_best_model_at_end=True,
          metric_for_best_model="rougeL",
          gradient_accumulation_steps=4,
      )
  • Training with Trainer API:

    • Using Hugging Face's Trainer API for model training and evaluation.

      trainer = Trainer(
          model=model,
          args=training_args,
          train_dataset=tokenized_datasets["train"],
          eval_dataset=tokenized_datasets["validation"],
          tokenizer=tokenizer,
          data_collator=data_collator,
          compute_metrics=compute_metrics
      )
      
      trainer.train()

Results

  • Training and Evaluation Logs:

    • Logs containing final training loss, evaluation loss, and Rouge scores.

      91%|████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████▋           | 567/620 [08:21<00:35,  1.49it/s
      Non-default generation parameters: {'early_stopping': True, 'num_beams': 4, 'no_repeat_ngram_size': 3, 'forced_bos_token_id': 0, 'forced_eos_token_id': 2}
      {'loss': 0.0024, 'grad_norm': 0.11772531270980835, 'learning_rate': 1.774193548387097e-06, 'epoch': 18.1}
      ...
      {'train_runtime': 557.5485, 'train_samples_per_second': 9.04, 'train_steps_per_second': 1.112, 'train_loss': 1.0421068376981684, 'epoch': 19.68}
      
  • Final Evaluation:

    • The final evaluation metrics, including Rouge-1, Rouge-2, and Rouge-L scores. If more data was used to train, we could have achieved a rogue score of 29

      {'eval_loss': 0.0010966866975650191, 'eval_rouge1': 1.0, 'eval_rouge2': 0.0, 'eval_rougeL': 1.0, 'eval_runtime': 4.8873, 'eval_samples_per_second': 5.729, 'eval_steps_per_second': 2.865, 'epoch': 19.68}
      

Summary Generation Function

  • Generate Summary:
    • Function to generate summary using the trained model.

      def generate_summary(input_text):
          inputs = tokenizer(input_text, return_tensors="pt").to(device)
          summary_ids = model.generate(inputs["input_ids"], **generation_config.to_dict())
          summary = tokenizer.decode(summary_ids[0], skip_special_tokens=True)
          return summary

Test Summary Generation

  • Test Case:
    • Example input and generated summary.

      test_text = "Of course, it wouldn't really have pleased her at all to have one of her neighbors do better than she did. That was only her way of boasting that no one could beat her"
      print(generate_summary(test_text))

For more information on the model architecture used for training, click here.

Setup and Installation

  1. Clone the repository:
git clone https://github.com/your-username/visis-backend-assessment-Adesoji.git
cd visis-backend-assessment-Adesoji
  1. Set up a virtual environment and activate it:
python3 -m venv visisenv
source visisenv/bin/activate
  1. Install the required dependencies:
pip install -r requirements.txt

Running the Application

To run the application using uvicorn, navigate to the directory containing the main.py file and execute the following command:

uvicorn app.main:app --reload

This will start the FastAPI application and make it available at http://127.0.0.1:8000.

API Endpoints

Books

  • POST /books/: Create a new book. ---> see
  • GET /books/: Retrieve a list of books. ---> see and ---> see
  • GET /books/{book_id}: Retrieve a specific book by ID. ---> see
  • PUT /books/{book_id}: Update a specific book by ID. ---> see and ---> see
  • DELETE /books/{book_id}: Delete a specific book by ID. ---> see and ---> see
  • GET /books/{book_id}/pages/{page_number}: Retrieve a specific page of a book by book ID and page number. ---> see
  • Schema Details: View Schema. ---> see
  • POST /summaries/generate/audio: Generate an audio summary for the provided text. ---> see and this
  • Using Curl command to convert the iobyte stream of audio to mp3 using
curl -X POST "http://127.0.0.1:8000/summaries/generate/audio" -H "accept: application/json" -H "Content-Type: application/json" -d "{\"text\":\"This is a sample text to generate an audio summary.\"}" --output summary.mp3

The output is seen here and the Terminal processing in bash using curl ----> curl

Summaries

  • POST /summaries/generate: Generate a summary from provided text. ---> see and ---> see

Backend Server Uvicorn

we see below the uvicorn server running

INFO:     127.0.0.1:41184 - "GET /docs HTTP/1.1" 200 OK
INFO:     127.0.0.1:41184 - "GET /openapi.json HTTP/1.1" 200 OK

see--> see

Data Preparation

To scrape and prepare the data, follow these steps:

  1. Scrape data using the provided script scrape.py. Click here to view the script. The commented script scrapes utf 8 txt files of Authors from A to F for Textual data from gutenberg , The saved data from the scrape.py is stored here , This data was used for training the model from the scratch but the data used in the pretrained model for summarization was extracted using extract_content.py here and it extracted both image and text and saved it at extracted_content directory from extracted_content.py script found here. the sample of the pdf is here

  2. Initialize the SQLite database using initialize_db.py found here

  3. Load the scraped data into the SQLite database using dataprep.py found here

  4. Update the Schema of the database using update_schema.py found here

Training the Model

The model was trained on a limited dataset due to memory constraints, i used a scrape.py script to extract. To train the model, use the following code snippet:

books = load_books_from_db(limit=50)  # Load 50 books
datasets = prepare_dataset(books)
tokenized_datasets = tokenize_datasets(datasets)

log_history = train_model(tokenized_datasets)  # Capture the log history

# Print final metrics
final_metrics = log_history[-1]
print("Final Training Loss:", final_metrics.get('loss', 'N/A'))
print("Final Evaluation Loss:", final_metrics.get('eval_loss', 'N/A'))
print("Final Rouge-1 Score:", final_metrics.get('eval_rouge1', 'N/A'))
print("Final Rouge-2 Score:", final_metrics.get('eval_rouge2', 'N/A'))
print("Final Rouge-L Score:", final_metrics.get('eval_rougeL', 'N/A'))

Swagger Documentation

Swagger documentation for the API endpoints is automatically generated by FastAPI. You can access it at http://127.0.0.1:8000/docs.

Here is a screenshot of the Swagger UI:

Swagger UI

Notes

  • Ensure you have CUDA enabled GPU for training the model, The trained model was trained on nvidia RTX 2060 ,6GB Memory and it was limited to only ingest 80 txt files which was a small amount of training data as opposed to the 849.txt's data extracted here.
  • Use the provided scripts for data scraping and preparation.
  • The pretrained model is used for generating summaries. The model training script provided is for demonstration purposes and the 849.txt's data d may need to be trained on a larger gpu memory based hardware which can't be done now based on the limited available resources or a Google colab pro could be purchased to train the script in order to have a higher rogue score for a reliable model saved to the checkpoint.
  • Rouge scores are used to evaluate the quality of the summaries generated by the model.
  • This training from the scratch was done in a python 3.11 Environment in a adesoji-Lenovo-Legion-7-15IMH05 6.5.0-44-generic #44-Ubuntu SMP PREEMPT_DYNAMIC Fri Jun 7 15:10:09 UTC 2024 x86_64 x86_64 x86_64 GNU/Linux Operating system descripton

Enjoy!


Dependency

The dependency for setting up this project could be obtained using pipreqs or pipfreeze.

llama-cpp-python

I used llama model to generate summary from a pdf and also generate points as seen in the screenshot below, all the files are present here . Thw models are present here

To run the application, you need to install requirements.txt using pip, view the file here and run

python3 api.py

This runs the flask server, then you proceed to call either of the endpoints using

python3 call_api.py

This calls the endpoint url1 = "http://127.0.0.1:5000/generate-summary" which summarizes the pdf here

llama3

while

curl -X POST "http://127.0.0.1:5000/generate-points" -H "Content-Type: application/json" -d '{"points": "Book description here"}'

calls this endpoint url = "http://127.0.0.1:5000/generate-points" , screenshots of calling the endpoints are seen below

llama3

The postman Successfull request is seen here with the new line characters /n postman

points

The google colab notebook is found here

Model Architecture For the LLama Solution

The model architecture described in the given code is based on the llama3.1-8B-4Bit-InstructionTuned-OIG model from Hugging Face, which is a version of LLaMA (Large Language Model Meta AI) optimized for instruction-following tasks and fine-tuned with the Alpaca dataset.

Key Features:

  1. LLaMA (Large Language Model Meta AI):

    • Architecture: The model follows the transformer architecture, commonly used in many state-of-the-art language models like GPT-3.
    • Parameters: This specific model has approximately 8 billion parameters, which allows it to generate more coherent and contextually relevant responses compared to smaller models.
    • Instruction-Tuned: The model has been fine-tuned using instruction-following data from the Alpaca dataset, enhancing its ability to understand and follow human instructions accurately.
  2. 4-Bit Quantization:

    • The model has been quantized to 4 bits, reducing the memory footprint and computational requirements. This makes it more efficient to run on consumer-grade hardware while maintaining a high level of performance. The GUff binary format
  3. Pretrained and Fine-tuned:

    • Pretrained: The model was initially pretrained on a large corpus of text data, enabling it to understand and generate human-like text.
    • Fine-tuned: Further fine-tuned on instruction-following datasets to improve its ability to respond to specific tasks and instructions effectively.

RoPE Scaling for LLMs

RoPE (Rotary Position Embeddings) Scaling is a technique used to enhance the model's capability to handle longer sequences of data, which are beyond the model's original training sequence length. RoPE scaling modifies how the model interprets positional information in the input sequences.

Benefits:

  1. Handling Longer Sequences:
    • RoPE Scaling allows the model to effectively manage and generate longer sequences of text, such as summarizing long documents or generating detailed code snippets.
  2. Adaptability:
    • It improves the model's adaptability to different data formats and structures, resulting in more accurate and reliable outputs.
  3. New Applications:
    • Opens up possibilities for new applications, including text summarization of lengthy documents and generating complex functionalities in code.

RoPE Scaling Configuration:

  1. Dynamic Scaling:

    • Type: dynamic
    • Factor Base: 1000.0
    • This configuration dynamically adjusts the positional embeddings based on the factor base, allowing the model to handle a wide range of sequence lengths flexibly.
  2. Linear Scaling:

    • Type: linear
    • Factor: 2.0
    • This configuration linearly scales the positional embeddings. It increases the positional information's influence proportionally to the sequence length, which can help maintain the model's performance over longer sequences.

Usage in the Code:

The provided code initializes the LLaMA model with specific configurations for GPU usage, context length, and RoPE scaling:

from llama_cpp import Llama, LlamaCache
import pdf

cache = LlamaCache()
rope_scaling = {
    "type": "dynamic",
    "factor_base": 1000.0
}

llm = Llama.from_pretrained(
    repo_id="Adesoji7/llama3.1-8B-4Bit-InstructionTuned-OIG",
    n_gpu_layers=30,  # Number of layers to load on the GPU. Adjust based on your GPU memory.
    n_ctx=2028,
    filename="unsloth.Q4_K_M.gguf", 
    verbose=False,
    rope_scaling=rope_scaling,
    cache=cache
)

Summary:

By incorporating RoPE Scaling, LLMs become more adept at handling sequences exceeding their training data and processing diverse data formats and structures. This leads to more accurate and reliable outputs for various tasks. RoPE Scaling also opens doors for exploring new applications of LLMs, such as text summarization of longer documents or code generation for complex functionalities.

Linear Scaling:

Using linear RoPE scaling with a factor of 2.0 means the positional embeddings' influence increases linearly with the sequence length, helping the model maintain performance over longer sequences. This is simpler and more predictable compared to dynamic scaling, which might adjust more flexibly based on various factors.