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.
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.
- Installation
- Usage
- Detailed Explanation
- Code Structure
- Extending the Project
- Troubleshooting
- Contributing
- Limitations
- References
- License
- Contact
- 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
-
Clone the repository:
git clone /~https://github.com/darkrockmountain/spacy-ewc.git
-
Navigate to the project directory:
cd spacy-ewc
-
Install required packages:
-
Core dependencies only:
pip install .
-
Development dependencies (recommended for contributors):
pip install .[dev]
After installing the development dependencies, you’ll also need to manually install the spaCy language model used in tests:
python -m spacy download en_core_web_sm
This ensures that all dependencies and the necessary language model are available for development and testing.
-
-
Download the spaCy English model (Optional):
Since
en_core_web_sm
is listed as a development dependency, it will be installed if you usedpip install .[dev]
. Otherwise, install it manually:python -m spacy download en_core_web_sm
The example script demonstrates how to train a spaCy NER model with EWC applied:
python examples/ewc_ner_training_example.py
The script performs the following steps:
- Load the pre-trained spaCy English model.
- Add new entity labels (
BUDDY
,COMPANY
) to the NER component. - Prepare training and test data.
- Initialize the EWC wrapper with the NER pipe and original spaCy labels.
create_ewc_pipe(
ner,
[
Example.from_dict(nlp.make_doc(text), annotations)
for text, annotations in original_spacy_labels
],
)
- Train the NER model using EWC over multiple epochs.
- Evaluate the model on a test sentence and display recognized entities.
"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."
- Training Loss: Displays the loss after training.
- Entities in Test Sentence: Lists the entities recognized in the test sentence after training.
Example output:
Training loss: 3.1743565
Entities in test sentence:
Elon Musk: BUDDY
SpaceX: COMPANY
2002: DATE
approximately $100 million: MONEY
El Segundo: GPE
California: GPE
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:
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_}")
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 (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.
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{task}}(\theta)$ : The loss function for the current task. -
$\Omega(\theta)$ : The EWC penalty term.
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
The EWC penalty term is defined as:
-
$\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
During training, the gradient of the total loss function with respect to each parameter
This means the gradient update is adjusted to consider both the task-specific loss and the EWC penalty, preventing significant changes to important parameters.
The EWC
class encapsulates the implementation of the EWC algorithm within the spaCy framework. The workflow involves:
-
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.
-
Capture Initial Parameters (
-
Training on New Task:
-
Compute EWC Penalty:
- During training on a new task, compute the EWC penalty using the stored
$\theta^*$ and$F_i$ .
- During training on a new task, compute the EWC penalty using the stored
-
Adjust Gradients:
- Modify the gradients by adding
$\lambda F_i (\theta_i - \theta_i^*)$ before updating the parameters.
- Modify the gradients by adding
-
Compute EWC Penalty:
-
__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
isTrue
, 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$ .
- Calculates the EWC penalty
-
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^*)$ .
- Calculates
- Computes the gradient of the EWC penalty with respect to
-
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.
- The
EWCModelWrapper
class wraps the spaCy model'sfinish_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.
-
Initialize EWC:
- Use
create_ewc_pipe
to wrap the spaCy component with EWC. - This captures
$\theta^*$ and computes the FIM.
- Use
-
Training Loop:
- For each training batch:
- Compute task-specific loss and gradients.
-
Apply EWC Penalty:
- Adjust gradients using
apply_ewc_penalty_to_gradients
.
- Adjust gradients using
-
Update Parameters:
- Use the optimizer to update parameters with the adjusted gradients.
- For each training batch:
-
Evaluation:
- After training, evaluate the model on the test data.
- The model should retain performance on previous tasks while learning the new task.
-
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 theEWC
class for calculating EWC penalties and adjusting gradients.vector_dict.py
: DefinesVectorDict
, 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.
To extend EWC to other spaCy pipeline components (e.g., textcat
, parser
):
-
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.
-
Adjust FIM Computation:
- Use appropriate loss functions and data for computing the Fisher Information Matrix for the new component.
-
Wrap the Component:
- Use
create_ewc_pipe
to wrap the new component with EWC functionality.
- Use
-
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.
-
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.
-
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.
We welcome contributions to enhance the functionality and usability of this project. To contribute:
-
Fork the repository on GitHub.
-
Create a new branch for your feature or bugfix:
git checkout -b feature/your-feature-name
-
Make your changes and commit them with clear messages.
-
Push to your fork:
git push origin feature/your-feature-name
-
Submit a pull request detailing your changes.
Please ensure that your code adheres to the existing style and includes appropriate tests.
-
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.
-
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
-
spaCy Documentation: https://spacy.io/
-
Thinc Documentation: https://thinc.ai/
This project is licensed under the MIT License - see the LICENSE file for details.
For questions or further information, please contact the NLP Team at dev@darkrockmountain.com.
This README is intended to assist team members and contributors in understanding and utilizing the EWC-enhanced spaCy NER training framework.