Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: enable eraseable syntax #31

Merged
merged 11 commits into from
Mar 25, 2025
Merged

feat: enable eraseable syntax #31

merged 11 commits into from
Mar 25, 2025

Conversation

nazarhussain
Copy link

Motivation

For future adoption of Typescript support in NodeJS, update the types.

Description

  • Enable eraseable syntax only
  • Update all the enums
  • Update all constructors.

Steps to test or reproduce

  • Run all tests

@nazarhussain nazarhussain changed the title feat: Enable eraseable syntax feat: enable eraseable syntax Mar 10, 2025
@nazarhussain
Copy link
Author

nazarhussain commented Mar 10, 2025

To enable eraseable syntax we have to get rid of enum from Typescript files. Which requires to split those enums into types and values respectively.

So we need to stick with some pattern for consistency as an alternate.

Look at the example enum below.

enum OutlierSensitivity = {
  Mild = 1.5,
  Strict = 3.0,
}

type EnumLike<T> = T[keyof T];

So now we have split this information into type and values.

Pattern 1 - capital case enum + camelCase values

const outlierSensitivity = {
  Mild: 1.5,
  Strict: 3.0,
} as const;

type OutlierSensitivity = EnumLike<typeof outlierSensitivity>;
  1. More related pattern to our code
  2. Most time may conflict with local variables

Pattern 2 - Capital case values + Options suffix type

const OutlierSensitivity = {
  Mild: 1.5,
  Strict: 3.0,
} as const;

type OutlierSensitivityOptions = EnumLike<typeof outlierSensitivity>;

Pattern 3 - Capital case values + E prefix type

const OutlierSensitivity = {
  Mild: 1.5,
  Strict: 3.0,
} as const;

type EOutlierSensitivity = EnumLike<typeof outlierSensitivity>;

Please suggest which one you prefer or suggest any other pattern you like.

Copy link

github-actions bot commented Mar 10, 2025

Performance Report

✔️ no performance regression detected

Full benchmark results
Benchmark suite Current: 034ecf0 Previous: null Ratio
sum array with raw for loop 933.67 us/op
sum array with reduce 9.9933 ms/op
sum array with reduce beforeEach 81.795 us/op
sum array with reduce before beforeEach 81.450 us/op
sum array with reduce high threshold 10.006 ms/op
sum array with reduce no threshold 10.250 ms/op

by benchmarkbot/action

@wemeetagain
Copy link
Member

imo something like pattern 2 is good with me. maybe with the type having Option appended instead of Options.

then we have code like:

const OutlierSensitivity = {
  Mild: 1.5,
  Strict: 3.0,
} as const;

type OutlierSensitivityOption = EnumLike<typeof outlierSensitivity>;


function average(data: number[], sensitivity: OutlierSensitivityOption): boolean {
  ...
}

const a = average([...], OutlierSensitivity.Mild);

@matthewkeil
Copy link
Member

imo something like pattern 2 is good with me. maybe with the type having Option appended instead of Options.

I tend to agree with this pattern but would suggest appending Type instead of Option

@nazarhussain
Copy link
Author

nazarhussain commented Mar 11, 2025

The option 2 is good, the only problem is naming convention. We stick with the camelCase for variable names. If we use option 2 we have to allow the PascalCase as well then over the code will have mixed notation. One workaround could be to have a fix convention that all enum values variables will start with E then we can configure the linter for it.

@wemeetagain
Copy link
Member

wemeetagain commented Mar 12, 2025

The reason we'd keep pascal case for "enum" variable names is because they're treated like a namespace (or rather, like an enum lol). They aren't treated as normal variables. Similar reason we uppercase constants.
The casing is a cue that the data has a different meaning from the other casing.

const Sensitivity = {Mild: 1, Strict: 3} as const;
type SensitivityOption = EnumOption<typeof Sensitivity>;

const sensitivity = Sensitivity.Mild;

@KatyaRyazantseva
Copy link

EOutlierSensitivity associates with errors in my experience. Pattern 2 seems to work best.

We already have random Type combos in the code. A couple of examples:

export type TypeJson<T> = {
  toJson: (data: T) => unknown; // server
  fromJson: (data: unknown) => T; // client
};
export type LodestarThreadType = "main" | "network" | "discv5";

How about Enum postfix or prefix? I'd prefer prefix.

type OutlierSensitivityEnum = EnumLike<typeof outlierSensitivity>;
// or
type EnumOutlierSensitivity = EnumLike<typeof outlierSensitivity>;

@ensi321
Copy link

ensi321 commented Mar 12, 2025

Might be irrelevant but I want to mention this example in Google Style Guide https://google.github.io/styleguide/tsguide.html#identifiers-constants

const UNIT_SUFFIXES = {
  'milliseconds': 'ms',
  'seconds': 's',
};

However the purpose in this example is to define this as constant, not for replacing an enum. Among the three options, I prefer 2, since we already have a convention of having PascalCase of const denoting namespace that is widely used in Lodestar.

My 2 cents will be to rename EnumLike to EnumValue and for the suffix, no strong opinion could be Option, Value, Type.

Strong opinion against using single character to denote things eg. E, I

@nazarhussain
Copy link
Author

Discussion for enum naming is summarized here ChainSafe/lodestar#7574

Copy link
Member

@matthewkeil matthewkeil left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just one thing and the question but looks good overall

@@ -23,6 +23,7 @@
"esModuleInterop": true,
// Babel builds source, Typescript is used only for .d.ts files
"incremental": true,
"preserveWatchOutput": true
"preserveWatchOutput": true,
"erasableSyntaxOnly": true
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just to check this does not alter the output it just enforces that the syntax meets the erasable syntax specs?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The output will just be changed for the code that we changed. There will be no implicit change in the compiled JS.

// Try first latest commit in branch
return await provider.readLatestInBranch(compareWith.branch);
}
}
}

export function renderCompareWith(compareWith: CompareWith): string {
export function renderCompareWith(compareWith: CompareWithType): string {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

just noting here that while adding a suffix to the type is ok, I don't think using Type as suffix is ideal since we use that for ssz types already, so maybe Options / Opts might be better?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am fine with either, it's just that we had a consensus on this pattern earlier ChainSafe/lodestar#7574

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the pattern would still be the same, just different suffix, see comment from NC, he voted for option 4 but voiced something similar here ChainSafe/lodestar#7574 (comment)

we are probably fine with using Type as suffix, just wanna point this out so we consider it at least

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am fine with either suffix, as long as everyone is aligned with it.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please no Options.... its a "type"....

Copy link
Member

@nflaig nflaig left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

Copy link
Member

@matthewkeil matthewkeil left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🙈 LGTM!!! 🥸

GaCache = "GaCache",
S3 = "S3",
}
export const HistoryProviderTypeEnum = {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see you just did the minimal diff here. But consider removing Type from the const and type

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was afraid to change that considering that may trigger another discussion. :)

Copy link
Member

@wemeetagain wemeetagain left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

lgtm

@nazarhussain nazarhussain merged commit 77c872d into main Mar 25, 2025
12 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants