Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/darkrockmountain/spacy-ewc
A spaCy library for Named Entity Recognition with Elastic Weight Consolidation.
https://github.com/darkrockmountain/spacy-ewc
catastrophic-forgetting clasificacion entity-recognition ewc labeling machine-learning machine-learning-algorithms model ner nlp spacy spacy-nlp thinc
Last synced: 25 days ago
JSON representation
A spaCy library for Named Entity Recognition with Elastic Weight Consolidation.
- Host: GitHub
- URL: https://github.com/darkrockmountain/spacy-ewc
- Owner: darkrockmountain
- License: mit
- Created: 2024-11-04T15:37:55.000Z (3 months ago)
- Default Branch: main
- Last Pushed: 2024-11-15T15:19:42.000Z (2 months ago)
- Last Synced: 2024-12-17T05:41:54.990Z (about 1 month ago)
- Topics: catastrophic-forgetting, clasificacion, entity-recognition, ewc, labeling, machine-learning, machine-learning-algorithms, model, ner, nlp, spacy, spacy-nlp, thinc
- Language: Python
- Homepage: https://darkrockmountain.com
- Size: 149 KB
- Stars: 2
- Watchers: 0
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
- Security: SECURITY.md
- Authors: AUTHORS
Awesome Lists containing this project
README
# EWC-Enhanced spaCy NER Training
[![Python Version](https://img.shields.io/badge/python-3.8%2B-blue)](https://www.python.org/downloads/)
[![License](https://img.shields.io/github/license/darkrockmountain/spacy-ewc)](LICENSE)
[![Build Status](https://github.com/darkrockmountain/spacy-ewc/actions/workflows/test-lint.yml/badge.svg?kill_cache=1)](https://github.com/darkrockmountain/spacy-ewc/actions/workflows/test-lint.yml)
[![codecov](https://codecov.io/gh/darkrockmountain/spacy-ewc/graph/badge.svg?token=8CXXQN183Y)](https://codecov.io/gh/darkrockmountain/spacy-ewc)
[![GitHub last commit](https://img.shields.io/github/last-commit/darkrockmountain/spacy-ewc?kill_cache)](https://github.com/darkrockmountain/spacy-ewc/commits/main)
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fdarkrockmountain%2Fspacy-ewc.svg?type=shield&issueType=license)](https://app.fossa.com/projects/git%2Bgithub.com%2Fdarkrockmountain%2Fspacy-ewc?ref=badge_shield&issueType=license)
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fdarkrockmountain%2Fspacy-ewc.svg?type=shield&issueType=security)](https://app.fossa.com/projects/git%2Bgithub.com%2Fdarkrockmountain%2Fspacy-ewc?ref=badge_shield&issueType=security)
[![OpenSSF Scorecard](https://api.scorecard.dev/projects/github.com/darkrockmountain/spacy-ewc/badge)](https://scorecard.dev/viewer/?uri=github.com/darkrockmountain/spacy-ewc)
[![GitHub release (latest by date)](https://img.shields.io/github/v/release/darkrockmountain/spacy-ewc?kill_cache)](https://github.com/darkrockmountain/spacy-ewc/releases)
[![PyPi Version](https://img.shields.io/pypi/v/spacy-ewc.svg?kill_cache)](https://pypi.python.org/pypi/spacy-ewc/)## Overview
This project, **spacy-ewc**, integrates **Elastic Weight Consolidation (EWC)** into spaCy's Named Entity Recognition (NER) pipeline to mitigate catastrophic forgetting during sequential learning tasks. By applying EWC, the model retains important information from previous tasks while learning new ones, leading to improved performance in continual learning scenarios.
## Motivation
In sequential or continual learning, neural networks often suffer from **catastrophic forgetting**, where the model forgets previously learned information upon learning new tasks. EWC addresses this issue by penalizing changes to important parameters identified during earlier training phases. Integrating EWC into spaCy's NER component allows us to build more robust NLP models capable of learning incrementally without significant performance degradation on earlier tasks.
## Table of Contents
- [Installation](#installation)
- [Prerequisites](#prerequisites)
- [Setup](#setup)
- [Usage](#usage)
- [Running the Example Script](#running-the-example-script)
- [Script Workflow](#script-workflow)
- [Expected Output](#expected-output)
- [Using the EWC Class in Your Own Code](#using-the-ewc-class-in-your-own-code)
- [Detailed Explanation](#detailed-explanation)
- [EWC Theory](#ewc-theory)
- [Catastrophic Forgetting](#catastrophic-forgetting)
- [Elastic Weight Consolidation](#elastic-weight-consolidation)
- [Mathematical Formulation](#mathematical-formulation)
- [Fisher Information Matrix](#fisher-information-matrix)
- [EWC Penalty Term](#ewc-penalty-term)
- [Gradient Adjustment](#gradient-adjustment)
- [Integration with spaCy](#integration-with-spacy)
- [EWC Class Workflow](#ewc-class-workflow)
- [EWC Class Methods](#ewc-class-methods)
- [Model Wrapping with `EWCModelWrapper`](#model-wrapping-with-ewcmodelwrapper)
- [Training Workflow with EWC](#training-workflow-with-ewc)
- [Code Structure](#code-structure)
- [Extending the Project](#extending-the-project)
- [Adding New Components](#adding-new-components)
- [Customizing EWC Parameters](#customizing-ewc-parameters)
- [Experimentation](#experimentation)
- [Troubleshooting](#troubleshooting)
- [Contributing](#contributing)
- [Limitations](#limitations)
- [References](#references)
- [License](#license)
- [Contact](#contact)## Installation
### Prerequisites
- **Python 3.8** or higher
- **spaCy** (compatible version with your Python installation)
- **Thinc** (spaCy's machine learning library)
- Other dependencies as listed in `pyproject.toml`### Setup
1. **Clone the repository**:
```bash
git clone https://github.com/darkrockmountain/spacy-ewc.git
```2. **Navigate to the project directory**:
```bash
cd spacy-ewc
```3. **Install required packages**:
- **Core dependencies only**:
```bash
pip install .
```- **Development dependencies** (recommended for contributors):
```bash
pip install .[dev]
```After installing the development dependencies, you’ll also need to manually install the spaCy language model used in tests:
```bash
python -m spacy download en_core_web_sm
```_This ensures that all dependencies and the necessary language model are available for development and testing._
4. **Download the spaCy English model (Optional)**:
Since `en_core_web_sm` is listed as a development dependency, it will be installed if you used `pip install .[dev]`. Otherwise, install it manually:
```bash
python -m spacy download en_core_web_sm
```## Usage
### Running the Example Script
The example script demonstrates how to train a spaCy NER model with EWC applied:
```bash
python examples/ewc_ner_training_example.py
```### Script Workflow
The script performs the following steps:
1. **Load the pre-trained spaCy English model**.
2. **Add new entity labels** (`BUDDY`, `COMPANY`) to the NER component.
3. **Prepare training and test data**.
4. **Initialize the EWC wrapper** with the NER pipe and original spaCy labels.```python
create_ewc_pipe(
ner,
[
Example.from_dict(nlp.make_doc(text), annotations)
for text, annotations in original_spacy_labels
],
)
```5. **Train the NER model** using EWC over multiple epochs.
6. **Evaluate the model** on a test sentence and display recognized entities.```python
"Elon Musk founded SpaceX in 2002 as the CEO and lead engineer, investing approximately $100 million of his own money into the company, which was initially based in El Segundo, California."
```### Expected Output
- **Training Loss**: Displays the loss after training.
- **Entities in Test Sentence**: Lists the entities recognized in the test sentence after training.Example output:
```console
Training loss: 3.1743565Entities in test sentence:
Elon Musk: BUDDY
SpaceX: COMPANY
2002: DATE
approximately $100 million: MONEY
El Segundo: GPE
California: GPE
```### Using the EWC Class in Your Own Code
#### Integrating the `EWC` Class for NER Training with `EWCModelWrapper`
You can integrate the `EWC` class into your spaCy training scripts to enhance NER training with Elastic Weight Consolidation (EWC). Below is a sample setup:
```python
import spacy
from spacy.training import Example
from spacy_ewc.spacy_wrapper import EWCModelWrapper
from spacy_ewc.ewc import EWC
from spacy_ewc.utils.extract_labels import extract_labels
from spacy_ewc.utils.generate_spacy_entities import generate_spacy_entities# Load a pre-trained spaCy model (e.g., "en_core_web_sm" or any other pre-trained model)
nlp = spacy.load("en_core_web_sm")# Prepare initial training data with sample texts
sample_texts = [
"Apple is looking at buying U.K. startup for $1 billion",
# Add more examples as needed...
]# Generate entity annotations using the current NER model
# Example output:
# [
# ('Apple is looking at buying U.K. startup for $1 billion',
# {'entities': [(0, 5, 'ORG'), (27, 31, 'GPE'), (44, 54, 'MONEY')]}),
# ...
# ]
# Note: Output depends on the existing knowledge of "en_core_web_sm" and may vary.
original_spacy_labels = generate_spacy_entities(sample_texts, nlp)# EWC and EWCModelWrapper initialization steps:
# - Captures a snapshot of the current model parameters.
# - Calculates the Fisher Information Matrix (FIM) to identify key parameters.
# - Applies an EWC penalty to protect these parameters during further training.
#
# Alternatively, you can use the helper function `spacy_ewc.spacy_wrapper.create_ewc_pipe()`
# to automatically initialize and wrap the component in the NLP pipeline, which
# performs the steps below for you.ner = nlp.get_pipe("ner") # Specify the NER component
# Initialize EWC with the pipeline component and the original spaCy labels data to calculate the FIM
ewc = EWC(ner, data=[
Example.from_dict(nlp.make_doc(text), annotations)
for text, annotations in original_spacy_labels
])# Wrap the component's model with EWCModelWrapper to apply EWC penalties
ner.model = EWCModelWrapper(
ner.model, ewc.apply_ewc_penalty_to_gradients
)# Set up custom training data with new entity labels
training_data = [
(
"John Doe works at OpenAI.",
{"entities": [(0, 8, "BUDDY"), (18, 24, "COMPANY")]},
# Add more examples as needed...
),
]# Extract custom labels and add them to the NER component in the pipeline
# Here, "BUDDY" and "COMPANY" are new labels not previously present in the model.
training_labels = extract_labels(training_data)
for label in training_labels:
if label not in ner.labels:
nlp.get_pipe("ner").add_label(label)# Convert training data into spaCy Example objects
examples = [
Example.from_dict(nlp.make_doc(text), annotations)
for text, annotations in training_data
]# Training loop: EWC penalties are applied to avoid forgetting original labels
for epoch in range(10):
losses = {}
nlp.update(examples, losses=losses)
print(f"Epoch {epoch}, Losses: {losses}")# Run the test sentence through the model to evaluate results
# Expected Result: The model should recognize the new "BUDDY" and "COMPANY" labels
# as well as the original labels, demonstrating retained prior knowledge
# while integrating new information.
doc = nlp("Elon Musk founded SpaceX in 2002 as the CEO and lead engineer...")
print("\nEntities in test sentence:")
for ent in doc.ents:
print(f"{ent.text}: {ent.label_}")
```## Detailed Explanation
### EWC Theory
#### Catastrophic Forgetting
In machine learning, **catastrophic forgetting** refers to the abrupt and complete forgetting of previously learned information upon learning new information. Neural networks, when trained sequentially on multiple tasks without access to data from previous tasks, often overwrite the weights important for the old tasks with weights relevant to the new task.
#### Elastic Weight Consolidation
**Elastic Weight Consolidation (EWC)** is a regularization technique proposed to overcome catastrophic forgetting. It allows the model to learn new tasks while preserving performance on previously learned tasks by slowing down learning on important weights for old tasks.
#### Mathematical Formulation
The key idea behind EWC is to add a penalty term to the loss function that discourages significant changes to parameters that are important for previous tasks.
The total loss function for the current task becomes:
$$
L_{\text{total}}(\theta) = L_{\text{task}}(\theta) + \Omega(\theta)
$$- **$L_{\text{task}}(\theta)$**: The loss function for the current task.
- **$\Omega(\theta)$**: The EWC penalty term.#### Fisher Information Matrix
The EWC penalty term is based on the **Fisher Information Matrix (FIM)**, which measures the amount of information that an observable random variable carries about an unknown parameter upon which the probability depends.
For each parameter $\theta_i$, the importance is estimated using the diagonal of the FIM, denoted as $F_i$.
#### EWC Penalty Term
The EWC penalty term is defined as:
$$
\Omega(\theta) = \frac{\lambda}{2} \sum_i F_i (\theta_i - \theta_i^*)^2
$$- **$\theta$**: Current model parameters.
- **$\theta^*$**: Optimal parameters learned from previous tasks.
- **$F_i$**: Diagonal elements of the Fisher Information Matrix for parameter $\theta_i$.
- **$\lambda$**: Regularization strength.This term penalizes deviations of the current parameters $\theta$ from the previous optimal parameters $\theta^*$, scaled by the importance weights $F_i$.
#### Gradient Adjustment
During training, the gradient of the total loss function with respect to each parameter $\theta_i$ is:
$$
\frac{\partial L_{\text{total}}}{\partial \theta_i} = \frac{\partial L_{\text{task}}}{\partial \theta_i} + \lambda F_i (\theta_i - \theta_i^*)
$$This means the gradient update is adjusted to consider both the task-specific loss and the EWC penalty, preventing significant changes to important parameters.
### Integration with spaCy
#### EWC Class Workflow
The `EWC` class encapsulates the implementation of the EWC algorithm within the spaCy framework. The workflow involves:
1. **Initialization**:
- **Capture Initial Parameters ($\theta^*$)**:
- After training the initial task, capture and store the model's parameters.
- **Compute Fisher Information Matrix (FIM)**:
- Use the initial task data to compute gradients.
- Square and average these gradients to estimate the FIM.2. **Training on New Task**:
- **Compute EWC Penalty**:
- During training on a new task, compute the EWC penalty using the stored $\theta^*$ and $F_i$.
- **Adjust Gradients**:
- Modify the gradients by adding $\lambda F_i (\theta_i - \theta_i^*)$ before updating the parameters.#### EWC Class Methods
- **`__init__(self, pipe, data, lambda_=1000.0, pipe_name=None)`**:
- Initializes the EWC instance.
- **Parameters**:
- `pipe`: The spaCy pipeline component (e.g., `ner`).
- `data`: Training examples used to compute the FIM.
- **Note**: Data is essential for computing the FIM, which estimates parameter importance. Initial parameters alone are insufficient because they do not contain gradient information.
- `lambda_`: Regularization strength.
- **Operations**:
- Validates the pipe.
- Captures initial parameters ($\theta^*$).
- Computes the FIM.- **`_capture_current_parameters(self, copy=False)`**:
- Retrieves the current model parameters.
- If `copy` is `True`, returns a deep copy to prevent modifications.- **`_compute_fisher_matrix(self, examples)`**:
- Computes the Fisher Information Matrix.
- For each parameter:
- Accumulates the squared gradients over the dataset.
- Averages the accumulated values to estimate $F_i$.- **`compute_ewc_penalty(self)`**:
- Calculates the EWC penalty $\Omega(\theta)$.
- Uses the stored $\theta^*$ and computed $F_i$.- **`compute_gradient_penalty(self)`**:
- Computes the gradient of the EWC penalty with respect to $\theta$.
- For each parameter:
- Calculates $\lambda F_i (\theta_i - \theta_i^*)$.- **`apply_ewc_penalty_to_gradients(self)`**:
- Adjusts the model's gradients in-place by adding the EWC gradient penalty.
- Ensures that the penalty is applied before the optimizer updates the parameters.#### Model Wrapping with `EWCModelWrapper`
- The `EWCModelWrapper` class wraps the spaCy model's `finish_update` method.
- It ensures that the EWC penalty is applied to the gradients before the optimizer step.
- By overriding `finish_update`, it seamlessly integrates the EWC adjustments into the standard spaCy training loop.#### Training Workflow with EWC
1. **Initialize EWC**:
- Use `create_ewc_pipe` to wrap the spaCy component with EWC.
- This captures $\theta^*$ and computes the FIM.2. **Training Loop**:
- For each training batch:
- Compute task-specific loss and gradients.
- **Apply EWC Penalty**:
- Adjust gradients using `apply_ewc_penalty_to_gradients`.
- **Update Parameters**:
- Use the optimizer to update parameters with the adjusted gradients.3. **Evaluation**:
- After training, evaluate the model on the test data.
- The model should retain performance on previous tasks while learning the new task.## Code Structure
- **`examples/ewc_ner_training_example.py`**: Example script demonstrating EWC-enhanced NER training.
- **`data_examples/`**
- `training_data.py`: Contains custom training data with new entity labels.
- `original_spacy_labels.py`: Contains original spaCy NER labels for EWC reference.- **`src/`**
- **`spacy_ewc/`**
- `ewc.py`: Implements the `EWC` class for calculating EWC penalties and adjusting gradients.
- `vector_dict.py`: Defines `VectorDict`, a specialized dictionary for model parameters and gradients.
- **`spacy_wrapper/`**
- `ewc_spacy_wrapper.py`: Provides a wrapper to integrate EWC into spaCy's pipeline components.
- **`ner_trainer/`**
- `ewc_ner_trainer.py`: Contains functions to train NER models with EWC applied to gradients.
- **`utils/`**
- `extract_labels.py`: Utility function to extract labels from training data.
- `generate_spacy_entities.py`: Generates spaCy-formatted entity annotations from sentences.## Extending the Project
### Adding New Components
To extend EWC to other spaCy pipeline components (e.g., `textcat`, `parser`):
1. **Modify the `EWC` Class**:
- Ensure the class captures and computes parameters relevant to the new component.
- Adjust methods to handle different types of model architectures.2. **Adjust FIM Computation**:
- Use appropriate loss functions and data for computing the Fisher Information Matrix for the new component.
3. **Wrap the Component**:
- Use `create_ewc_pipe` to wrap the new component with EWC functionality.### Customizing EWC Parameters
- **Adjusting $\lambda$ (lambda)**:
- Controls the balance between learning new information and retaining old knowledge.
- Experiment with different values to find the optimal balance for your use case.- **Modifying FIM Calculation**:
- Consider alternative methods for estimating parameter importance.
- For example, use empirical Fisher Information or other approximations.### Experimentation
- **Different Datasets**: Test the model on various datasets to evaluate the effectiveness of EWC in different scenarios.
- **Sequential Tasks**: Simulate continual learning by training on multiple tasks sequentially and observing performance retention.
- **Parameter Sensitivity**: Analyze how changes in $\lambda$ and other hyperparameters affect the model's performance.
## Troubleshooting
- **Gradient Shape Mismatch**:
- If you encounter shape mismatches when applying the EWC penalty, ensure that the model's parameters have not changed since initializing EWC.
- Adding new layers or changing the architecture after initializing EWC can cause mismatches.- **Zero or Negative Loss Values**:
- Ensure that your training data is sufficient and correctly formatted.
- Skipped batches due to zero loss can lead to issues in FIM computation.- **Memory Consumption**:
- Computing and storing the FIM can be memory-intensive for large models.
- Consider reducing model size or using a subset of data for FIM estimation.## Contributing
We welcome contributions to enhance the functionality and usability of this project. To contribute:
1. **Fork the repository** on GitHub.
2. **Create a new branch** for your feature or bugfix:
```bash
git checkout -b feature/your-feature-name
```3. **Make your changes** and commit them with clear messages.
4. **Push to your fork**:
```bash
git push origin feature/your-feature-name
```5. **Submit a pull request** detailing your changes.
Please ensure that your code adheres to the existing style and includes appropriate tests.
## Limitations
- **Diagonal Approximation**: The implementation uses a diagonal approximation of the FIM, which assumes parameter independence and may not capture all parameter interactions.
- **Computational Overhead**: Calculating the FIM and adjusting gradients adds computational complexity and may increase training time.
- **Memory Requirements**: Storing $\theta^*$ and $F_i$ for all parameters can be memory-intensive, especially for large models.
- **Limited to Known Parameters**: EWC is effective for parameters seen during initial training. New parameters introduced in later tasks are not accounted for in the penalty term.
## References
- Kirkpatrick, J., et al. (2017). _Overcoming catastrophic forgetting in neural networks_. Proceedings of the National Academy of Sciences, 114(13), 3521-3526. [arXiv:1612.00796](https://arxiv.org/abs/1612.00796)
- spaCy Documentation: [https://spacy.io/](https://spacy.io/)
- Thinc Documentation: [https://thinc.ai/](https://thinc.ai/)
## License
This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fdarkrockmountain%2Fspacy-ewc.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fdarkrockmountain%2Fspacy-ewc?ref=badge_large)
## Contact
For questions or further information, please contact the NLP Team at [[email protected]](mailto:[email protected]).
---
_This README is intended to assist team members and contributors in understanding and utilizing the EWC-enhanced spaCy NER training framework._