I'm currently trying to write a codemod that will remove all the $ReadOnly<T>
generic from a TypeScript codebase, keeping only T
(T
being an object/union)
So far, this is what I came up with
module.exports = (fileInfo, api) => {
const j = api.jscodeshift
const source = j(fileInfo.source)
source
.find(j.TSTypeAliasDeclaration)
.find(j.TSTypeReference)
.filter(x => {
return x.value.typeName.name === '$ReadOnly' && x.value.typeParameters.params[0].type === 'TSTypeLiteral'
})
.replaceWith(nodePath => {
const members = []
nodePath.value.typeParameters.params[0].members.forEach(x => {
members.push(j.tsPropertySignature(x.key, x.typeAnnotation))
})
return j.tsTypeLiteral(members)
})
return source
.toSource()
}
The idea is to modify something like this:
export type MyType = $ReadOnly<{
someProps: string,
}>
To that:
export type MyType = {
someProps: string,
}
Unfortunately, this is what I end up with, with a duplicate type
keyword:
export type type MyType = {
someProps: string,
}
Any idea what could've gone wrong here?
It can be written in a declarative way with putout code transformer (you can try it in putout.cloudcmd.io):
This is replacer type of
putout plugin
.It will change input:
Into output:
Putout
has a simple test runner so you can test yourcodemod
with predefined fixtures using:To run this
codemod
save it into~/.putout
and run on your codebase with:Or save in a directory
your-project/codemods
and run: