La repo per le definizioni di tipi Typescript di alta qualità.
Puoi leggere questo README anche in Spagnolo, Coreano, Russo, Cinese, Portoghese e Giapponese!
Link per il manuale dell'amministratore
Questa sezione tiene traccia della salute della repo e del processo di pubblicazione. Può tornare utile per i contributori che stanno avendo problemi con le loro PR e package.
- Ultima build type-checked/linted pulita:
- Tutti i package sono sottoposti a controllo dei tipi e linting con typescript@next:
- Tutti i package vengono pubblicati su npm in under an hour:
- Il bot di Typescript è stato attivo su Definitely Typed
- Aggiornamenti dello stato dell'infrastruttura attuale
Se qualcosa sembra sbagliato o una qualunque delle cose qui in alto fallisce, fatecelo sapere sul canale di Definitely Typed nel server discord della comunità di Typescript:
Leggi il manuale di TypeScript.
Il metodo preferito per installarli è usando npm.
Ad esempio:
npm install --save-dev @types/node
I tipi dovrebbero poi essere inclusi automaticamente dal complilatore.
Potresti dover aggiungere un riferimento types
se non stai usando i moduli:
/// <reference types="node" />
Approfondisci leggendo il manuale.
Per un package "pippo", i tipi associati saranno disponibili sul package "@types/pippo". Se non riesci a trovare i tipi di un package, cercalo su TypeSearch.
Se ancora non riesci a trovarlo, vuol dire che sono inclusi nel package stesso.
Di solito vengono specificati nel campo "types"
o "typings"
nel package.json
;
se vuoi puoi semplicemente cercare per dei file ".d.ts" nel package ed includerli manualmente con /// <reference path="" />
.
Definitely Typed testa packages solo su versioni di TypeScript che hanno meno di due anni.
Attualmente vengono testate le versioni 3.7 e successive.
Se stai usando una versiond di TypeScript tra 2.0 e la 3.6, puoi ancora provare ad installare i package @types
, in quanto la maggior parte di questi non usano le funzionalità più all'avanguardia di Typescript.
Non c'è comunque nessuna garanzia che funzioneranno.
Ecco le informazioni riguardanti le versioni supportate:
Versione | Data rilascio | Data fine supporto |
---|---|---|
2.8 | Marzo 2018 | Marzo 2020 |
2.9 | Maggio 2018 | Maggio 2020 |
3.0 | Luglio 2018 | Agosto 2020 |
3.1 | Settembre 2018 | Settembre 2020 |
3.2 | Novembre 2018 | Novembre 2020 |
3.3 | Gennaio 2019 | Gennaio 2021 |
3.4 | Marzo 2019 | Marzo 2021 |
3.5 | Maggio 2019 | Maggio 2021 |
3.6 | Agosto 2019 | Agosto 2021 |
3.7 | Novembre 2019 | Novembre 2021 |
3.8 | Febbraio 2020 | Febbraio 2022 |
3.9 | Maggio 2020 | Maggio 2022 |
4.0 | Agosto 2020 | Agosto 2022 |
4.1 | Novembre 2020 | Novembre 2022 |
4.2 | Febbraio 2021 | Febbraio 2023 |
4.3 | Maggio 2021 | Maggio 2023 |
4.4 | Agosto 2021 | Agosto 2023 |
I package @types
hanno dei tag per la versione di TypeScript che supportano, quindi di solito puoi installare vecchi package che sono più vecchi di due anni.
Ad esempio se esegui npm dist-tags @types/react
, vedrai che TypeScript 2.5 potrà usare i tipi per [email protected], mentre TypeScript 2.6 e 2.7 potranno usare i tipi per [email protected]:
Tag | Versione |
---|---|
ultima | 16.9.23 |
ts2.0 | 15.0.1 |
... | ... |
ts2.5 | 16.0.36 |
ts2.6 | 16.4.7 |
ts2.7 | 16.4.7 |
... | ... |
- Scarica manualmente la branch
master
di questa repo e collocala nel tuo progetto. Typings(usa altre alternative migliori, in quanto typings è deprecata)NuGet(usa altre alternative, in quanto la pubblicazione di tipi su nuget DT non è più possibile)
Potresti dover aggiungere manualmente i riferimenti.
Definitely Typed ha successo e può funzionare solamente grazie a contributori come te!
Prima di condividere il tuo contributo col mondo, provalo tu stesso.
Per testare localmente le tue modifiche, puoi usare module augmentation per estendere i tipi esistenti del modulo DT su cui vuoi lavorare.
Altrimenti, puoi modificare i tipi direttamente nel node_modules/@types/foo/index.d.ts
per validare le tue modifiche, per poi riportarle su questa repo seguendo i passi che troverai scritti più in basso.
Aggiungi al tuo tsconfig.json
:
"baseUrl": "types",
"typeRoots": ["types"],
Crea un types/foo/index.d.ts
contenente le dichiarazioni per il tuo modulo "pippo".
Ora dovresti essere in grado di importare "pippo"
nel tuo codice, con i tipi riferiti alle dichiarazioni che hai appena creato.
Poi fai una build ed esegui il codice per essere sicuro che le definizioni dei tipi corrispondano effettivamente a ciò che capita a runtime.
Una volta che hai testato le definizioni su del codice reale, fai una pull request, poi segui le istruzioni per [modificare un package preesistente](#modificare-un-package-preesistente] o creare un nuovo package.
Una volta che hai testato il tuo package, puoi condividerlo su Definitely Typed.
Inanzitutto, sdoppia questa repo, clonala, installa node ed esegui npm install
. Se stai usando npm
v7 devi aggiungere --legacy-peer-deps
al comando.
Utilizziamo un bot per far sì che un gran numero di pull request su Definitely Typed possano essere gestite interamente in modo autonomo. Puoi scoprire di più a riguardo di come e perchè qui.
Ecco qui un'immagine che mostra il ciclo vitale di una pull request su Definitely Typed.
Puoi clonare l'intera repo come di consuetudine che però è molto grande ed include numerose cartelle di package DT. Ci vorrà quindi del tempo per clonarla e può essere un procedimento inutilmente lungo.
Per clonare la repo in modo più agibile, clonando solo i package DT per te rilevanti, pui usare le funzionalità sparse-checkout
, --filter
ed --depth
di git.
⚠️ Richiede almeno la versione di git 2.27.0, che solitamente è più recente di quelle installate di default. Per le versioni più vecchie sono disponibili procedure più complesse, che non trattiamo qui.
git clone --sparse --filter=blob:none --depth=1 <forkedUrl>
--sparse
initializes the sparse-checkout file so the working directory starts with only the files in the root of the repository.--filter=blob:none
will exclude files, fetching them only as needed.--depth=1
will further improve clone speed by truncating commit history, but it may cause issues as summarized here.
git sparse-checkout add types/<type> types/<dependency-type> ...
cd types/<package to edit>
- Fai le tue modifiche. Ricorda di testle. Se fai modifiche essenziali, non ti dimenticare di aggiornare il major della versione.
Quando crei una pull request ad un package che esiste già, dt-bot
dovrebbe @menzionare gli autori precedenti.
Se non lo fa, puoi farlo direttamente tu nel commento associato alla pull request.
Se sei l'autore della libreria ed il tuo package è scritto in TypeScript, [includi i file di dichiarazione generati automaticamente] (https://www.typescriptlang.org/docs/handbook/declaration-files/publishing.html) nel tuo package invece di pubblicarli su Definitely Typed.
Se stai aggiungendo i tipo per un package npm, crea una cartella con lo stesso nome.
Se il package a cui stai aggiungendo i tipi non è su npm, assicurati che il nome che scegli non entri in conflitto con quello di un package npm.
(Puoi usare npm info <nome-package>
per verificare l'esistenza del package <nome-package
).
Il tuo package dovrebbe avere questa struttura:
File | Scopo |
---|---|
index.d.ts |
Contiene le dichiarazioni dei tipi del package. |
<nome-package>-tests.ts |
Contiene codice di esempio con test delle dichiarazioni dei tipi. Se il codice non funziona anche se viene traspilato da tsc senza errori. |
tsconfig.json |
Ti permette di eseguire tsc all'interno del package. |
tslint.json |
Abilita il linting. |
Generali eseguento npx dts-gen --dt --name <mio-package> --template module
se hai npm ≥ 5.2.0, altrimenti npm install -g dts-gen
and dts-gen --dt --name <my-package> --template module
.
Leggi tutte le opzioni su dts-gen.
Se hai file .d.ts
oltre all'index.d.ts
, assicurati che siano referenziati o nell'index.d.ts
o nei test.
I membri di Definitely Typed controllano continuamente le nuove pull request, perciò sii al corrente che un alto numero di pull request potrebbe rallentarci il lavoro.
Per l'esempio di un buon package, guarda base64-js.
Quando un package include i suoi dichiarazioni, esse dovrebbero venire rimosse da Definitely Typed per evitare di far confusione.
Puoi rimuoverli eseguendo npm run not-needed -- <nome-package> <versione> [<nome-libreria>]
.
<nome-package>
: È il nome della cartella da rimuovere.<versione>
: Verrà pubblicato uno stab su@types/<nome-package>
con questa versione. Dev'essere più alto della versione attualmente pubblicata e deve essere una versione di<ome-libreria>
su npm.<nome-libreria>
: Nome del package npm che sostituisce il package DT. Solitamente è identioc a<nome-package>
e puoi ometterlo.
Qualunque altro package di Definitely Typed che si riferisce ad un altro package DT eliminato dovrebbe venir aggiornato facendolo riferire ai tipi inclusi nel package stesso.
Puoi farlo controllando gli errori che escono eseguendo npm run test-all
.
Per correggere gli errori, aggiungi un package.json
con "dependencies": { "<nome-libreria>": "x.y.z" }
.
Ad esempio:
{
"private": true,
"dependencies": {
"<nome-libreria>": "^2.6.0"
}
}
Quando aggiungi un package.json
a dei package che dipendono da <nome-libreria>
, devi aprire anche una pull request per aggiungere <nome-libreria>
ad allowedPackageJsonDependencies.txt su DefinitelyTyped-tools.
Se un package non è mai stato su Definitely Typed, non c'è bisogno che venga aggiunto a notNeededPackages.json
.
Testa le tue modifiche eseguendo npm test <package da testare>
dove <package da testare>
è il nome del tuo package.
Lo script usa dtslint per eseguire il compilatore TypeScript sui tuoi file dts.
Se stai aggiungendo i tipi ad un package npm, crea una cartella con lo stesso nome.
Se invece non li stai aggiungendo ad un package npm, assicurati prima che non esista un package npm con lo stesso nome, per evitare futuri conflitti.
(Puoi usare npm info <nome-package>
per controllare l'esistenza di un package <nome-package>
).
Se un package che non è di npm è in conflitto con un package npm, prova ad aggiungere -browser alla fine del nome, ottenendo quindi <mio-package>-browser
Ci dev'essere un file <my-package>-tests.ts
, che viene considerato il tuo file di test, assieme ad ogni file *.ts
che importa.
Se non vedi un file di test nella cartella del modulo, crea tu un file <nome-package>-test.ts
.
Questi file sono usati per validare l'API esportata dai file *.d.ts
che vengono pubblicati come @types/<nome-package>
.
Ad esempio, questo codice cambia una funzione nel file .d.ts
, aggiungendoci un nuovo parametro:
index.d.ts
:
- export function twoslash(body: string): string
+ export function twoslash(body: string, config?: { version: string }): string
<mio-package>-tests.ts
:
import {twoslash} from "./"
// $ExpectType string
const result = twoslash("//")
+ // Handle options param
+ const resultWithOptions = twoslash("//", { version: "3.7" })
+ // When the param is incorrect
+ // $ExpectError
+ const resultWithOptions = twoslash("//", { })
Se ti stai chiedendo da dove cominciare per fare i test, gli esempi nel README del modulo sono un buon punto da dove partire.
Puoi validare le tue modifiche con npm test <package da testare>
nella root di questa repo, che prende in considerazione i file cambiati.
Usa $ExpectType
per asserire che un'espressione è del tipo dato e $ExpectError
per asserire un errore di compilazione. Ad esempio:
// $ExpectType void
f(1);
// $ExpectError
f("one");
Per maggiori dettagli, leggi il readme di dtslint.
Il file di configurazione del linter, tslint.json
, dovrebbe contenere { "extends": "dtslint/dt.json" }
e nessun'altra regola.
Se per qualche ragione qualche regola necessita di essere disabilitata, disabilitala solo per la riga di codice in cui dovrebbe esserlo usando // tslint:disable-next-line:[ruleName]
e non disabilitandola per tutto il package.
tsconfig.json
dovrebbe avere noImplicitAny
, noImplicitThis
, strictNullChecks
e strictFunctionTypes
settati a true
.
Potresti dover editare tsconfig.json
per aggiungere nuovi file di test, per aggiungere le opzioni di traspilazione "target": "es6"
(necessario per funzioni asincrone), "lib"
o "jsx"
.
Solitamente non ne avrai bisogno.
Un package.json
potrebbe essere incluso per specificare le dipendenze che non sono altri @types
package.
Pikaday ne è un buon esempio.
Anche se stai scrivendo un tuo package.json
, upoi solo specificare dipendenze; altri campi come "description"
non sono consentiti.
Puotresti anche aver bisogno di aggiungere una dipendenza alla lista dei package consentiti.
Questa lista è aggiornata da persone, che ci danno l'opportunità di assicurarci che i package @types
non dipendono da package rischiosi.
Nel raro caso in cui un package @types
viene eliminato a favore dei tipi inclusi nel package npm analogo E avessi bisogno di dipendere dal vecchio package @types
, puoi aggiungere una dipendenza ad un package @types
.
Assicurati di spiegare tutto ciò quando lo aggiungi alla lista dei package consentiti cosicchè gli altri possano capire cosa stai facendo.
Se un file non è nè testato nè riferito nell'index.d.ts
, aggiungilo in un file chiamato OTHER_FILES.txt
. Questo file è una lista di altri file che serve che siano inclusi nel package DT. Elencali uno per riga.
- Inanzitutto segui i consigli nel manuale.
- Formattazione: Usa 4 spazi. Prettier è abilitato su questa repo, quindi puoi eseguire
npm run prettier -- --write path/to/package/**/*.ts
. Quando usi le assertion, aggiungi// prettier-ignore
per marcare le linee di codice da escludere quando si fa la formattazione:// prettier-ignore const incompleteThemeColorModes: Theme = { colors: { modes: { papaya: { // $ExpectError
function sum(nums: number[]): number
: UsaReadonlyArray
se una funzione non modifica i suoi parametri.interface Foo { new(): Foo; }
: Definisce un tipo di oggetto su cui si può fare new. Probabilmente ciò che vuoi èdeclare class Foo { constructor(); }
.const Class: { new(): IClass; }
: Preferire sempre la dichiarazione di una classeclass Class { constructor(); }
al posto di una costante new.getMeAT<T>(): T
: Se il parametro di un tipo non appare nei tipi di un parametro, ciò che vuoi non è una vera funzione generica. È consigliato usare una vera asserzione di tipo, ad esempiogetMeAT() as number
. Esempio dove un parametro è accettabile:function id<T>(value: T): T;
. Esempio in cui non lo è:function parseJson<T>(json: string): T;
. Eccezione:new Map<string, number>()
va bene.- Usare i tipi
Function
eObject
non è quasi mai una buona idea. Nel 99% si può specificare un tipo più preciso. Alcuni esempi sono(x: number) => number
per le funzioni e{ x: number, y: number }
per gli oggetti. Se non si è sicuri del tipo,any
è la scelta migliore, nonObject
. Se l'unica cosa di cui si sa è che il tipo è un oggetto di qualche tipo, usa il tipoobject
, nonObject
o{ [key: string]: any }
. var foo: string | any
: Quandoany
è usato in un tipo disgiuntipo, il tipo risultanto rimaneany
. Quindi mentre la porzionestring
di questo tipo potrebbe sembrare utile, in realtà non offre nessuna precisazione rispetto ad un banalissimoany
. In funzione delle tue intenzioni, delle alternative accettabili sonoany
,string
ostring | object
.
DT ha il concetto di "Proprietari delle definizioni", che sono coloro i quali vogliono mantenere la qualità delle definizioni dei tipi di un certo modulo.
- Aggiungerti da solo farà sì che tu venga notificato (tramite il tuo nome utente GitHub) ogni volta che qualcuno fa una pull request o un issue su quel package.
- Le tue PR review avranno precedenza maggiore di quelle dei bot che mantengono questa repo.
- I mantenitori di DT stanno ponendo la loro fiducia sui proprietari delle definizioni per mantenere un ecosistema stabile, quindi non aggiungerti senza sapere quello che fai.
Per aggiungerti come pprietario delle definizioni:
- Aggiungi il tuo nome alla fine della riga, come su
// Definitions by: Alice <https://github.com/alice>, Bob <https://github.com/bob>
. - Se ci sono più perone, puoi separarlo in più righe:
// Definitions by: Alice <https://github.com/alice> // Bob <https://github.com/bob> // Steve <https://github.com/steve> // John <https://github.com/john>
Una volta alla settimana i proprietari delle definizioni saranno sincronizzati nel file .github/CODEOWNERS che è la nostra fonte di verità.
Il branch master
viene pubblicato automaticamente nello scope @types
di npm grazie a DefinitelyTyped-tools.
Dipende, ma alla maggior parte delle pull request viene fatto un merge in meno di una settimana. Ad altre pull request può essere fatto il merge dai proprietari di un modulo e ciò può avvenire molto più velocemente. It depends, but most pull requests will be merged within a week.
All'incirca:
Le pull request che cambiano solo i tipi di un modulo e che hanno i test corrispondenti vengono vengono accettati più velocemente
Alle pull request che sono state approvate da un'autore presente nella lista dei proprietari vengono di solito fatti i merge più rapidamente; pull request per nuove definizioni ci mettono più tempo in quando hanno bisogno di più controlli da parte dei mantenitori. Ogni pull request viene controllata da un membro di TypeScript o Definitely Type prima di ottenere un merge, quindi abbi pazienza in quanto fattori umani possono causare ritardi. Leggi la New Pull Request Status Board tper vedere a che punto sono i mantenitori nel controllare le pull request.
I package dovrebbero venire aggiornati us npm in pochi minuti. Se ci mette più di un'ora, riferisciti al numero della pull request sul serve discord di DefinitelyTyped nella community di TypeScript ed il mantenitore assegnerà a chi di dovere per investigare.
Sto scrivendo un file dts che dipende da altri dts. Cosa dovrei usare, <reference types="" />
o import
?
Se il modulo da cui dipendi è un modulo esterno (usa export
), utilizza import
.
Se il modulo da cui dipendi è un modulo d'ambiente (usa declare module
o semplicemente dichiara le cose globali), usa <reference types="" />
.
Alcuni package non hanno un tslint.json
ed ad altri mancano il "noImplicitAny": true
, il "noImplicitThis": true
, o il"strictNullChecks": true
nel tsconfig.json
.
Questo significa che non vanno bene e noi non ce ne siamo ancora accorti. Puoi contribuire inviando una pull request che li sistema.
Se un modulo che utilizzi non ha ancora delle definizioni, puoi chiedere che vengano implementate aprendo un Issue. Qui trovi le richieste di implementazione attuali.
Se i tipi fanno parte di uno standard web, si può contribuire al TSJS-lib-generator, in modo tale che diventino parte del lib.dom.d.ts
ufficiale.
È consigliato aggiunge un namespace vuoto ai package che non esportano un modulo, così da poter usare gli import ES6?
Alcuni package, come chai-http, esportano una funzione.
Importare questo modulo tramite un import ES6 nella forma import * as pippo from "pippo";
dà errore.
error TS2497: Module 'foo' resolves to a non-module entity and cannot be imported using this construct
Questo errore può essere risolto fondendo la dichiarazione della funzione con un namespace vuoto dello stesso nome, ma di norma va evitato. Questa risposta su Stack Overflow è tipicamente citata quando si parla di problemi di questo tipo.
Risulta preferibile importare un modulo usando la sintassi import foo = require("foo");
.
In ogni caso, se vuoi usare un import default come import foo from "foo";
, hai due possibilità:
- puoi usare l'opzione del traspilatore TypeScript
--allowSyntheticDefaultImports
se il tuo modulo supporta a ntime
Importare il modulo usando la sintassi import foo = require("foo");
è più appropriato.
Comunque, se vuoi usare un default import come import foo from "foo";
hai due possibilità:
- puoi usare l'opzione di compilazione
--allowSyntheticDefaultImports
se il tuo ambiente di runtime supporta uno schema interop per moduli non ECMAScript, come nel caso in cui gli import di default sono supportati (Webpack, SystemJS, esm, ...) - puoi usare l'opzione di compilazione
--esModuleInterop
se sia Typescript ad occuparsi dell'interop di moduli non ECMAScript (da TypeScript 2.7 in su).
Un package usa export =
, ma preferirei usare un import di default. Posso cambiare export =
in export default
?
Come già scritto nella risposta precedente, fai riferimento alle opzioni di compilazione --allowSyntheticDefaultImports
e --esModuleInterop
.
Non cambiare la definizione di tipo se è apposita.
Per un package npm, export =
è apposita se node -p 'require("foo")'
funziona per importare un modulo, mentre export default
è apposita se funziona node -p 'require("foo").default'
.
Allora devi aggiungere un commento all'ultima riga del tuo header di definizione (dopo // Definitions: https://github.com/DefinitelyTyped/DefinitelyTyped
): // Minimum TypeScript Version: X.Y
. Ciò specifichera la versione minima supportata.
In ogni caso, se il tuo progetto avesse bisogno di mantenere tipi compatibili con, ad esempio, versioni 3.7 e successive ed allo stesso tempo vesioni 3.6 o precedenti, dovrai usare la funzionalità typesVersions
.
Puoi trovare una spiegazione nel dettaglio di questa funzionalità nella documentazione ufficiale di Typescript.
Ecco qui un breve esempio per iniziare:
-
Aggiungi un file
package.json
alle definizioni dei tuoi package, con il contenuto seguente.{ "private": true, "types": "index", "typesVersions": { "<=3.6": { "*": ["ts3.6/*"] } } }
-
Crea la sottocartella menzionata dal campo
typesVersions
dentro la tua cartella dei tipi (che nel nostro esempio èts3.6/
).ts3.6/
suppporterà le versioni uguali od inferiori alla 3.6, quindi copia i tipi esistenti con i test lì.Dovrai eliminare l'header delle definizioni da
ts3.6/index.d.ts
dal momento che solo la rootindex.d.ts
può averlo. -
Cambia le opzioni
baseUrl
etypeRoots
ints3.6/tsconfig.json
per correggere i path, dovrebbe essere simile a:{ "compilerOptions": { "baseUrl": "../../", "typeRoots": ["../../"] } }
-
Nel root del package, aggiungi le funzionalità di Typescript 3.7 che vuoi usare. Quando il package viene installato, Typescript 3.6 o inferiore partirà da
ts3.6/index.d.ts
, mentre Typescript 3.7 o superiore partirà daindex.d.ts
.Dai un'occhiata a styled-components per averne un esempio.
Questo potrebbe riguardare TSJS-Lib-Generator. Dai un'occhiata a quelle guide.
Se lo standard è ancora una bozza, allora riguarda questa repo.
Usa un nome che comincia per 'dom-' ed includi un collegamento allo standard come il collegamento "Project" nell'header.
Quando smetterà di essere una bozza, potremo rimuoverlo da Definitely Typed e deprecare il package @types
associato.
Come fanno le versioni dei package Definitely Typed a coincidere con le versioni della libreria corrispondente?
NOTA: Questa sezione assume familiarità con Versioni semantiche
Ogni pacchetto Definitly Typed è versionato quando viene pubblicato su npm.
Il DefinitelyTyped-tools, che è lo strumento che pubblica i pacchetti @types
su npm) metterà la versione al pacchetto '@typesusando la versione
major.minorscritta nella prima riga del suo file
index.d.ts. Ad esempio, queste sono le prime righe delle [dichiarazioni di tipi di Node](https://github.com/DefinitelyTyped/DefinitelyTyped/blob/1253faabf5e0d2c5470db6ea87795d7f96fef7e2/types/node/index.d.ts) for version
10.12.x` at the time of writing:
// Type definitions for Node.js 10.12
// Project: https://nodejs.org/
// Definitions by: Microsoft TypeScript <https://github.com/Microsoft>
// Definitely Typed <https://github.com/DefinitelyTyped>
// Alberto Schiabel <https://github.com/jkomyno>
Siccome c'è 10.12
alla fine della prima riga commentata, la versione del pacchetto npm @types/node
sarà 10.12.x
.
Nota bene che la prima riga commentata del index.d.ts
deve contenere solo le prime due parti (major.minor
) della versione (come 10.12
), senza contenere la terza parte "patch" della versione (come 10.12.4
).
Questo è naturale, in quanto solo aggiornamenti delle parti major.minor
della versione possono comportare variazioni delle definizioni e sono quindi allineate con la versione del pacchetto @types
.
La parte patch del pacchetto @types
(come .0
in 10.12.0
) viene inizializzato a zero da Definitely Typed e viene incrementato di uno ogni volta che un nuovo pacchetto @type
, con la stessa versione major.minor
, viene pubblicato su npm per la stessa libreria.
A volte le versioni delle dichiarazioni dei tipi e quelle dei pacchetti corrispondenti può desincronizzarsi. Qui sotto ci sono alcune delle cause più comuni, ordinate in base a quanti inconvenienti causano a chi utilizza la libreria. Solitamente, solo l'ultima causa da problemi.
- Come messo precedentemente in evidenza, la parte patch della versione del pacchetto
@types
non è allineata con quella della libreria analoga. Questo permette a Definitely Typed di aggiornare in modo automatico e sicuro le dichiarazioni dei tipi di una libreria che hanno le stesse major/minor. - Se aggiorni un pacchetto perchè introduci una nuova funzionalità, ricordati di aggiornare la versione anche sul pacchetto
@types
. Se gli autori dei pacchetti fanno corrispondere le versioni di JavaScript con quelle dei pacchetti@types
analoghi,npm update
dovrebbe funzionare alla perfezione. - Molto spesso capita che le versioni delle dichiaraizoni dei tipi rimangano indietro rispetto a quelle del pacchetto JavaScript, questo perchè molte volte non sono gli stessi autori dei pacchetti JavaScript
a farne anche le dichiarazioni dei tipi ma utenti terzi. Per questo motivo potrebbero esserci dei ritardi di giorni, settimane o perfino mesi prima che arrivi una pull request da qualche benefattore che riallinea il pacchetto
@type
con l'ultima versione JavaScript. Se ti ritrovi in questa situazione, puoi essere tu stesso a fare la differenza ed a risolvere il problema diventando un membro a tutti gli effetti della Community.
❗ Se stai aggiornando le dichiarazioni dei tipi per una libreria, ricordati di cambiare sempre la major.minor
nella prima riga del index.d.ts
in modo che con quella della libreria dei cui tipi stai dichiarando! ❗
Se l'aggiornamento di una libreria comprende modifiche sostanziali (aggiornamento major), come faccio ad aggiornare il suo pacchetto '@types`?
Le versioni semantiche richiedono che agli aggiornamenti con modifiche sostanziali venga incrementata la parte major della versione.
Ad esempio, se ad una libreria viene rimossa una funzione che esporta e la sua ultima versione è la 3.5.8
, la sua versione deve essere aggiornata alla 4.0.0
.
Inoltre, quando questa nuova versione 4.0.0
viene pubblicata, la versione del suo pacchetto di dichiarazione dei tipi dev'essere anch'esso aggiornato alla 4.0.0
, modificandone ovviamente le dichiarazioni in modo che riflettano le modifiche sostanziali della nuova versione.
Molte librerie hanno un vasto numero di utenti che le utilizzano (compresi mantenitori di altri pacchetti che usano la libreria come dipendenza) e che probabilimente non passeranno subito alla nuova versione major. Questo perchè potrebbero volerci mesi prima che il mantenitore trovi il tempo per adattare il codice alle nuove modifiche della libreria. Nel frattempo, utenti che usano la vecchia versione della libreria potrebbero richiedere aggiornamenti per la vecchia versione delle dichiarazioni dei tipi, ad esempio nel caso in cui un utente che usa la vecchia versione della libreria trova un erroore nelle definizioni dei tipi.
Se hai intenzione di continuare a mantenere ed aggiornare le dichiarazioni dei tipi per le vecchie versioni di una libreria, puoi creare una sottocartella (es. /v2/
) chiamata come la versione che presto diventerà vecchia e copiarci dentro tutte le dichiarazioni attuali.
Siccome la cartella principale deve sempre contenere le dichiarazioni dei tipi per la versione più nuova, avrai anche bisogno di fare alcune modifiche alle dichiarazioni della vecchia versione che ora sta nella sottocartella, per essere sicuro che gli eventuali indirizzi relativi siano riferiti alla sottocartella e non più alla cartella principale.
- Aggiorna gli indirizzi relativi nel
tsconfig.json
ed in un eventualetslint.json
. - Aggiungi delle regole per mappare gli indirizzi cosicchè i test vengano eseguiti per la versione voluta.
Ad esempio, la libreria history
ha avuto delle modifiche sostanziali passando dalla versione 2.x
alla 3.x
.
Siccome molti utenti sono rimasti alla 2.x
, un mantenitore che voleva fare ulteriori aggiornamenti alle dichiarazioni dei tipi di questa vecchia versione ha aggiunto una sottocartella v2
nella repo delle dichiarazioni dei tipi di questa libreria.
Nel momento in cui questo README è stato scritto, il tsconfig.json
della history v2 è così:
{
"compilerOptions": {
"baseUrl": "../../",
"typeRoots": ["../../"],
"paths": {
"history": [ "history/v2" ]
}
},
"files": [
"index.d.ts",
"history-tests.ts"
]
}
Se ci sono pacchetti su Definitely Typed che sono incompatibili con la vesione più nuova di una libreria, dovrai mappare gli indirizzi alla vecchia versione, continuando ricorsivamente per gli altri pacchetti che dipendono da essa.
Per esempio, react-router
dipende da history@2
, quindi il tsconfig.json
di react-router ha mappato gli indirizzi alla versione vecchia ("history": [ "history/v2" ]
) fino a che non è passato alla nuova versione di react-router
.
A sua volta, la libreria react-router-bootstrap
(che dipende da react-router
) ha dovuto aggiungere lo stesso mapping nel suo tsconfig.json
, fino a che la sua dipendenza react-router
non è stata aggiornata all'ultima versione.
Nota che /// <reference types=".." />
non funziona con il mapping degli indirizzi, quindi le dipendenze devono usare import
.
Come si scrivono le definizioni dei tipi per le librerie che possono essere usata sia globalmente che come modulo?
La guida di Typescript spiega benissimo come scrivere le definizioni dei tipi ed ha anche un file di esempio che è esattamente una libreria che può essere usata sia come modulo su nodejs che come libreria globale in una pagina web.
Per accertarti che le tue definizioni possono essere usate sia globalmente che che modulo importato, crea una cartella test
e creaci dentro due file di test.
Chiamane uno NomeLibreria-global.test.ts
e l'altro YourLibraryName-module.test.ts
.
Il file di test global dovrebbe controllare che le definizioni funzionano bene quando la libreria è usata globalmente in una pagina web (in questo caso non bisogna specificare un import
).
Il file di test module, invece, controlla se le definzioni funzionano quando la libreria viene importata come un modulo.
Se aggiungi la proprietà files
nel tuo tsconfig.json
, assicurati di includere entrambi questi file di test. Un esempio pratico è disponibile nelle dichiarazioni dei tipi di big.js
.
I tipi per un pacchetto @foo/bar
devono andare su types/foo__bar
. Nota il doppio underscore.
Quando dts-gen
viene usato per uno pacchetto con scope, nel tsconfig.json
, la proprietà paths
va cambiata manualmente per referenziarlo correttamente.
{
"paths": {
"@foo/*": ["foo__*"]
}
}
Github non supporta la cronologia per file rinominati. Usa invece git log --follow
instead.
Questo progetto è sotto la licenza MIT.
I copyright nei file di definizione valgono per ogni contributore elencato all'inizio di ogni file di definizione.