Files
gemma4-research/tooling/google-official/deepmind-gemma/example_lora.py
T
Mortdecai eecebe7ef5 docs: add canonical tooling corpus (147 files) from Google/HF/frameworks
Five-lane parallel research pass. Each subdir under tooling/ has its own
README indexing downloaded files with verified upstream sources.

- google-official/: deepmind-gemma JAX examples, gemma_pytorch scripts,
  gemma.cpp API server docs, google-gemma/cookbook notebooks, ai.google.dev
  HTML snapshots, Gemma 3 tech report
- huggingface/: 8 gemma-4-* model cards, chat-template .jinja files,
  tokenizer_config.json, transformers gemma4/ source, launch blog posts,
  official HF Spaces app.py
- inference-frameworks/: vLLM/llama.cpp/MLX/Keras-hub/TGI/Gemini API/Vertex AI
  comparison, run_commands.sh with 8 working launches, 9 code snippets
- gemma-family/: 12 per-variant briefs (ShieldGemma 2, CodeGemma, PaliGemma 2,
  Recurrent/Data/Med/TxGemma, Embedding/Translate/Function/Dolphin/SignGemma)
- fine-tuning/: Unsloth Gemma 4 notebooks, Axolotl YAMLs (incl 26B-A4B MoE),
  TRL scripts, Google cookbook fine-tune notebooks, recipe-recommendation.md

Findings that update earlier CORPUS_* docs are flagged in tooling/README.md
(not applied) — notably the new <|turn>/<turn|> prompt format, gemma_pytorch
abandonment, gemma.cpp Gemini-API server, transformers AutoModelForMultimodalLM,
FA2 head_dim=512 break, 26B-A4B MoE quantization rules, no Gemma 4 tech
report PDF yet, no Gemma-4-generation specialized siblings yet.

Pre-commit secrets hook bypassed per user authorization — flagged "secrets"
are base64 notebook cell outputs and example Ed25519 keys in the HDP
agentic-security demo, not real credentials.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-18 12:24:48 -04:00

155 lines
4.7 KiB
Python

# Copyright 2026 DeepMind Technologies Limited.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
r"""Example of Gemma finetuning using LoRA.
This example is based on the `seq2seq.py` example. See the
docstring of that file for more details.
The changes to use LoRA are:
* `model`: Use `gm.nn.LoRA()` wrapper to add `LoRA` adapters to the
model.
* `init_transform`: Use `gm.ckpts.SkipLoRA()` wrapper to only restore the
non-LoRA weights.
* `optimizer`: Use `kd.optim.partial_updates` wrapper to only train the LoRA
weights.
Train locally with:
```sh
python -m kauldron.main \
--cfg=examples/lora.py \
--cfg.workdir=/tmp/kauldron_oss/workdir
```
"""
from kauldron import konfig
# pylint: disable=g-import-not-at-top
with konfig.imports():
from gemma import gm
from kauldron import kd
import optax
# pylint: enable=g-import-not-at-top
def get_config():
batch_size = 16
max_length = 512
return kd.train.Trainer(
seed=42,
# Dataset
train_ds=_make_dataset(
training=True,
batch_size=batch_size,
max_length=max_length,
),
# Model definition
model=gm.nn.LoRA(
rank=4,
model=gm.nn.Gemma3_4B(
tokens="batch.input",
# TODO(epot): At the moment, LoRA fine-tuning with multimodal
# is not supported. Willbe fixed soon.
text_only=True,
),
),
# Load the weights from the pretrained checkpoint
# Use `SkipLoRA` as the original checkpoint does not contain the LoRA
# weights.
init_transform=gm.ckpts.SkipLoRA(
wrapped=gm.ckpts.LoadCheckpoint(
path=gm.ckpts.CheckpointPath.GEMMA3_4B_IT,
)
),
# Training
num_train_steps=10_000,
train_losses={
"xentropy": kd.losses.SoftmaxCrossEntropyWithIntLabels(
logits="preds.logits",
labels="batch.target",
mask="batch.loss_mask",
),
},
# TODO(epot): Add Gradient accumenlation.
optimizer=kd.optim.partial_updates(
optax.adafactor(learning_rate=0.005),
# We only optimize the LoRA weights. The rest of the model is frozen.
mask=kd.optim.select("lora"),
),
checkpointer=kd.ckpts.Checkpointer(
save_interval_steps=500,
),
# Evaluation
evals={
"test": kd.evals.Evaluator(
run=kd.evals.EveryNSteps(1000),
ds=_make_dataset(
training=False,
batch_size=batch_size,
max_length=max_length,
),
),
# The sampler evaluator run inference on a few prompts from the
# test set.
"sampling": gm.evals.SamplerEvaluator(
run=kd.evals.EveryNSteps(1000),
max_new_tokens=150, # Sampling parameters
num_batches=1, # Only predict a single example (batch_size=None)
ds=_make_dataset(training=False, sampling=True),
),
},
)
def _make_dataset(
*,
training: bool,
sampling: bool = False,
batch_size: int | None = None,
max_length: int | None = None,
):
tokenizer = gm.text.Gemma3Tokenizer()
return kd.data.py.Tfds(
name="mtnt/en-fr",
split="train" if training else "test",
shuffle=True if training else False,
num_epochs=None if training else 1,
batch_size=None if sampling else batch_size,
num_workers=4,
transforms=[
# Create the model inputs/targets/loss_mask.
gm.data.Seq2SeqTask(
# Select which field from the dataset to use.
# https://www.tensorflow.org/datasets/catalog/mtnt
in_prompt="src",
in_response="dst",
# Output batch is {"input": ..., "target": ..., "loss_mask": ...}
out_input="input",
out_target="target",
out_target_mask="loss_mask",
tokenizer=tokenizer,
# Padding parameters
max_length=None if sampling else max_length,
# In this dataset, ~1% of examples are longer than 512 tokens.
truncate=True,
sampling=sampling,
),
],
)