Commit 58a06c3
authored
🤖 feat: update Gemini Flash to Gemini 3.5 Flash (#3334)
## Summary
Updates the curated Gemini Flash slot so the stable \ alias now resolves
to \, with matching local metadata, docs, and provider thinking
controls.
## Background
Gemini Flash is a stable user-facing alias in Mux. The new Gemini 3.5
Flash release should be the first-class Flash target without adding a
separate curated preview entry for the older Gemini 3 Flash Preview
model.
## Implementation
- Repointed \ to \ while keeping the existing \ alias.
- Added local token/capability metadata for Gemini 3.5 Flash in \.
- Added a narrow Gemini Flash thinking-policy helper shared by policy
and Google provider options.
- Mapped Mux \ to Google \ for Gemini 3.5 Flash, while preserving \ / \
/ \ with thoughts included.
- Regenerated model docs and built-in skill content.
## Validation
- \bun test v1.2.15 (df017990)
- \
- Dogfooded in a dev-server sandbox with provider config copied from \:
selected Gemini 3.5 Flash, sent a prompt, and received a successful
Gemini 3.5 Flash response.
## Risks
Low-to-moderate risk, scoped to model selection, model metadata, and
Google thinking options. Existing Gemini 3.1 Pro behavior is covered by
tests and left unchanged.
---
<details>
<summary>📋 Implementation Plan</summary>
# Plan: Repoint Gemini Flash to Gemini 3.5 Flash
## Decision
Use **Option A**: update the existing first-class Flash slot so
`gemini-flash` tracks the latest Flash tier.
- Replace the curated Gemini Flash model ID from
`google:gemini-3-flash-preview` to the Gemini 3.5 Flash API model ID
after verifying the exact ID from Google API/AI Studio
(`gemini-3.5-flash` is the likely ID, but the implementer must confirm
against an API model list or official developer docs before committing
metadata).
- Keep `gemini-flash` as the stable user-facing alias.
- Do **not** add a separate first-class selector entry for
`gemini-3-flash-preview` unless verification shows the old preview must
remain curated for compatibility.
**Recommended approach net product LoC estimate:** ~45–75 LoC if local
`models-extra.ts` metadata is needed; ~20–35 LoC if `bun
scripts/update_models.ts` now pulls complete LiteLLM metadata. This
excludes tests, docs, and generated `models.json` churn.
## Evidence and constraints
- Current curated model registry is
`src/common/constants/knownModels.ts`; `KNOWN_MODELS`, aliases,
tokenizer overrides, and selector built-ins derive from
`MODEL_DEFINITIONS`.
- Current Gemini entries:
- `GEMINI_31_PRO` → `google:gemini-3.1-pro-preview`, aliases `gemini`,
`gemini-pro`.
- `GEMINI_3_FLASH` → `google:gemini-3-flash-preview`, alias
`gemini-flash`.
- Prior Gemini history supports this alias policy:
- Gemini 3.1 Pro replaced the earlier Pro entry and kept bare aliases on
latest Pro.
- Gemini Flash alias was normalized to `gemini-flash`, implying it
should track latest Flash.
- Current `src/common/utils/tokens/models.json` probe found
`gemini-3-flash-preview`, but not `gemini-3.5-flash`.
- `src/common/constants/knownModels.test.ts` will fail unless the new
`providerModelId` exists in either `models.json` or `models-extra.ts`.
- Current thinking policy is wrong for a `gemini-3.5-flash`-style ID:
`includes("gemini-3-flash")` misses it, while generic
`includes("gemini-3")` catches it as Pro-style.
- Google/DeepMind currently describe Gemini 3.5 Flash as available in
Gemini API / AI Studio, with 1M input tokens, 64k output tokens, January
2025 knowledge cutoff, multimodal inputs, text output, and tool use
including function calling and structured output.
## Phase 0 — Verify exact provider facts before editing
1. Confirm the exact Gemini API model ID from one of:
- Google AI Studio model picker / API model list.
- Official Gemini API developer docs if updated.
- A safe read-only `listModels` call using a configured Google API key,
if available.
2. Confirm pricing source:
- Prefer official Gemini API pricing docs if updated for Gemini 3.5
Flash.
- If official pricing is not yet published in developer docs, either:
- use verified LiteLLM metadata from `bun scripts/update_models.ts`, or
- add conservative local metadata with a comment that it must be
revisited once Google publishes official pricing.
3. Confirm thinking semantics:
- Gemini Flash family should expose `minimal`, `low`, `medium`, `high`
on the Google API side.
- Mux should continue exposing user-facing `off`, `low`, `medium`,
`high`, mapping `off` to Google `minimal` for Flash models that do not
support true thinking-off.
**Quality gate:** record the exact source used for model ID, limits,
pricing, and thinking levels in code comments near local metadata or
provider mapping if official docs are incomplete/ambiguous.
## Phase 1 — Repoint the curated model registry
Edit `src/common/constants/knownModels.ts`:
1. Keep the existing `GEMINI_3_FLASH` key by default for a minimal
Option A diff. Add or update its comment to say it tracks the latest
Flash tier. Only rename to `GEMINI_35_FLASH` if `rg "GEMINI_3_FLASH"`
shows negligible references and the resulting diff is smaller/clearer.
2. Set `providerModelId` to the verified API ID, expected:
```ts
providerModelId: "gemini-3.5-flash"
```
3. Keep only the stable alias unless product explicitly wants
version-specific slash aliases:
```ts
aliases: ["gemini-flash"]
```
Users can still select the exact full model string with `/model
google:gemini-3.5-flash`; avoiding a version alias minimizes future
cleanup.
4. Keep tokenizer override unless `ai-tokenizer` has added a better
exact tokenizer:
```ts
tokenizerOverride: "google/gemini-2.5-pro"
```
**Quality gate:** run `bun test
src/common/constants/knownModels.test.ts` after metadata work; alias
uniqueness and token metadata coverage should pass. Add a targeted alias
assertion if not already covered by nearby tests:
`MODEL_ABBREVIATIONS["gemini-flash"] === "google:<verified-id>"` or
`resolveModelAlias("gemini-flash") === "google:<verified-id>"`.
## Phase 2 — Add or refresh token/capability metadata
Preferred path:
1. Run `bun scripts/update_models.ts` before adding manual metadata.
2. Inspect the generated diff. Keep it only if the churn is acceptable
and it adds a **bare** key for the verified model ID, expected
`"gemini-3.5-flash"`, with complete pricing/context/capability fields.
3. If the refresh only adds provider-scoped keys such as
`gemini/gemini-3.5-flash`, `knownModels.test.ts` will still fail for a
`google:` known model; add a bare-key fallback in `models-extra.ts`
instead of relying on scoped-only metadata.
Fallback path if LiteLLM is not updated, creates broad unrelated churn,
or lacks a bare key:
1. Add an entry to `src/common/utils/tokens/models-extra.ts` keyed by
the bare provider model ID, expected `"gemini-3.5-flash"`.
2. Include at minimum:
- `max_input_tokens: 1048576`
- `max_output_tokens: 65536`
- `input_cost_per_token` and `output_cost_per_token` from a verified
pricing source
- `cache_read_input_token_cost` only if the verified pricing source
confirms context-cache pricing
- `litellm_provider: "vertex_ai-language-models"`
- `mode: "chat"`
- `supports_function_calling: true`
- `supports_vision: true`
- `supports_pdf_input: true`
- `supports_reasoning: true`
- `supports_response_schema: true`
- `knowledge_cutoff: "2025-01"`
3. If storing official multimodal support locally, extend the local
`ModelData` interface in `models-extra.ts` to include:
- `supports_audio_input?: boolean`
- `supports_video_input?: boolean`
**Quality gate:** add/adjust
`src/common/utils/tokens/modelStats.test.ts` and
`src/common/utils/ai/modelCapabilities.test.ts` only around behavior
that matters: context size, nonzero pricing, and media support. Avoid
tautological tests that only repeat static prose.
## Phase 3 — Fix Gemini Flash thinking policy and provider mapping
Edit `src/common/utils/thinking/policy.ts`:
1. Replace literal substring detection for Flash with a narrow helper
that matches only verified chat Flash IDs, for example:
```ts
function isGeminiFlashThinkingLevelModelName(modelName: string): boolean
{
return (
modelName === "gemini-3-flash-preview" ||
modelName === "gemini-3.5-flash" ||
modelName === "gemini-3.5-flash-preview" // only keep if this ID is
verified
);
}
```
Use the helper before the generic Gemini 3/3.1 Pro branch. Avoid a broad
regex that accidentally treats `gemini-3.1-flash-lite-preview`, image,
TTS, or other non-chat variants as the same model.
2. Return Mux levels for verified Flash chat models:
```ts
["off", "low", "medium", "high"]
```
3. Keep Pro behavior separate. If current docs now say Gemini 3.1 Pro
supports `medium`, decide whether to broaden Pro in a separate change;
do not conflate that with Gemini 3.5 Flash support unless required by
failing tests or verified product behavior.
Edit `src/common/utils/ai/providerOptions.ts` as a required part of this
change:
1. Reuse the same Flash detection helper, or extract a tiny shared
helper, so policy and provider option mapping cannot drift.
2. The current Google branch sends `thinkingConfig.thinkingLevel` for
`capModelName.includes("gemini-3")`; `gemini-3.5-flash` should still
enter that branch.
3. For verified Flash chat models, map Mux `off` to Google `minimal` and
**do not** set `includeThoughts` for that lowest mode unless verified
docs require it:
```ts
thinkingConfig = { thinkingLevel: "minimal" };
```
Do not rely on omitting `thinkingConfig`; Gemini 3.5 Flash may default
to `medium`, which would make Mux `off` misleading.
4. For Flash `low`, `medium`, and `high`, pass through the level and
keep `includeThoughts: true`:
```ts
thinkingConfig = { includeThoughts: true, thinkingLevel:
effectiveThinking };
```
5. If `xhigh` or `max` somehow reaches provider mapping despite policy
enforcement, defensively map to `high` rather than throwing in the
request path. Add a short comment that policy should clamp before
provider options, but the provider adapter avoids sending invalid Google
values.
**Quality gate:** extend `src/common/utils/thinking/policy.test.ts` and
`src/common/utils/ai/providerOptions.test.ts` to prove:
- `google:gemini-3.5-flash` gets `off/low/medium/high`.
- gateway form like `mux-gateway:google/gemini-3.5-flash` behaves the
same.
- Optional explicit gateway form like
`openrouter:google/gemini-3.5-flash` behaves correctly if current
normalization supports it.
- Flash `off` maps to `{ thinkingConfig: { thinkingLevel: "minimal" } }`
without `includeThoughts` unless docs prove otherwise.
- Flash `medium` maps to `{ thinkingConfig: { includeThoughts: true,
thinkingLevel: "medium" } }`.
- Gemini 3.1 Pro behavior remains unchanged.
- Optional custom model mapping: a provider model entry `mappedToModel:
"google:gemini-3.5-flash"` uses Flash mapping for policy/provider
options.
## Phase 4 — Update docs and generated/model-adjacent outputs
1. Run or update `scripts/gen_docs.ts` output so
`docs/config/models.mdx` lists:
- `Gemini 3.5 Flash`
- `google:<verified-id>`
- alias `gemini-flash`
2. If display output is unexpectedly wrong, add a focused
`src/common/utils/ai/modelDisplay.test.ts` case. The current generic
Gemini formatter likely needs no production change, but a dotted-version
expectation is cheap if touched nearby.
3. Search for stale `KNOWN_MODELS.GEMINI_3_FLASH` references only if the
key is renamed. If the key is kept, no reference churn is expected.
**Quality gate:** do not hand-edit generated docs if an existing
generation script owns the table; run the generator and keep only
expected diffs.
## Phase 5 — Validation
Run targeted tests first:
```bash
bun test src/common/constants/knownModels.test.ts
bun test src/common/utils/thinking/policy.test.ts
bun test src/common/utils/ai/providerOptions.test.ts
bun test src/common/utils/tokens/modelStats.test.ts
bun test src/common/utils/ai/modelCapabilities.test.ts
bun test src/common/utils/ai/modelDisplay.test.ts
```
Then run broader checks:
```bash
make typecheck
make fmt-check
make static-check
```
If `bun scripts/update_models.ts` produces broad generated churn,
inspect whether it is acceptable; if too broad, prefer `models-extra.ts`
for this targeted launch support.
## Phase 6 — Dogfooding plan
Because this is a model-selection/provider behavior change, dogfood in
the desktop app with a configured Google provider.
1. Start Mux:
```bash
make dev
```
2. In Settings → Providers, confirm Google is configured and enabled.
3. Use the model selector and confirm:
- `Gemini 3.5 Flash` appears.
- `gemini-flash` resolves to `google:<verified-id>`.
- old `Gemini 3 Flash Preview` is no longer the curated `gemini-flash`
target.
4. Send smoke prompts at all Flash thinking levels:
- `off` / numeric `0`
- `low`
- `medium`
- `high`
5. Use `agent-browser` to capture reviewer evidence:
- Screenshot of the model selector showing Gemini 3.5 Flash.
- Screenshot of a successful response using `gemini-flash`.
- Screenshot of thinking-level control or slash-command usage.
- Video recording of selecting the model and sending one prompt.
6. Multimodal smoke check if provider/API key allows it:
- Attach a small image or PDF and verify the send path is allowed.
- Capture a screenshot of the attachment flow and successful response.
## Acceptance criteria
- `gemini-flash` resolves to the verified Gemini 3.5 Flash Google model
ID.
- No new version-specific alias is added unless product explicitly asks
for it.
- The first-class model selector lists Gemini 3.5 Flash when
Google/direct or configured gateway routing makes it available.
- The known-model metadata invariant passes with a bare metadata key for
the verified provider model ID in `models.json` or `models-extra.ts`.
- Token meter/context warnings use Gemini 3.5 Flash limits and costs.
- Gemini 3.5 Flash thinking policy exposes Mux levels
`off/low/medium/high`.
- Provider options translate Mux `off` to Google `minimal` for Gemini
3.5 Flash instead of accidentally using the API default, and omit
`includeThoughts` for this lowest mode unless docs prove otherwise.
- Provider options pass Flash `low/medium/high` through with
`includeThoughts: true`.
- Existing Gemini Pro behavior is unchanged unless explicitly verified
and intentionally updated.
- Docs table reflects Gemini 3.5 Flash.
- Targeted tests, typecheck, fmt-check, and static-check pass.
- Dogfooding screenshots and a video recording are captured for reviewer
verification.
## Risks and mitigations
- **API model ID ambiguity:** block implementation until exact ID is
verified from official API/AI Studio, not inferred only from marketing
copy.
- **Pricing docs lag:** prefer LiteLLM refresh if available; otherwise
add local metadata with a clear source/revisit comment. Do not commit
press/blog-derived pricing unless official API pricing, LiteLLM, or
another trusted provider metadata source confirms it.
- **Thinking-level drift:** keep tests focused on observed provider
behavior, especially `off` → `minimal` and absence of `includeThoughts`
for the lowest mode unless docs require it.
- **Overbroad Flash matching:** use a narrow verified-ID helper so
image, TTS, Flash Lite, or future non-chat variants do not inherit
chat-model thinking behavior accidentally.
- **Generated metadata churn:** if `models.json` refresh touches many
unrelated entries or lacks a bare key, use `models-extra.ts` for a
surgical release.
- **Alias compatibility:** existing users selecting
`google:gemini-3-flash-preview` explicitly can still use it as a custom
model; only the curated `gemini-flash` alias changes.
</details>
---
_Generated with [\](https://github.com/coder/mux) • Model: \ • Thinking:
\ • Cost: \_
<!-- mux-attribution: model=openai:gpt-5.5 thinking=xhigh costs=47.10
-->1 parent 6e9a0c0 commit 58a06c3
12 files changed
Lines changed: 295 additions & 18 deletions
File tree
- docs/config
- src
- common
- constants
- utils
- ai
- thinking
- tokens
- node/services/agentSkills
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
25 | 25 | | |
26 | 26 | | |
27 | 27 | | |
28 | | - | |
| 28 | + | |
29 | 29 | | |
30 | 30 | | |
31 | 31 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
29 | 29 | | |
30 | 30 | | |
31 | 31 | | |
| 32 | + | |
| 33 | + | |
| 34 | + | |
| 35 | + | |
32 | 36 | | |
33 | 37 | | |
34 | 38 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
115 | 115 | | |
116 | 116 | | |
117 | 117 | | |
118 | | - | |
| 118 | + | |
| 119 | + | |
119 | 120 | | |
120 | | - | |
| 121 | + | |
121 | 122 | | |
122 | 123 | | |
123 | 124 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
47 | 47 | | |
48 | 48 | | |
49 | 49 | | |
| 50 | + | |
| 51 | + | |
| 52 | + | |
| 53 | + | |
| 54 | + | |
| 55 | + | |
| 56 | + | |
| 57 | + | |
| 58 | + | |
50 | 59 | | |
51 | 60 | | |
52 | 61 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
45 | 45 | | |
46 | 46 | | |
47 | 47 | | |
| 48 | + | |
48 | 49 | | |
49 | 50 | | |
50 | 51 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
749 | 749 | | |
750 | 750 | | |
751 | 751 | | |
| 752 | + | |
| 753 | + | |
| 754 | + | |
| 755 | + | |
| 756 | + | |
| 757 | + | |
| 758 | + | |
| 759 | + | |
| 760 | + | |
| 761 | + | |
| 762 | + | |
| 763 | + | |
| 764 | + | |
| 765 | + | |
| 766 | + | |
| 767 | + | |
| 768 | + | |
| 769 | + | |
| 770 | + | |
| 771 | + | |
| 772 | + | |
| 773 | + | |
| 774 | + | |
| 775 | + | |
| 776 | + | |
| 777 | + | |
| 778 | + | |
| 779 | + | |
| 780 | + | |
| 781 | + | |
| 782 | + | |
| 783 | + | |
| 784 | + | |
| 785 | + | |
| 786 | + | |
| 787 | + | |
| 788 | + | |
| 789 | + | |
| 790 | + | |
| 791 | + | |
| 792 | + | |
| 793 | + | |
| 794 | + | |
| 795 | + | |
| 796 | + | |
| 797 | + | |
| 798 | + | |
| 799 | + | |
| 800 | + | |
| 801 | + | |
| 802 | + | |
| 803 | + | |
| 804 | + | |
| 805 | + | |
| 806 | + | |
| 807 | + | |
| 808 | + | |
| 809 | + | |
| 810 | + | |
| 811 | + | |
| 812 | + | |
| 813 | + | |
| 814 | + | |
| 815 | + | |
| 816 | + | |
| 817 | + | |
| 818 | + | |
| 819 | + | |
| 820 | + | |
| 821 | + | |
| 822 | + | |
| 823 | + | |
| 824 | + | |
| 825 | + | |
| 826 | + | |
| 827 | + | |
| 828 | + | |
| 829 | + | |
| 830 | + | |
| 831 | + | |
| 832 | + | |
| 833 | + | |
| 834 | + | |
| 835 | + | |
| 836 | + | |
| 837 | + | |
| 838 | + | |
| 839 | + | |
| 840 | + | |
| 841 | + | |
| 842 | + | |
| 843 | + | |
| 844 | + | |
| 845 | + | |
| 846 | + | |
| 847 | + | |
| 848 | + | |
| 849 | + | |
| 850 | + | |
| 851 | + | |
| 852 | + | |
| 853 | + | |
| 854 | + | |
| 855 | + | |
| 856 | + | |
| 857 | + | |
| 858 | + | |
| 859 | + | |
| 860 | + | |
| 861 | + | |
| 862 | + | |
| 863 | + | |
| 864 | + | |
| 865 | + | |
| 866 | + | |
| 867 | + | |
| 868 | + | |
| 869 | + | |
| 870 | + | |
| 871 | + | |
| 872 | + | |
| 873 | + | |
| 874 | + | |
| 875 | + | |
| 876 | + | |
| 877 | + | |
| 878 | + | |
| 879 | + | |
| 880 | + | |
| 881 | + | |
| 882 | + | |
| 883 | + | |
| 884 | + | |
| 885 | + | |
| 886 | + | |
| 887 | + | |
| 888 | + | |
| 889 | + | |
| 890 | + | |
| 891 | + | |
| 892 | + | |
| 893 | + | |
| 894 | + | |
| 895 | + | |
| 896 | + | |
| 897 | + | |
| 898 | + | |
| 899 | + | |
| 900 | + | |
752 | 901 | | |
753 | 902 | | |
754 | 903 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
23 | 23 | | |
24 | 24 | | |
25 | 25 | | |
| 26 | + | |
26 | 27 | | |
27 | 28 | | |
28 | 29 | | |
| |||
409 | 410 | | |
410 | 411 | | |
411 | 412 | | |
412 | | - | |
| 413 | + | |
| 414 | + | |
| 415 | + | |
413 | 416 | | |
414 | 417 | | |
415 | | - | |
| 418 | + | |
| 419 | + | |
| 420 | + | |
| 421 | + | |
| 422 | + | |
416 | 423 | | |
417 | 424 | | |
418 | 425 | | |
419 | 426 | | |
420 | | - | |
421 | | - | |
422 | | - | |
423 | | - | |
424 | | - | |
425 | | - | |
426 | | - | |
427 | | - | |
| 427 | + | |
| 428 | + | |
| 429 | + | |
| 430 | + | |
| 431 | + | |
428 | 432 | | |
429 | 433 | | |
430 | 434 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
1 | 1 | | |
2 | | - | |
| 2 | + | |
| 3 | + | |
| 4 | + | |
| 5 | + | |
| 6 | + | |
| 7 | + | |
3 | 8 | | |
4 | 9 | | |
5 | 10 | | |
| |||
386 | 391 | | |
387 | 392 | | |
388 | 393 | | |
| 394 | + | |
| 395 | + | |
| 396 | + | |
| 397 | + | |
| 398 | + | |
| 399 | + | |
| 400 | + | |
| 401 | + | |
| 402 | + | |
| 403 | + | |
| 404 | + | |
| 405 | + | |
| 406 | + | |
| 407 | + | |
| 408 | + | |
| 409 | + | |
| 410 | + | |
| 411 | + | |
| 412 | + | |
| 413 | + | |
| 414 | + | |
| 415 | + | |
| 416 | + | |
| 417 | + | |
| 418 | + | |
| 419 | + | |
| 420 | + | |
| 421 | + | |
| 422 | + | |
| 423 | + | |
| 424 | + | |
| 425 | + | |
| 426 | + | |
| 427 | + | |
| 428 | + | |
| 429 | + | |
| 430 | + | |
| 431 | + | |
| 432 | + | |
| 433 | + | |
| 434 | + | |
| 435 | + | |
| 436 | + | |
| 437 | + | |
| 438 | + | |
| 439 | + | |
| 440 | + | |
| 441 | + | |
| 442 | + | |
389 | 443 | | |
390 | 444 | | |
391 | 445 | | |
| |||
411 | 465 | | |
412 | 466 | | |
413 | 467 | | |
| 468 | + | |
| 469 | + | |
| 470 | + | |
| 471 | + | |
| 472 | + | |
| 473 | + | |
| 474 | + | |
414 | 475 | | |
415 | 476 | | |
416 | 477 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
25 | 25 | | |
26 | 26 | | |
27 | 27 | | |
| 28 | + | |
| 29 | + | |
| 30 | + | |
| 31 | + | |
| 32 | + | |
| 33 | + | |
| 34 | + | |
| 35 | + | |
| 36 | + | |
| 37 | + | |
| 38 | + | |
| 39 | + | |
| 40 | + | |
| 41 | + | |
28 | 42 | | |
29 | 43 | | |
30 | 44 | | |
| |||
36 | 50 | | |
37 | 51 | | |
38 | 52 | | |
39 | | - | |
| 53 | + | |
| 54 | + | |
40 | 55 | | |
41 | 56 | | |
42 | 57 | | |
| |||
95 | 110 | | |
96 | 111 | | |
97 | 112 | | |
98 | | - | |
99 | | - | |
| 113 | + | |
| 114 | + | |
100 | 115 | | |
101 | 116 | | |
102 | 117 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
43 | 43 | | |
44 | 44 | | |
45 | 45 | | |
| 46 | + | |
| 47 | + | |
| 48 | + | |
| 49 | + | |
| 50 | + | |
| 51 | + | |
| 52 | + | |
| 53 | + | |
| 54 | + | |
46 | 55 | | |
47 | 56 | | |
48 | 57 | | |
| |||
0 commit comments