Giulio@740: Giulio@740: Giulio@740: L'uso quotidiano di Mercurial Giulio@740: Giulio@740: Giulio@740: Dire a Mercurial di quali file tenere traccia Giulio@740: Giulio@740: Mercurial non lavora con i file nel vostro repository a meno che voi non gli diciate di gestirli. Il comando hg status vi dirà quali file Mercurial non conosce; usa un ? per mostrare tali file. Giulio@740: Giulio@740: Per dire a Mercurial di tenere traccia di un file, usate il comando hg add. Una volta che avete aggiunto un file, la voce per quel file nell'elenco visualizzato da hg status cambia da ? a A. Giulio@740: Giulio@740: &interaction.daily.files.add; Giulio@740: Giulio@740: Dopo aver eseguito hg commit, i file che avete aggiunto prima dell'inserimento non verranno più elencati dal comando hg status. La ragione per questo è che di default, hg status vi segnala solo i file interessanti&emdash;quelli che avete (per esempio) modificato, rimosso, o rinominato. Se avete un repository che contiene miglia di file, vorrete raramente sapere qualcosa dei file che Mercurial ha già registrato ma che non sono cambiati. (Potete comunque ottenere questa informazione, come vedremo più avanti.) Giulio@740: Giulio@740: Una volta che aggiungete un file, Mercurial non fa nulla con quello immediatamente. Invece, scatterà una fotografia dello stato del file la prossima volta che eseguirete un commit. Poi continuerà a tenere traccia dei cambiamenti che apportate al file ogni volta che effettuate un commit, fino a quando non rimuovete il file. Giulio@740: Giulio@740: Giulio@740: Designazione esplicita o implicita dei file Giulio@740: Giulio@740: Un comportamento utile che Mercurial ha è che se passate un nome di una directory a un comando, ogni comando Mercurial tratterà questo come Voglio operare su ogni file in questa directory e nelle sue sottodirectory. Giulio@740: Giulio@740: &interaction.daily.files.add-dir; Giulio@740: Giulio@740: Notate in questo esempio che Mercurial ha stampato i nomi dei file che ha aggiunto, laddove non ha fatto così quando abbiamo aggiunto il file chiamato myfile.txt nell'esempio precedente. Giulio@740: Giulio@740: Quello che sta succedendo è che nel primo caso, abbiamo esplicitamente designato il file da aggiungere sulla linea di comando. L'assunzione fatta da Mercurial in questi casi è che sappiamo quello che stiamo facendo, per cui non stampa alcun ~output~. Giulio@740: Giulio@740: Tuttavia, quando implichiamo i nomi dei file dando il nome di una directory, Mercurial compie il passo aggiuntivo di stampare il nome di ogni file con cui fa qualcosa. Questo rende più chiaro ciò che sta succedendo e riduce la probabilità di una sorpresa sgradita e silenziosa. Questo comportamento è comune alla maggior parte dei comandi Mercurial. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Mercurial registra i file, non le directory Giulio@740: Giulio@740: Mercurial non tiene traccia delle informazioni sulle directory. Invece, tiene traccia del percorso di un file. Prima di creare un file, come prima cosa crea tutte le directory mancanti che ne compongono il percorso. Dopo che ha cancellato un file, cancella ogni directory vuota che faceva parte del percorso del file cancellato. Questa sembra una distinzione insignificante, ma ha una minore conseguenza pratica: non è possibile rappresentare una directory completamente vuota in Mercurial. Giulio@740: Giulio@740: Le directory vuote sono raramente utili, e ci sono soluzioni non invadenti che potete usare per ottenere un effetto appropriato. Quindi, gli sviluppatori di Mercurial hanno sentito che la complessità che sarebbe stata richiesta per gestire le directory vuote non valesse il limitato beneficio che questa funzionalità avrebbe portato. Giulio@740: Giulio@740: Se avete bisogno di una directory vuota nel vostro repository, ci sono alcuni modi per ottenerla. Uno di questi è quello di creare una directory, poi di usare hg add per aggiungere un file nascosto a quella directory. Sui sistemi di tipo Unix, qualsiasi file il cui nome comincia con un punto (.) viene considerato nascosto dalla maggior parte dei comandi e dalle applicazioni con interfaccia grafica. Questo approccio è illustrato qui di seguito. Giulio@740: Giulio@740: &interaction.daily.files.hidden; Giulio@740: Giulio@740: Un altro modo per fronteggiare il bisogno di una directory vuota è semplicemente quello di crearne una nel vostro script automatico di ~build~ nel momento in cui ne avete bisogno. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Come smettere di tenere traccia di un file Giulio@740: Giulio@740: Una volta che avete deciso che un file non appartiene più al vostro repository, usate il comando hg remove. Questo cancella il file e dice a Mercurial di non tenerne più traccia (cosa che avverrà nel prossimo commit). Un file rimosso viene rappresentato nell'elenco prodotto da hg status con una R. Giulio@740: Giulio@740: &interaction.daily.files.remove; Giulio@740: Giulio@740: Dopo che avete rimosso un file tramite hg remove, Mercurial non terrà più traccia di quel file anche se ricreate un file con lo stesso nome nella vostra directory di lavoro. Se ricreate effettivamente un file con lo stesso nome e volete che Mercurial registri il nuovo file, usate semplicemente hg add. Mercurial saprà che il nuovo file non è in alcun modo legato al vecchio file con lo stesso nome. Giulio@740: Giulio@740: Giulio@740: La rimozione di un file non ha effetti sulla sua cronologia. Giulio@740: Giulio@740: È importante capire che la rimozione di un file ha solo due effetti. Giulio@740: Giulio@740: Rimuove la versione corrente del file dalla directory di lavoro. Giulio@740: Giulio@740: Ferma Mercurial dal tenere traccia dei cambiamenti al file, dal momento del commit successivo. Giulio@740: Giulio@740: La rimozione di un file non altera la cronologia del file in alcun modo. Giulio@740: Giulio@740: Se aggiornate la directory di lavoro a un changeset che era stato inserito quando stava ancora tenendo traccia del file che più tardi avete rimosso, il file riapparirà nella directory di lavoro, con i contenuti che aveva quando avete inserito quel changeset. Se poi aggiornate la directory di lavoro a un changeset successivo, in cui il file è stato rimosso, Mercurial rimuoverà ancora una volta il file dalla directory di lavoro. Giulio@740: Giulio@740: Giulio@740: Giulio@740: File mancanti Giulio@740: Giulio@740: Mercurial considera mancante un file che avete cancellato senza usare hg remove. Un file mancante viene rappresentato con ! nell'elenco mostrato da hg status. Di solito, i comandi Mercurial non fanno nulla con i file mancanti. Giulio@740: Giulio@740: &interaction.daily.files.missing; Giulio@740: Giulio@740: Se il vostro repository contiene un file che hg status riporta come mancante, e volete che il file rimanga assente, potete eseguire hg remove in ogni momento più tardi, per dire a Mercurial che volevate effettivamente rimuovere il file. Giulio@740: Giulio@740: &interaction.daily.files.remove-after; Giulio@740: Giulio@740: D'altra parte, se avete cancellato il file mancante per errore, date al comando hg revert il nome del file da recuperare. Il file riapparirà, senza alcun cambiamento. Giulio@740: Giulio@740: &interaction.daily.files.recover-missing; Giulio@740: Giulio@740: Giulio@740: Giulio@740: Digressione: perché dire esplicitamente a Mercurial di rimuovere un file? Giulio@740: Giulio@740: Potreste chiedervi perché Mercurial richiede che gli diciate esplicitamente che state cancellando un file. Nelle prime fasi di sviluppo, Mercurial vi permetteva di cancellare un file nel modo che preferivate; avrebbe notato l'assenza del file automaticamente quando avreste successivamente eseguito hg commit, e avrebbe smesso di tenere traccia del file. In pratica, questo rendeva troppo facile rimuovere un file accidentalmente senza accorgersene. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Utile scorciatoia&emdash;aggiungere e rimuovere file in un unico passo Giulio@740: Giulio@740: Mercurial offre un comando combinato chiamato hg addremove che aggiunge i file non ancora registrati e segnala i file mancanti come rimossi. Giulio@740: Giulio@740: &interaction.daily.files.addremove; Giulio@740: Giulio@740: Il comando hg commit fornisce anche un'opzione che effettua questo stesso aggiungi-e-rimuovi, immediatamente seguito da un commit. Giulio@740: Giulio@740: &interaction.daily.files.commit-addremove; Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Copiare i file Giulio@740: Giulio@740: Mercurial fornisce un comando hg copy che vi permette di creare una nuova copia di un file. Quando copiate un file usando questo comando, Mercurial registra il fatto che il nuovo file è una copia del file originale e tratta queste copie in maniera speciale quando unite il vostro lavoro con quello di qualcun altro. Giulio@740: Giulio@740: Giulio@740: I risultati di una copia durante un'unione Giulio@740: Giulio@740: Quello che succede durante un'unione è che i cambiamenti seguono la copia. Per illustrare al meglio cosa questo significa, creiamo un esempio. Cominceremo con il solito piccolo repository che contiene un singolo file. Giulio@740: Giulio@740: &interaction.daily.copy.init; Giulio@740: Giulio@740: Abbiamo bisogno di fare del lavoro in parallelo, in modo da avere qualcosa da incorporare. Quindi cloniamo il nostro repository. Giulio@740: Giulio@740: &interaction.daily.copy.clone; Giulio@740: Giulio@740: Tornando al nostro repository iniziale, usiamo il comando hg copy per fare una copia del primo file che abbiamo creato. Giulio@740: Giulio@740: &interaction.daily.copy.copy; Giulio@740: Giulio@740: Se successivamente osserviamo il risultato del comando hg status, il file copiato appare come un normale file aggiunto. Giulio@740: Giulio@740: &interaction.daily.copy.status; Giulio@740: Giulio@740: Ma se passiamo l'opzione al comando hg status, otterremo un'altra riga nell'elenco stampato: questo è il file da cui il nostro file appena aggiunto è stato copiato. Giulio@740: Giulio@740: &interaction.daily.copy.status-copy; Giulio@740: Giulio@740: Ora, tornando al repository che abbiamo clonato, apportiamo un cambiamento in parallelo. Aggiungeremo una riga di contenuto al file originale che abbiamo creato. Giulio@740: Giulio@740: &interaction.daily.copy.other; Giulio@740: Giulio@740: Ora abbiamo un file modificato in questo repository. Quando estraiamo i cambiamenti dal primo repository e uniamo le due teste, Mercurial propagherà i cambiamenti che abbiamo apportato localmente a file nella sua copia, new-file. Giulio@740: Giulio@740: &interaction.daily.copy.merge; Giulio@740: Giulio@740: Giulio@740: Giulio@740: Perché i cambiamenti dovrebbeo seguire le copie? Giulio@740: Giulio@740: Questo comportamento&emdash;dei cambiamenti a un file che si propagano alle copie del file&emdash;potrebbe sembrare esoterico, ma nella maggior parte dei casi è altamente desiderabile. Giulio@740: Giulio@740: Prima di tutto, ricordatevi che questa propagazione avviene solamente durante un'unione. Quindi se usate hg copy su un file e in seguito modificate il file originale nel normale corso del vostro lavoro, non accadrà nulla. Giulio@740: Giulio@740: La seconda cosa da sapere è che le modifiche si propagheranno alla copia solo se il changeset da cui state incorporando le modifiche non ha ancora visto la copia. Giulio@740: Giulio@740: Il motivo per cui Mercurial si comporta in questo modo è il seguente. Diciamo che correggo un bug importante in un file sorgente e inserisco i miei cambiamenti nel repository. Nel frattempo, voi avete deciso di eseguire hg copy per fare una copia del file nel vostro repository, senza sapere del bug o aver visto la correzione, e avete cominciato a lavorare sulla vostra copia del file. Giulio@740: Giulio@740: Se aveste estratto e incorporato i miei cambiamenti, e Mercurial non avesse propagato i cambiamenti attraverso le copie, il vostro nuovo file sorgente ora conterrebbe il bug, e a meno che voi sapeste come propagare la correzione a mano, il bug rimarrebbe nella vostra copia del file. Giulio@740: Giulio@740: Propagando automaticamente le modifiche che hanno corretto il bug dal file originale alla copia, Mercurial previene questo tipo di problemi. A quanto ne so, Mercurial è l'unico sistema di controllo di revisione che propaga i cambiamenti verso le copie in questo modo. Giulio@740: Giulio@740: Una volta che la vostra cronologia dei cambiamenti contiene la registrazione che la copia e la successiva unione sono avvenute, di solito non c'è ulteriore bisogno di propagare cambiamenti dal file originale al file copiato, e questo è il motivo per cui Mercurial propaga i cambiamenti verso le copie durante la prima unione, e non successivamente. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Come <emphasis>evitare</emphasis> che i cambiamenti seguano una copia Giulio@740: Giulio@740: Se, per qualche ragione, decidete che questo affare di propagare automaticamente i cambiamenti verso le copie non fa per voi, utilizzate il normale comando per la copia dei file fornito dal vostro sistema (cp per i sistemi di tipo Unix) per fare la copia di un file, poi aggiungete a mano la nuova copia invocando hg add. Prima di farlo, però, rileggete la e prendete una decisione informata sulla appropriatezza di questo comportamento nel vostro caso specifico. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Il comportamento del comando <command role="hg-cmd">hg copy</command> Giulio@740: Giulio@740: Quando usate il comando hg copy, Mercurial esegue la copia di ogni file sorgente nello stato in cui si trova nella directory di lavoro in quel momento. Questo significa che se fate alcune modifiche a un file, poi lo copiate tramite hg copy senza prima aver inserito quelle modifiche nel repository, anche la nuova copia conterrà le modifiche che avete apportato fino a quel momento. (Trovo che questo comportamento sia leggermente controintuitivo ed è per questo che lo menziono qui.) Giulio@740: Giulio@740: Il comando hg copy agisce in maniera simile al comando Unix cp (potete usare l'alias hg cp se preferite). Dobbiamo fornirgli due o più argomenti, di cui l'ultimo viene trattato come destinazione e tutti gli altri vengono trattati come sorgenti. Giulio@740: Giulio@740: Se invocate hg copy con un singolo file come sorgente, e la destinazione non esiste, il comando crea un nuovo file con quel nome. Giulio@740: Giulio@740: &interaction.daily.copy.simple; Giulio@740: Giulio@740: Se la destinazione è una directory, Mercurial copia le sorgenti in quella directory. Giulio@740: Giulio@740: &interaction.daily.copy.dir-dest; Giulio@740: Giulio@740: Copiare una directory è un'operazione ricorsiva e preserva la struttura delle directory della sorgente. Giulio@740: Giulio@740: &interaction.daily.copy.dir-src; Giulio@740: Giulio@740: Se la sorgente e la destinazione sono entrambe directory, l'albero della sorgente viene ricreato nella directory di destinazione. Giulio@740: Giulio@740: &interaction.daily.copy.dir-src-dest; Giulio@740: Giulio@740: Come con il comando hg remove, se copiate un file manualmente e poi volete far sapere a Mercurial che avete copiato il file, usate semplicemente l'opzione per hg copy. Giulio@740: Giulio@740: &interaction.daily.copy.after; Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Rinominare i file Giulio@740: Giulio@740: È molto più comune avere bisogno di rinominare un file piuttosto che copiarlo. La ragione per cui ho discusso il comando hg copy prima di parlare di come rinominare i file è che Mercurial tratta un cambiamento di nome essenzialmente nello stesso modo di una copia. Perciò, sapere cosa fa Mercurial qunado copiate un file vi dice che cosa aspettarvi quando rinominate un file. Giulio@740: Giulio@740: Quando usate il comando hg rename, Mercurial crea una copia di ogni file sorgente, poi li cancella e segnala i file come rimossi. Giulio@740: Giulio@740: &interaction.daily.rename.rename; Giulio@740: Giulio@740: Il comando hg status mostra la nuova copia del file come aggiunta e il file da cui è stata effettuata la copia come rimosso. Giulio@740: Giulio@740: &interaction.daily.rename.status; Giulio@740: Giulio@740: Come con i risultati del comando hg copy, dobbiamo dare l'opzione al comando hg status per vedere che il file aggiunto viene considerato da Mercurial come una copia del file originale ora rimosso. Giulio@740: Giulio@740: &interaction.daily.rename.status-copy; Giulio@740: Giulio@740: Come con hg remove e hg copy, potete informare Mercurial del cambiamento di nome dopo che il fatto è avvenuto usando l'opzione option. Nella maggior parte degli altri aspetti, il comportamento del comando hg rename e le opzioni che accetta sono simili a quelli del comando hg copy. Giulio@740: Giulio@740: Se avete familiarità con la riga di comando Unix, sarete contenti di sapere che il comando hg rename può essere invocato come hg mv. Giulio@740: Giulio@740: Giulio@740: Rinominare i file e unire i cambiamenti Giulio@740: Giulio@740: Dato che i cambiamenti di nome in Mercurial sono implementati come copia-e-rimuovi, la stessa propagazione dei cambiamenti avviene quando unite dopo un cambiamento di nome che dopo una copia. Giulio@740: Giulio@740: Se io modifico un file e voi lo rinominate e poi uniamo i nostri rispettivi cambiamenti, le mie modifiche al file sotto il suo nome originale verranno propagate al file sotto il suo nuovo nome. (Questo è qualcosa che potreste aspettarvi che funzioni e basta ma in realtà non tutti i sistemi di controllo di revisione lo fanno.) Giulio@740: Giulio@740: Sebbene avere i cambiamenti che seguono una copia è una funzionalità dove potreste forse annuire e dire sì, questo potrebbe essere utile, deve essere chiaro che averli che seguono un cambiamento di nome è assolutamente importante. Senza questa agevolazione, sarebbe semplicemente troppo facile per i cambiamenti venire abbandonati quando i file vengono rinominati. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Cambiamenti di nome divergenti e unioni Giulio@740: Giulio@740: Il caso dei nomi divergenti accade quando due sviluppatori cominciano con un file&emdash;chiamiamolo foo&emdash;nei loro rispettivi repository. Giulio@740: Giulio@740: &interaction.rename.divergent.clone; Giulio@740: Giulio@740: Anna cambia il nome del file a bar. Giulio@740: Giulio@740: &interaction.rename.divergent.rename.anne; Giulio@740: Giulio@740: Nel frattempo, Bruno lo rinomina quux. (Ricordatevi che hg mv è un alias di hg rename.) Giulio@740: Giulio@740: &interaction.rename.divergent.rename.bob; Giulio@740: Giulio@740: Mi piace pensare a questo come a un conflitto perché entrambi gli sviluppatori hanno espresso intenzioni differenti a proposito di come il file dovrebbe essere chiamato. Giulio@740: Giulio@740: Cosa pensate che dovrebbe accadere quando uniscono il loro lavoro? Il reale comportamento di Mercurial è quello di preservare sempre entrambi i nomi quando unisce changeset che contengono cambiamenti di nome divergenti. Giulio@740: Giulio@740: &interaction.rename.divergent.merge; Giulio@740: Giulio@740: Notate che, se anche Mercurial vi avverte del cambiamento di nome divergente, lascia che siate voi a risolvere la divergenza dopo l'unione. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Cambiamenti di nome convergenti e unioni Giulio@740: Giulio@740: Un altro tipo di conflitto tra i cambiamenti di nome avviene quando due persone rinominano differenti file sorgenti alla stessa destinazione. In questo caso, Mercurial esegue l'unione normalmente e lascia che siate voi a guidarlo verso una risoluzione ragionevole. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Altri casi particolari legati ai nomi Giulio@740: Giulio@740: Mercurial ha un bug permanente che gli impedisce di portare a termine un'unione dove una parte contiene un file con un certo nome mentre l'altra contiene una directory con lo stesso nome. Questo è documentato come problema 29. Giulio@740: Giulio@740: &interaction.issue29.go; Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Rimediare agli errori Giulio@740: Giulio@740: Mercurial possiede alcuni utili comandi che vi aiuteranno a rimediare ad alcuni errori comuni. Giulio@740: Giulio@740: Il comando hg revert vi permette di annullare i cambiamenti che avete apportato alla vostra directory di lavoro. Per esempio, se avete aggiunto un file invocando hg add per errore, vi basta eseguire hg revert con il nome del file che avete aggiunto, e il file non verrà toccato in alcun modo e non sarà più registrato per essere aggiunto da Mercurial. Potete anche usare hg revert per disfarvi di cambiamenti sbagliati apportati a un file. Giulio@740: Giulio@740: È vantaggioso ricordare che il comando hg revert è utile per i cambiamenti che non avete ancora inserito. Una volta che avete inserito un cambiamento, se decidete che è stato un errore potete ancora fare qualcosa, sebbene le vostre opzioni siano molto più limitate. Giulio@740: Giulio@740: Per maggiori informazioni sul comando hg revert e dettagli su come trattare i cambiamenti che avete gia inserito, leggete la FIXME. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Affrontare unioni complesse Giulio@740: Giulio@740: In un progetto grande o complicato, non è inusuale che l'unione tra due changeset risulti in qualche mal di testa. Supponete che ci sia un file sorgente di grandi dimensioni che è stato ampiamente modificato da entrambe le parti di un'unione: quasi inevitabilmente, questo risulterà in conflitti, alcuni dei quali potrebbero volerci alcuni tentativi per risolverli. Giulio@740: Giulio@740: Costruiamoci un semplice esempio di questo e vediamo come affrontarlo. Cominceremo con un repository contenente un file e lo cloneremo due volte. Giulio@740: Giulio@740: &interaction.ch04-resolve.init; Giulio@740: Giulio@740: In uno dei cloni, modificheremo il file in un modo. Giulio@740: Giulio@740: &interaction.ch04-resolve.left; Giulio@740: Giulio@740: Nell'altro, modificheremo il file in modo differente. Giulio@740: Giulio@740: &interaction.ch04-resolve.right; Giulio@740: Giulio@740: Poi, propagheremo entrambi i cambiamenti nel nostro repository originale. Giulio@740: Giulio@740: &interaction.ch04-resolve.pull; Giulio@740: Giulio@740: Ora ci aspettiamo che il nostro repository contenga due teste. Giulio@740: Giulio@740: &interaction.ch04-resolve.heads; Giulio@740: Giulio@740: Normalmente, se eseguissimo il comando hg merge a questo punto, ci presenterebbe un'applicazione grafica che ci permetterebbe di riconciliare manualmente le modifiche in conflitto su myfile.txt. Tuttavia, per semplificare le cose ai fini della presentazione, vorremmo invece che l'unione fallisse immediatamente. Ecco un modo in cui possiamo farlo. Giulio@740: Giulio@740: &interaction.ch04-resolve.export; Giulio@740: Giulio@740: Abbiamo chiesto al meccanismo di unione di Mercurial di eseguire il comando false (che, come desideriamo, fallisce immediatamente) se si accorge di un'unione che non è in grado di risolvere automaticamente. Giulio@740: Giulio@740: Se ora lanciamo hg merge, dovrebbe fermarsi e riportare un fallimento. Giulio@740: Giulio@740: &interaction.ch04-resolve.merge; Giulio@740: Giulio@740: Se anche non abbiamo notato che l'unione è fallita, Mercurial eviterà di farci accidentalmente inserire i risultati di un'unione fallita. Giulio@740: Giulio@740: &interaction.ch04-resolve.cifail; Giulio@740: Giulio@740: Quando hg commit fallisce in questo caso, suggerisce di usare il comando hg resolve a noi sconosciuto. Come al solito, hg help resolve stamperà un'utile sinossi. Giulio@740: Giulio@740: Giulio@740: Gli stati di risoluzione di un file Giulio@740: Giulio@740: Quando avviene un'unione, di solito la maggior parte dei file rimarrà tale e quale. Per ogni file per cui deve fare qualcosa, Mercurial terrà traccia del suo stato. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Un file risolto è stato unito con successo, automaticamente da Mercurial oppure con un intervento umano. Giulio@740: Giulio@740: Giulio@740: Un file irrisolto non è stato unito con successo e necessità di ulteriori attenzioni. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Se Mercurial vede un qualsiasi file nello stato irrisolto dopo un'unione, considera l'unione come fallita. Fortunatamente, non abbiamo bisogno di ricominciare l'intera unione da zero. Giulio@740: Giulio@740: L'opzione o per il comando hg resolve mostra lo stato di ogni file coinvolto in un'unione. Giulio@740: Giulio@740: &interaction.ch04-resolve.list; Giulio@740: Giulio@740: Nell'elenco stampato da hg resolve, un file risolto è contrassegnato con una R mentre un file irrisolto è contrassegnato con una U. Se un file qualsiasi viene elencato con una U, sappiamo che un tentativo di inserire i risultati dell'unione nel repository andrebbe incontro al fallimento. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Risolvere un'unione di file Giulio@740: Giulio@740: Abbiamo diverse opzioni per muovere un file dallo stato irrisolto a quello risolto. Quella di gran lunga più comune consiste nell'eseguire nuovamente hg resolve. Se passiamo i nomi di singoli file o directory, il comando riproverà a unire i file irrisolti presenti in quelle ubicazioni. Possiamo anche passare l'opzione o , che riproverà a unire tutti i file irrisolti. Giulio@740: Giulio@740: Mercurial ci permette anche di modificare direttamente lo stato di risoluzione di un file. Possiamo manualmente contrassegnare un file come risolto usando l'opzione , o come irrisolto usando l'opzione . Questo ci consente di ripulire a mano un'unione particolarmente confusa e di tenere traccia dei nostri progressi con ogni file man mano che procediamo. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Formati di diff più utili Giulio@740: Giulio@740: L'uscita di default del comando hg diff è compatibile all'indietro con il normale comando diff, ma questo presenta alcuni svantaggi. Giulio@740: Giulio@740: Considerate il caso in cui usiamo hg rename per rinominare un file. Giulio@740: Giulio@740: &interaction.ch04-diff.rename.basic; Giulio@740: Giulio@740: Il risultato di hg diff mostrato qui sopra oscura il fatto che abbiamo semplicemente rinominato un file. Il comando hg diff accetta l'opzione o per usare un formato di diff più nuovo che mostra queste informazioni in una forma più leggibile. Giulio@740: Giulio@740: &interaction.ch04-diff.rename.git; Giulio@740: Giulio@740: Questa opzione ci viene in aiuto anche in un caso che altrimenti risulterebbe confuso: un file che sembra essere stato modificato secondo hg status, ma per il quale hg diff non stampa nulla. Questa situazione può presentarsi se cambiamo i permessi di esecuzione di un file. Giulio@740: Giulio@740: &interaction.ch04-diff.chmod; Giulio@740: Giulio@740: Il normale comando diff non fa attenzione di permessi dei file, e questo è il motivo per cui hg diff non stampa nulla di default. Se forniamo al comando l'opzione , ci dice che cos'è realmente accaduto. Giulio@740: Giulio@740: &interaction.ch04-diff.chmod.git; Giulio@740: Giulio@740: Giulio@740: Giulio@740: Quali file gestire e quali file evitare Giulio@740: Giulio@740: I sistemi di controllo di revisione sono generalmente migliori a gestire file di testo che sono scritti da esseri umani, come il codice sorgente, dove i file non cambiano molto da una revisione all'altra. Alcuni sistemi centralizzati di controllo di revisione possono anche affrontare abbastanza bene i file binari, come immagini bitmap. Giulio@740: Giulio@740: Per esempio, il gruppo di sviluppo di un gioco dovrà tipicamente gestire sia il proprio codice sorgente sia i propri asset binari (e.g. dati geometrici, texture, schemi di mappe) in un sistema di controllo di revisione. Giulio@740: Giulio@740: Dato che di solito è impossibile unire due modifiche in conflitto a un file binario, i sistemi centralizzati spesso forniscono un meccanismo di bloccaggio dei file che permette a un utente di dire sono la sola persona che può modificare questo file. Giulio@740: Giulio@740: Confrontato con un sistema centralizzato, un sistema distribuito di controllo di revisione modifica alcuni dei fattori che guidano le decisioni su quali file gestire e come. Giulio@740: Giulio@740: Per esempio, un sistema distribuito di controllo di revisione non può, per la sua natura, offrire un meccanismo di bloccaggio dei file. Qunidi non c'è alcun meccanismo predefinito per evitare che due persone apportino cambiamenti in conflitto a un file binario. Se avete un gruppo dove diverse persone potrebbero modificare frequentemente i file binari, potrebbe non essere una buona idea impiegare Mercurial&emdash;o un qualsiasi altro sistema distribuito di controllo di revisione&emdash;per gestire quei file. Giulio@740: Giulio@740: Quando memorizza le modifiche a un file, di solito Mercurial salva solo le differenze tra la versione corrente del file e quella precedente. Per la maggior parte dei file di testo, questo si rivela estremamente efficiente. Tuttavia, alcuni file (in particolare i file binari) sono costruiti in modo tale che persino un piccolo cambiamento al contenuto logico del file risulta nel cambiamento di molti o della maggior parte dei byte contenuti nel file. Per esempio, i file compressi sono particolarmente suscettibili a questo. Se le differenze tra ogni versione successiva di un file sono sempre grandi, Mercurial non riuscirà a memorizzare la cronologia del file in maniera molto efficiente. Questo potrebbe avere effetti sia sul bisogno di spazio di memorizzazione locale sia sulla quantità di tempo che viene impiegata per clonare un repository. Giulio@740: Giulio@740: Per avere un'idea di come questo potrebbe riguardarvi nella pratica, supponete di voler usare Mercurial per gestire un documento OpenOffice. OpenOffice memorizza i documenti su disco sotto forma di file zip compressi. Modificate anche solo una lettera nel vostro documento in OpenOffice e quasi ogni byte nell'intero file cambieranno quando lo salverete. Ora supponete che quel file sia di 2MB. Dato che la maggior parte del file cambia ogni volta che lo salvate, Mercurial dovrà memorizzare tutti i 2MB del file ogni volta che eseguite un commit, anche se dal vostro punto di vista forse solo poche parole vengono cambiate ogni volta. Un singolo file modificato frequentemente che non rispetti amichevolmente le assunzioni dei meccanismi di memorizzazione di Mercurial può facilmente avere un effetto fuori misura sulle dimensioni del repository. Giulio@740: Giulio@740: Anche peggio, se voi e qualcun altro modificate il documento OpenOffice su cui state lavorando, non c'è alcun modo utile di effettuare un'unione tra le diverse versioni. In effetti, non c'è nemmeno un buon modo di capire quali sono le differenze tra i vostri rispettivi cambiamenti. Giulio@740: Giulio@740: Quindi, ci sono alcune chiare raccomandazioni sui tipi di file con i quali dovete fare molta attenzione. Giulio@740: Giulio@740: Giulio@740: Giulio@740: I file che sono molto grandi e incomprimibili, e.g. le immagini ISO dei CD-ROM, renderanno la clonazione attraverso la rete molto lenta semplicemente a causa delle loro dimensioni. Giulio@740: Giulio@740: Giulio@740: I file che cambiano parecchio da una revisoine alla successiva potrebbero essere costosi da memorizzare se li modificate frequentemente, e i conflitti causati da modifiche in parallelo potrebbero essere difficili da risolvere. Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Giulio@740: Backup e ~mirroring~ Giulio@740: Giulio@740: Dato che Mercurial mantiene una copia completa della cronologia in ogni clone, chiunque usi Mercurial per collaborare su un progetto può potenzialmente agire come una sorgente di backup nell'eventualità di una catastrofe. Se un repository centrale diventa inaccessibile, potete costruire un rimpiazzo semplicemente clonando la copia del repository da un collaboratore ed estraendo qualsiasi cambiamento che potrebbe non aver visto da altre persone. Giulio@740: Giulio@740: È semplice usare Mercurial per effettuare backup ~off-site~ e mirror remoti. Impostate un ~job~ periodico (e.g. tramite il comando cron) su un server remoto per estrarre i cambiamenti dai vostri repository principali ogni ora. Questo sarà complicato solo nell'improbabile caso in cui il numero di repository principali che mantenete cambia frequentemente, caso in cui avrete bisogno di usare uno script per programmare l'aggiornamento della lista dei repository di cui fare il backup. Giulio@740: Giulio@740: Se effettuate un backup tradizionale dei vostri repository principali su nastro o disco e volete fare il backup di un repository chiamato myrepo, usate il comando hg clone -U myrepo myrepo.bak per creare un clone di myrepo prima di cominciare i vostri backup. L'opzione non estrae una directory di lavoro dopo che la clonazione si è conclusa, dato che sarebbe superfluo e renderebbe più lungo il backup. Giulio@740: Giulio@740: Se poi effettuate il backup di myrepo.bak invece di myrepo, avrete la garanzia di possedere una fotografia consistente del vostro repository a cui nessuno sviluppatore insonne trasmetterà i propri cambiamenti nel bel mezzo di un'operazione di backup. Giulio@740: Giulio@740: