A place to cache linked articles (think custom and personal wayback machine)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

index.html 57KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935
  1. <!doctype html><!-- This is a valid HTML5 document. -->
  2. <!-- Screen readers, SEO, extensions and so on. -->
  3. <html lang=fr>
  4. <!-- Has to be within the first 1024 bytes, hence before the <title>
  5. See: https://www.w3.org/TR/2012/CR-html5-20121217/document-metadata.html#charset -->
  6. <meta charset=utf-8>
  7. <!-- Why no `X-UA-Compatible` meta: https://stackoverflow.com/a/6771584 -->
  8. <!-- The viewport meta is quite crowded and we are responsible for that.
  9. See: https://codepen.io/tigt/post/meta-viewport-for-2015 -->
  10. <meta name=viewport content="width=device-width,minimum-scale=1,initial-scale=1,shrink-to-fit=no">
  11. <!-- Required to make a valid HTML5 document. -->
  12. <title>30 options de commande Git qui gagnent à être connues (archive) — David Larlet</title>
  13. <!-- Generated from https://realfavicongenerator.net/ such a mess. -->
  14. <link rel="apple-touch-icon" sizes="180x180" href="/static/david/icons/apple-touch-icon.png">
  15. <link rel="icon" type="image/png" sizes="32x32" href="/static/david/icons/favicon-32x32.png">
  16. <link rel="icon" type="image/png" sizes="16x16" href="/static/david/icons/favicon-16x16.png">
  17. <link rel="manifest" href="/manifest.json">
  18. <link rel="mask-icon" href="/static/david/icons/safari-pinned-tab.svg" color="#5bbad5">
  19. <link rel="shortcut icon" href="/static/david/icons/favicon.ico">
  20. <meta name="apple-mobile-web-app-title" content="David Larlet">
  21. <meta name="application-name" content="David Larlet">
  22. <meta name="msapplication-TileColor" content="#da532c">
  23. <meta name="msapplication-config" content="/static/david/icons/browserconfig.xml">
  24. <meta name="theme-color" content="#f0f0ea">
  25. <!-- That good ol' feed, subscribe :p. -->
  26. <link rel=alternate type="application/atom+xml" title=Feed href="/david/log/">
  27. <meta name="robots" content="noindex, nofollow">
  28. <meta content="origin-when-cross-origin" name="referrer">
  29. <!-- Canonical URL for SEO purposes -->
  30. <link rel="canonical" href="http://www.git-attitude.fr/2014/09/15/30-options-git-qui-gagnent-a-etre-connues/">
  31. <style>
  32. /* http://meyerweb.com/eric/tools/css/reset/ */
  33. html, body, div, span,
  34. h1, h2, h3, h4, h5, h6, p, blockquote, pre,
  35. a, abbr, address, big, cite, code,
  36. del, dfn, em, img, ins,
  37. small, strike, strong, tt, var,
  38. dl, dt, dd, ol, ul, li,
  39. fieldset, form, label, legend,
  40. table, caption, tbody, tfoot, thead, tr, th, td,
  41. article, aside, canvas, details, embed,
  42. figure, figcaption, footer, header, hgroup,
  43. menu, nav, output, ruby, section, summary,
  44. time, mark, audio, video {
  45. margin: 0;
  46. padding: 0;
  47. border: 0;
  48. font-size: 100%;
  49. font: inherit;
  50. vertical-align: baseline;
  51. }
  52. /* HTML5 display-role reset for older browsers */
  53. article, aside, details, figcaption, figure,
  54. footer, header, hgroup, menu, nav, section { display: block; }
  55. body { line-height: 1; }
  56. blockquote, q { quotes: none; }
  57. blockquote:before, blockquote:after,
  58. q:before, q:after {
  59. content: '';
  60. content: none;
  61. }
  62. table {
  63. border-collapse: collapse;
  64. border-spacing: 0;
  65. }
  66. /* http://practicaltypography.com/equity.html */
  67. /* https://calendar.perfplanet.com/2016/no-font-face-bulletproof-syntax/ */
  68. /* https://www.filamentgroup.com/lab/js-web-fonts.html */
  69. @font-face {
  70. font-family: 'EquityTextB';
  71. src: url('/static/david/css/fonts/Equity-Text-B-Regular-webfont.woff2') format('woff2'),
  72. url('/static/david/css/fonts/Equity-Text-B-Regular-webfont.woff') format('woff');
  73. font-weight: 300;
  74. font-style: normal;
  75. font-display: swap;
  76. }
  77. @font-face {
  78. font-family: 'EquityTextB';
  79. src: url('/static/david/css/fonts/Equity-Text-B-Italic-webfont.woff2') format('woff2'),
  80. url('/static/david/css/fonts/Equity-Text-B-Italic-webfont.woff') format('woff');
  81. font-weight: 300;
  82. font-style: italic;
  83. font-display: swap;
  84. }
  85. @font-face {
  86. font-family: 'EquityTextB';
  87. src: url('/static/david/css/fonts/Equity-Text-B-Bold-webfont.woff2') format('woff2'),
  88. url('/static/david/css/fonts/Equity-Text-B-Bold-webfont.woff') format('woff');
  89. font-weight: 700;
  90. font-style: normal;
  91. font-display: swap;
  92. }
  93. @font-face {
  94. font-family: 'ConcourseT3';
  95. src: url('/static/david/css/fonts/concourse_t3_regular-webfont-20190806.woff2') format('woff2'),
  96. url('/static/david/css/fonts/concourse_t3_regular-webfont-20190806.woff') format('woff');
  97. font-weight: 300;
  98. font-style: normal;
  99. font-display: swap;
  100. }
  101. /* http://practice.typekit.com/lesson/caring-about-opentype-features/ */
  102. body {
  103. /* http://www.cssfontstack.com/ Palatino 99% Win 86% Mac */
  104. font-family: "EquityTextB", Palatino, serif;
  105. background-color: #f0f0ea;
  106. color: #07486c;
  107. font-kerning: normal;
  108. -moz-osx-font-smoothing: grayscale;
  109. -webkit-font-smoothing: subpixel-antialiased;
  110. text-rendering: optimizeLegibility;
  111. font-variant-ligatures: common-ligatures contextual;
  112. font-feature-settings: "kern", "liga", "clig", "calt";
  113. }
  114. pre, code, kbd, samp, var, tt {
  115. font-family: 'TriplicateT4c', monospace;
  116. }
  117. em {
  118. font-style: italic;
  119. color: #323a45;
  120. }
  121. strong {
  122. font-weight: bold;
  123. color: black;
  124. }
  125. nav {
  126. background-color: #323a45;
  127. color: #f0f0ea;
  128. display: flex;
  129. justify-content: space-around;
  130. padding: 1rem .5rem;
  131. }
  132. nav:last-child {
  133. border-bottom: 1vh solid #2d7474;
  134. }
  135. nav a {
  136. color: #f0f0ea;
  137. }
  138. nav abbr {
  139. border-bottom: 1px dotted white;
  140. }
  141. h1 {
  142. border-top: 1vh solid #2d7474;
  143. border-bottom: .2vh dotted #2d7474;
  144. background-color: #e3e1e1;
  145. color: #323a45;
  146. text-align: center;
  147. padding: 5rem 0 4rem 0;
  148. width: 100%;
  149. font-family: 'ConcourseT3';
  150. display: flex;
  151. flex-direction: column;
  152. }
  153. h1.single {
  154. padding-bottom: 10rem;
  155. }
  156. h1 span {
  157. position: absolute;
  158. top: 1vh;
  159. left: 20%;
  160. line-height: 0;
  161. }
  162. h1 span a {
  163. line-height: 1.7;
  164. padding: 1rem 1.2rem .6rem 1.2rem;
  165. border-radius: 0 0 6% 6%;
  166. background: #2d7474;
  167. font-size: 1.3rem;
  168. color: white;
  169. text-decoration: none;
  170. }
  171. h2 {
  172. margin: 4rem 0 1rem;
  173. border-top: .2vh solid #2d7474;
  174. padding-top: 1vh;
  175. }
  176. h3 {
  177. text-align: center;
  178. margin: 3rem 0 .75em;
  179. }
  180. hr {
  181. height: .4rem;
  182. width: .4rem;
  183. border-radius: .4rem;
  184. background: #07486c;
  185. margin: 2.5rem auto;
  186. }
  187. time {
  188. display: bloc;
  189. margin-left: 0 !important;
  190. }
  191. ul, ol {
  192. margin: 2rem;
  193. }
  194. ul {
  195. list-style-type: square;
  196. }
  197. a {
  198. text-decoration-skip-ink: auto;
  199. text-decoration-thickness: 0.05em;
  200. text-underline-offset: 0.09em;
  201. }
  202. article {
  203. max-width: 50rem;
  204. display: flex;
  205. flex-direction: column;
  206. margin: 2rem auto;
  207. }
  208. article.single {
  209. border-top: .2vh dotted #2d7474;
  210. margin: -6rem auto 1rem auto;
  211. background: #f0f0ea;
  212. padding: 2rem;
  213. }
  214. article p:last-child {
  215. margin-bottom: 1rem;
  216. }
  217. p {
  218. padding: 0 .5rem;
  219. margin-left: 3rem;
  220. }
  221. p + p,
  222. figure + p {
  223. margin-top: 2rem;
  224. }
  225. blockquote {
  226. background-color: #e3e1e1;
  227. border-left: .5vw solid #2d7474;
  228. display: flex;
  229. flex-direction: column;
  230. align-items: center;
  231. padding: 1rem;
  232. margin: 1.5rem;
  233. }
  234. blockquote cite {
  235. font-style: italic;
  236. }
  237. blockquote p {
  238. margin-left: 0;
  239. }
  240. figure {
  241. border-top: .2vh solid #2d7474;
  242. background-color: #e3e1e1;
  243. text-align: center;
  244. padding: 1.5rem 0;
  245. margin: 1rem 0 0;
  246. font-size: 1.5rem;
  247. width: 100%;
  248. }
  249. figure img {
  250. max-width: 250px;
  251. max-height: 250px;
  252. border: .5vw solid #323a45;
  253. padding: 1px;
  254. }
  255. figcaption {
  256. padding: 1rem;
  257. line-height: 1.4;
  258. }
  259. aside {
  260. display: flex;
  261. flex-direction: column;
  262. background-color: #e3e1e1;
  263. padding: 1rem 0;
  264. border-bottom: .2vh solid #07486c;
  265. }
  266. aside p {
  267. max-width: 50rem;
  268. margin: 0 auto;
  269. }
  270. /* https://fvsch.com/code/css-locks/ */
  271. p, li, pre, code, kbd, samp, var, tt, time, details, figcaption {
  272. font-size: 1rem;
  273. line-height: calc( 1.5em + 0.2 * 1rem );
  274. }
  275. h1 {
  276. font-size: 1.9rem;
  277. line-height: calc( 1.2em + 0.2 * 1rem );
  278. }
  279. h2 {
  280. font-size: 1.6rem;
  281. line-height: calc( 1.3em + 0.2 * 1rem );
  282. }
  283. h3 {
  284. font-size: 1.35rem;
  285. line-height: calc( 1.4em + 0.2 * 1rem );
  286. }
  287. @media (min-width: 20em) {
  288. /* The (100vw - 20rem) / (50 - 20) part
  289. resolves to 0-1rem, depending on the
  290. viewport width (between 20em and 50em). */
  291. p, li, pre, code, kbd, samp, var, tt, time, details, figcaption {
  292. font-size: calc( 1rem + .6 * (100vw - 20rem) / (50 - 20) );
  293. line-height: calc( 1.5em + 0.2 * (100vw - 50rem) / (20 - 50) );
  294. margin-left: 0;
  295. }
  296. h1 {
  297. font-size: calc( 1.9rem + 1.5 * (100vw - 20rem) / (50 - 20) );
  298. line-height: calc( 1.2em + 0.2 * (100vw - 50rem) / (20 - 50) );
  299. }
  300. h2 {
  301. font-size: calc( 1.5rem + 1.5 * (100vw - 20rem) / (50 - 20) );
  302. line-height: calc( 1.3em + 0.2 * (100vw - 50rem) / (20 - 50) );
  303. }
  304. h3 {
  305. font-size: calc( 1.35rem + 1.5 * (100vw - 20rem) / (50 - 20) );
  306. line-height: calc( 1.4em + 0.2 * (100vw - 50rem) / (20 - 50) );
  307. }
  308. }
  309. @media (min-width: 50em) {
  310. /* The right part of the addition *must* be a
  311. rem value. In this example we *could* change
  312. the whole declaration to font-size:2.5rem,
  313. but if our baseline value was not expressed
  314. in rem we would have to use calc. */
  315. p, li, pre, code, kbd, samp, var, tt, time, details, figcaption {
  316. font-size: calc( 1rem + .6 * 1rem );
  317. line-height: 1.5em;
  318. }
  319. p, li, pre, details {
  320. margin-left: 3rem;
  321. }
  322. h1 {
  323. font-size: calc( 1.9rem + 1.5 * 1rem );
  324. line-height: 1.2em;
  325. }
  326. h2 {
  327. font-size: calc( 1.5rem + 1.5 * 1rem );
  328. line-height: 1.3em;
  329. }
  330. h3 {
  331. font-size: calc( 1.35rem + 1.5 * 1rem );
  332. line-height: 1.4em;
  333. }
  334. figure img {
  335. max-width: 500px;
  336. max-height: 500px;
  337. }
  338. }
  339. figure.unsquared {
  340. margin-bottom: 1.5rem;
  341. }
  342. figure.unsquared img {
  343. height: inherit;
  344. }
  345. @media print {
  346. body { font-size: 100%; }
  347. a:after { content: " (" attr(href) ")"; }
  348. a, a:link, a:visited, a:after {
  349. text-decoration: underline;
  350. text-shadow: none !important;
  351. background-image: none !important;
  352. background: white;
  353. color: black;
  354. }
  355. abbr[title] { border-bottom: 0; }
  356. abbr[title]:after { content: " (" attr(title) ")"; }
  357. img { page-break-inside: avoid; }
  358. @page { margin: 2cm .5cm; }
  359. h1, h2, h3 { page-break-after: avoid; }
  360. p3 { orphans: 3; widows: 3; }
  361. img {
  362. max-width: 250px !important;
  363. max-height: 250px !important;
  364. }
  365. nav, aside { display: none; }
  366. }
  367. ul.with_columns {
  368. column-count: 1;
  369. }
  370. @media (min-width: 20em) {
  371. ul.with_columns {
  372. column-count: 2;
  373. }
  374. }
  375. @media (min-width: 50em) {
  376. ul.with_columns {
  377. column-count: 3;
  378. }
  379. }
  380. ul.with_two_columns {
  381. column-count: 1;
  382. }
  383. @media (min-width: 20em) {
  384. ul.with_two_columns {
  385. column-count: 1;
  386. }
  387. }
  388. @media (min-width: 50em) {
  389. ul.with_two_columns {
  390. column-count: 2;
  391. }
  392. }
  393. .gallery {
  394. display: flex;
  395. flex-wrap: wrap;
  396. justify-content: space-around;
  397. }
  398. .gallery figure img {
  399. margin-left: 1rem;
  400. margin-right: 1rem;
  401. }
  402. .gallery figure figcaption {
  403. font-family: 'ConcourseT3'
  404. }
  405. footer {
  406. font-family: 'ConcourseT3';
  407. display: flex;
  408. flex-direction: column;
  409. border-top: 3px solid white;
  410. padding: 4rem 0;
  411. background-color: #07486c;
  412. color: white;
  413. }
  414. footer > * {
  415. max-width: 50rem;
  416. margin: 0 auto;
  417. }
  418. footer a {
  419. color: #f1c40f;
  420. }
  421. footer .avatar {
  422. width: 200px;
  423. height: 200px;
  424. border-radius: 50%;
  425. float: left;
  426. -webkit-shape-outside: circle();
  427. shape-outside: circle();
  428. margin-right: 2rem;
  429. padding: 2px 5px 5px 2px;
  430. background: white;
  431. border-left: 1px solid #f1c40f;
  432. border-top: 1px solid #f1c40f;
  433. border-right: 5px solid #f1c40f;
  434. border-bottom: 5px solid #f1c40f;
  435. }
  436. </style>
  437. <h1>
  438. <span><a id="jumper" href="#jumpto" title="Un peu perdu ?">?</a></span>
  439. 30 options de commande Git qui gagnent à être connues (archive)
  440. <time>Pour la pérennité des contenus liés. Non-indexé, retrait sur simple email.</time>
  441. </h1>
  442. <section>
  443. <article>
  444. <h3><a href="http://www.git-attitude.fr/2014/09/15/30-options-git-qui-gagnent-a-etre-connues/">Source originale du contenu</a></h3>
  445. <p>Vous croyez connaître Git ? Peut-être bien… Et pourtant, je parierais ma chemise<sup id="fnref:1"><a href="#fn:1" rel="footnote">1</a></sup> que pas mal de petites options sympathiques vous sont inconnues.</p>
  446. <p>En effet, au fil des versions de Git, beaucoup de petites options font surface, soit pour des raisons de confort, soit pour de la puissance brute. Mais comme ce n’est pas une nouvelle commande, ou pas trop mis en avant dans les annonces de sortie, ça passe sous le radar.</p>
  447. <p>Je vous ai sélectionné une trentaine d’options, répartie sur une quinzaine de commandes, qui rendent la vie plus belle quand on fait du Git. Voici une excellente manière de rentabiliser les prochaines minutes !</p>
  448. <p><em>(The English version of this is <a href="https://medium.com/@porteneuve/30-git-cli-options-you-should-know-about-15423e8771df">on Medium</a>)</em></p>
  449. <p>J’indique intentionnellement, le plus souvent, l’option que je vais détailler dans le titre de la section. Cependant, n’en profitez pas pour sauter le texte sous prétexte que vous avez déjà vu l’option : il se peut que je l’utilise sur une commande, ou pour une raison, que vous n’avez pas vue venir ! Et puis, j’ajoute souvent des infos complémentaires sur des options corollaires ou des variables de configuration associées.</p>
  450. <h2>Staging / unstaging partiel avec <code>-p</code></h2>
  451. <p>Vous avez ouvert un fichier pour une raison précise, par exemple passer ce fichu tracker en asynchrone… Et voilà que vous remarquez au passage que des rôles ARIA manquent à certains éléments d’UX, et aussi que le footer est toujours en dur au lieu de dépendre du layout, et que sais-je encore…</p>
  452. <p>Au moment de faire le commit, vous vous rendez compte que ce fichier contient une bonne demi-douzaine (si ce n’est plus) de modifications qui <strong>recouvrent plusieurs sujets bien distincts</strong>. À ce moment-là, trois possibilités :</p>
  453. <ul>
  454. <li>Vous faites un bon gros <strong>commit fourre-tout dégueulasse</strong>, avec un message à base de « + » voire, si vous êtes d’humeur flemmasse, le bon vieux « Correctifs », « Plein de modifs », « Plein de trucs »… C’est gentil de pourrir l’historique comme ça, merci, merci bien.</li>
  455. <li>Vous <strong>copiez-collez le fichier</strong> quelque part puis enquillez les <em>Undo</em>, si tant est que ça soit possible, pour ne garder ensuite que le premier sujet, committer, puis ré-appliquer le 2ème, committer, puis le 3ème… Tout ça à la main, bien entendu. <strong>Probabilité de gaufrage : 99%.</strong></li>
  456. <li>Vous nous lisez ou allez à nos formations, et vous connaissez <code>-p</code> !</li>
  457. </ul>
  458. <p>La commande <code>git add -p</code> est en fait un affinage de <code>git add -i</code> : elle pré-sélectionne <strong>le mode <em>patch</em> de l’ajout interactif</strong>. Dans la pratique, vous indiquez aussi le fichier, pour faire bonne mesure. Par exemple :</p>
  459. <p>J’en profite pour vous rappeler que <code>git add</code> ne sert <strong>pas</strong> à mettre sous gestion de version, mais plutôt à <strong><em>stager</em></strong> une modification, c’est-à-dire à la valider pour le prochain commit.</p>
  460. <p>Lorsque vous faites un tel ajout, Git va auto-découper le contenu en <em>hunks</em>, en groupes de modifications, par proximité dans le fichier, en se basant sur les lignes inchangées pour faire sa découpe. Si vos modifs sont trop proches, il sera sans doute tenté de ne pas pré-découper, et vous devrez le faire vous-mêmes avec la commande <code>s</code> (Git vous proposera une pléthore de commandes par leur initiale en bas d’affichage ; dans le doute, utilisez <code>?</code> pour afficher l’aide), pour <em>split</em> (découper).</p>
  461. <p>Notez que même si vous avez des modifs <em>adjacentes</em> (sur des lignes qui se suivent immédiatement, sans ligne inchangée pour la découpe), vous pouvez éditer à la volée le <em>hunk</em> pour l’amener à la version que vous voulez <em>stager</em>, en utilisant la commande <code>e</code> (<em>edit</em>) pour ce faire. C’est un peu comme du Photoshop express sur la photo du fichier qui part en <em>stage</em>. En fait, si vous savez d’entrée de jeu que le fichier que vous allez <em>stager</em> contient des modifs adjacentes à retoucher, vous pouvez utiliser <code>-e</code> au lieu de <code>-p</code> pour présélectionner ce mode retouche, mais vous n’aurez alors pas la découpe automatique des <em>hunks</em> pour vous.</p>
  462. <p>À l’issue de l’opération, votre fichier apparaîtra normalement <strong>à la fois <em>staged</em> et <em>modified</em></strong>. C’est bien normal, car en effet :</p>
  463. <ul>
  464. <li>La dernière version committée n’est pas la même que dans le <em>stage</em> : le fichier est donc <em>staged</em></li>
  465. <li>La version dans le <em>stage</em> n’est pas la même que celle dans le <em>working directory</em> : le fichier est donc <em>modified</em>.</li>
  466. </ul>
  467. <p>Vous pouvez consulter le diff du <em>stage</em> pour ce fichier avec un <code>git diff --staged index.html</code>. Si vous voulez carrément voir le <em>snapshot</em> présent dans le <em>stage</em>, plutôt qu’une série de diffs : <code>git show :0:index.html</code>.</p>
  468. <p>Du coup, ayez soin de ne pas faire un <code>git commit -a</code> (par exemple <code>git ci -am "Tracker asynchrone"</code>), car ce <code>-a</code> va auto-<em>stager</em> toutes les modifs connues, écrasant au passage le <em>stage</em> « sculpté » que vous aviez mis en place.</p>
  469. <p>Enfin, peu de gens savent que <code>git reset</code> aussi a une option <code>-p</code>, qui a exactement la même UX que pour <code>add</code>, mais réalise évidemment l’inverse : elle <em>unstage</em> les <em>hunks</em> retenus. On l’utilise souvent pour redécouper le dernier commit effectué, en faisant:</p>
  470. <p>Les modifs qui nous sont alors présentées sont des annulations de celles réalisées par le dernier commit en date. On valide les annulations qu’on veut, on amende le commit (voir plus loin), puis on réalise le ou les commits complémentaire(s) qu’on souhaitait effectuer pour la découpe.</p>
  471. <p>C’est une manière très « vite et bien » de piloter une découpe de commit au sein d’un <a href="http://www.git-attitude.fr/2014/05/04/bien-utiliser-git-merge-et-rebase/#d-coupe%20au%20laser"><em>rebase</em> interactif</a>, à l’aide d’une commande de script <code>edit</code>.</p>
  472. <h2>Prendre en compte les renommages d’un coup avec <code>-A</code></h2>
  473. <p>Vous le savez peut-être, par défaut (en tout cas avant la 2.0), <code>git add</code> se comportait en fait comme <code>git add --no-all</code> ou, si vous préférez, <code>git add --ignore-removal</code>. Il se basait uniquement sur le <em>working directory</em> pour établir sa liste de fichiers à prendre en compte, ce qui comprenait donc :</p>
  474. <ul>
  475. <li>Les modifications aux fichiers connus</li>
  476. <li>Les nouveaux fichiers</li>
  477. </ul>
  478. <p>En revanche, les fichiers connus de l’index Git et introuvables sur le disque, qui apparaissent donc dans le statut comme supprimés, n’étaient pas traités.</p>
  479. <p><strong>C’était gênant pour les renommages ou déplacements</strong>, qui entraînent à la fois une « suppression » de l’ancien chemin de fichier, et l’apparition du nouveau.</p>
  480. <p>Pour ça, on a <code>git add -A</code>, ou sa version plus longue, <code>git add --all</code><sup id="fnref:2"><a href="#fn:2" rel="footnote">2</a></sup>. <strong>Ça prend la totale en compte.</strong> Lorsque l’index a connaissance des deux mouvements, il peut se rendre compte que c’est un renommage (même si une partie du contenu a bougé par-dessus le marché), ce qui permet notamment de demander plus tard à <code>git log</code> de suivre le fichier à travers ces renommages.</p>
  481. <p><strong>À partir de Git 2.0</strong>, c’est le comportement par défaut de <code>git add</code> si on lui précise un chemin (genre <code>git add .</code>). Au passage autre changement important en 2.0 : avant, si on faisait <code>git add -A</code> sans chemin, ça ne prenait que le répertoire courant et ses sous-dossiers, alors que maintenant, ça prend <em>tout le dépôt</em>, où qu’on soie à l’intérieur.</p>
  482. <h2>Entrer dans les répertoires <em>untracked</em> lors du <code>status</code></h2>
  483. <p>Vous l’avez sûrement remarqué : quand on rajoute un dossier à son dépôt, <code>git status</code> se contente de nous lister le dossier lui-même comme <em>untracked</em>, et non son contenu. Imaginons par exemple que je viens de rajouter un super plugin à mon projet, avec un JS et une CSS à l’intérieur :</p>
  484. <p><img src="http://www.git-attitude.fr/images/n-options-status-untracked-1.png"/></p>
  485. <p>C’est assez pénible, je trouve… Pourtant, on peut demander à <code>status</code> de rentrer dans les dossiers avec <code>-u</code> :</p>
  486. <p><img src="http://www.git-attitude.fr/images/n-options-status-untracked-2.png"/></p>
  487. <p>Je trouve d’ailleurs ça tellement pratique que je me suis collé la bonne variable de configuration pour le rendre systématique :</p>
  488. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  489. </pre></td><td class="code"><pre><code class=""><span class="line">git config --global status.showUntrackedFiles all</span></code></pre></td></tr></table></div></figure>
  490. <h2>Faire des diffs plus utiles</h2>
  491. <p>Les diffs fournis par <code>git diff</code>, <code>git log</code> et <code>git show</code>, pour ne citer qu’eux, sont sympa, mais y’a carrément moyen de les améliorer. En voici trois qui me tiennent particulièrement à cœur :</p>
  492. <p>Ou sa version longue, <code>git diff --ignore-all-space</code> (plus explicite, forcément). Cette option permet aux diff d’ignorer tout changement de whitespace à l’intérieur ou en bordure des lignes, y compris lorsqu’on passe de la présence à l’absence de whitespace, ou inversement (à l’exception de lignes entières : 1+ ligne vide là où on n’en avait aucune, ou inversement).</p>
  493. <p>Ça peut te jouer des tours quand tu travailles sur des fichiers à indentation significative, mais pour tous les autres cas, c’est une super manière de « dépolluer » l’affichage pour se concentrer sur les parties utiles.</p>
  494. <p>Sans <code>-w</code> :</p>
  495. <p><img src="http://www.git-attitude.fr/images/n-options-diff-regular.png"/></p>
  496. <p>Avec <code>-w</code> :</p>
  497. <p><img src="http://www.git-attitude.fr/images/n-options-diff-w.png"/></p>
  498. <p>L’autre option que j’adore a trait au mode d’affichage unitaire des diffs. Par défaut, on fonctionne par ligne, ce qui est parfois un peu limite :</p>
  499. <p><img src="http://www.git-attitude.fr/images/n-options-diff-lines.png"/></p>
  500. <p>Il est déjà possible de demander à <code>diff</code> de n’afficher qu’une fois la ligne, en découpant par mot, avec <code>--word-diff</code>, la définition de « mot » étant basée sur la présence de <em>whitespace</em>. Pour du rédactionnel, ce n’est pas gênant :</p>
  501. <p><img src="http://www.git-attitude.fr/images/n-options-diff-words.png"/></p>
  502. <p>Si vous trouvez que les cadrages +/- pourrissent un peu l’affichage, vous pouvez utiliser <code>--word-diff=color</code> pour affiner ça. En fait, on peut même raccourcir ça en <code>--color-words</code> (si c’est pas mignon…).</p>
  503. <p><img src="http://www.git-attitude.fr/images/n-options-diff-color-words.png"/></p>
  504. <p>Mais ça nous laisse avec un souci lorsqu’on diffe du code, où les <em>whitespaces</em> sont loin d’être les seuls délimiteurs utiles. Voyez plutôt :</p>
  505. <p><img src="http://www.git-attitude.fr/images/n-options-diff-words-suck.png"/></p>
  506. <p>Du coup on va préférer indiquer une <a href="http://www.js-attitude.fr/2012/08/13/enfin-maitriser-les-expressions-rationnelles/">regex</a> qui décrit « ce qu’est un mot ». À l’extrême, on dira « au minimum un caractère, quelconque » avec juste <code>.</code>, mais en cas de découpe résultante excessive, on allongera peut-être, genre <code>.{3,}</code> pour au moins 3 caractères.</p>
  507. <p>Ça donnerait une commande un peu longuette : <code>--word-diff=color --word-diff-regex=.</code> ce qui, comme les notes d’examen du <a href="http://fr.wikiquote.org/wiki/La_Cit%C3%A9_de_la_peur">commissaire Bialès</a>, est bien mais pas top. On préfèrera la version courte :</p>
  508. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  509. </pre></td><td class="code"><pre><code class=""><span class="line">git diff --color-words=.</span></code></pre></td></tr></table></div></figure>
  510. <p>Démonstration :</p>
  511. <p><img src="http://www.git-attitude.fr/images/n-options-diff-words-regex.png"/></p>
  512. <p>Notez que si vous souhaitez toujours passer par une telle regex (c’est souvent mon cas), vous pouvez configurer l’option <code>diff.wordRegex</code> à la valeur souhaitée (par exemple <code>.</code>), de sorte que tout type de <em>word diff</em> (<code>--word-diff</code> ou <code>--color-words</code>) sans argument utilisera cette regex (si vous précisez la regex manuellement, la vôtre a naturellement priorité).</p>
  513. <h2>Réparer le dernier commit avec <code>--amend</code></h2>
  514. <p>Il arrive souvent qu’en lisant le bilan affiché par un <code>git commit</code>, on se rende compte qu’on a fait une bévue : on a oublié un fichier, committé un fichier en trop, ce genre de choses…</p>
  515. <p>Une manière simple de corriger le tir, tant qu’on n’a pas <em>pushé</em> ce commit auprès des copains, consiste à se replacer dans la bonne situation (par exemple faire un <code>git add</code>, <code>git reset</code> ou <code>git rm --cached</code> sur le fichier problématique, peut-être accompagné d’un complément dans le <code>.gitignore</code>…) puis à faire :</p>
  516. <p>Cette option n’est en fait rien de plus qu’un <code>git reset --soft HEAD^</code> suivi du commit demandé, mais comme la majorité des gens <em>ne maîtrisent pas <code>reset</code></em>, ça facilite ce cas de figure.</p>
  517. <p>Notez que, la plupart du temps, le message de commit initial était le bon. Je doute que vous ayiez dit « Passage à Bootstrap 3.1, et au fait je versionne à tort le mdp root du serveur » dans le commit original. Du coup, pour vous éviter de retaper le message, ou simplement d’avoir à le revalider dans l’éditeur, vous pouvez faire :</p>
  518. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  519. <span class="line-number">2</span>
  520. <span class="line-number">3</span>
  521. <span class="line-number">4</span>
  522. </pre></td><td class="code"><pre><code class=""><span class="line">git commit --amend --no-edit
  523. </span><span class="line">
  524. </span><span class="line"># Ou si vous êtes antérieur à 1.7.9 :
  525. </span><span class="line">git commit --amend -C HEAD</span></code></pre></td></tr></table></div></figure>
  526. <p>C’est un cas de figure tellement courant que je vois souvent un alias <code>git oops</code> pour cette commande :-)</p>
  527. <p>Dans votre historique, seule la dernière version du commit est visible : c’est comme si vous n’aviez pas dérapé à la base (trop fort !). La, voire les, anciennes versions du commit sont évidemment toujours présentes dans votre <a href="https://www.atlassian.com/fr/git/tutorial/rewriting-git-history#!reflog">reflog</a>, la maxime générale de Git restant vraie : « si ça a été commité, c’est virtuellement impossible à paumer ».</p>
  528. <h2>Fouiller intelligemment les logs avec <code>-S</code> et <code>-G</code></h2>
  529. <p>La commande <code>git log</code> est truffée d’options (plus d’une centaine !), dont elle partage la majeure partie avec sa cousine germaine, <code>git diff</code>.</p>
  530. <p>Nombre de ces options servent à <strong>filtrer le log</strong> avant même de l’afficher (ce qui est infiniment plus performant et pratique que de faire un <code>grep</code> par-dessus) : filtrage sur dates, chemins, branches, auteurs et committers, messages de commit… mais aussi <strong>contenus des diffs</strong>.</p>
  531. <p>Le filtrage sur diffs est extrêmement utile pour <strong>retrouver l’origine d’un code, voire d’un bug</strong>. Trop de gens ont tendance à passer par <code>git blame</code>, parce qu’ils étaient habitués à <code>svn blame</code>, mais cette commande est tout aussi limitée que sa consœur chez SVN :</p>
  532. <ul>
  533. <li>Elle affiche juste quel commit <em>a touché à la ligne en dernier</em>, sans dire pourquoi ; si ça se trouve, ça virait les espaces de fin de ligne, c’est tout</li>
  534. <li>Elle n’affiche <em>que les lignes actuelles</em>, donc si le problème est qu’une ligne a dégagé, ou a été déplacée, ça ne nous aide en rien.</li>
  535. </ul>
  536. <p>En revanche, si on filtre le log sur <strong>les contenus des diffs</strong>, on saura effectivement <strong>quel commit a introduit le changement qui nous intéresse</strong>.</p>
  537. <p>Si on s’intéresse juste à la présence de la chaîne dans les lignes activecs du diff, peu importe pourquoi et comment, on utilisera plutôt <code>-G</code> (penser alors à échapper les caractères spéciaux de regex) :</p>
  538. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  539. </pre></td><td class="code"><pre><code class=""><span class="line">git log -G'Secure_?Random' -2 -- path/to/problematic_file</span></code></pre></td></tr></table></div></figure>
  540. <p>(Généralement, dans une telle recherche, on limite aux 1 ou 2 commits les plus récents.)</p>
  541. <p>En revanche, si on cherche spécifiquement des diffs qui ont <strong>retiré ou introduit le texte</strong>, on passera par <code>-S</code>, qui ne renvoie que les diffs ayant <strong>changé le nombre d’occurrences</strong> du texte. Par défaut, <code>-S</code> prend un texte simple, mais si vous voulez une regex, vous pouvez rajouter <code>--pickaxe-regex</code> :</p>
  542. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  543. </pre></td><td class="code"><pre><code class=""><span class="line">git log -S'Secure_?Random' --pickaxe-regex -2 -- path/to/problematic_file</span></code></pre></td></tr></table></div></figure>
  544. <p>Si vous avez besoin que vos textes, ou regex, soient insensibles à la casse, ajoutez <code>-i</code>. Les regex sont systématiquement traitées comme de la syntaxe étendue (ERE). Enfin, si vous souhaitez afficher le diff à la volée (qui peut être lourd, attention), ajoutez comme d’habitude <code>-p</code> (raison de plus pour pré-filtrer sur le fichier qui vous intéresse).</p>
  545. <h2>Accélérer les branches avec <code>-b</code>, <code>-v</code> et <code>-vv</code></h2>
  546. <p>Alors déjà, si vous ne saviez pas encore que <code>git checkout -b</code> crée la branche cible à la volée, maintenant vous n’avez plus d’excuse.</p>
  547. <p>Pourquoi se faire ch… à faire :</p>
  548. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  549. <span class="line-number">2</span>
  550. </pre></td><td class="code"><pre><code class=""><span class="line">$ git branch ticket-12
  551. </span><span class="line">$ git checkout ticket-12</span></code></pre></td></tr></table></div></figure>
  552. <p>Alors qu’on peut parfaitement faire :</p>
  553. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  554. </pre></td><td class="code"><pre><code class=""><span class="line">git checkout -b ticket-12</span></code></pre></td></tr></table></div></figure>
  555. <p>Bien sûr, rien ne vous empêche d’indiquer en 2ème argument la base de la branche (par défaut le <code>HEAD</code>, comme pratiquement tout le temps).</p>
  556. <p>Notez un cas où le <code>-b</code> est superflu car <code>checkout</code> est super-malin : lorsque vous voulez commencer à bosser sur la branche <code>super-feature</code> présente sur le <em>remote</em>, et que vous n’avez donc pas encore de branche locale qui la tracke. Si vous faites tout bêtement :</p>
  557. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  558. </pre></td><td class="code"><pre><code class=""><span class="line">git checkout super-feature</span></code></pre></td></tr></table></div></figure>
  559. <p>Git va se rendre compte que la branche locale <code>super-feature</code> n’existe pas, mais qu’il y en a une sur le <em>remote</em> par défaut, et va <strong>automatiquement</strong> faire l’équivalent de ceci (en supposant que votre <em>remote</em> par défaut s’appelle <code>origin</code>, ce qui est généralement le cas) :</p>
  560. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  561. </pre></td><td class="code"><pre><code class=""><span class="line">git checkout -b -t super-feature origin/super-feature</span></code></pre></td></tr></table></div></figure>
  562. <p>Pourquoi faire long quand on peut faire court, hein ?</p>
  563. <p>Parlons à présent de <code>-v</code> et sa version agressive, <code>-vv</code>.</p>
  564. <p>Vous avez sans doute l’habitude lister vos branches locales avec un simple <code>git branch</code> :</p>
  565. <p><img src="http://www.git-attitude.fr/images/n-options-git-branch-simple.png"/></p>
  566. <p>Saviez-vous qu’en fait vous pouvez avoir un peu plus d’infos (SHA, écart avec l’upstream éventuel, première ligne du message de commit) avec <code>-v</code> ?</p>
  567. <p><img src="http://www.git-attitude.fr/images/n-options-git-branch-v.png"/></p>
  568. <p>Et même vérifier l’upstream tracké en poussant jusqu’à <code>-vv</code> ?</p>
  569. <p><img src="http://www.git-attitude.fr/images/n-options-git-branch-vv.png"/></p>
  570. <p>Sympa, non ? En temps normal le nom de l’upstream est en bleu, mais ça pue sur du noir, j’ai donc configuré <code>color.branch.upstream</code> à <code>cyan</code>…</p>
  571. <h2>Une aide plus facile d’emploi avec <code>-w</code></h2>
  572. <p>Les <em>manpages</em>, c’est bien, facile, et ça marche de partout, y compris au travers d’une connexion SSH. … Enfin, presque partout. Certains environnements ne gèrent pas très bien <code>man</code>.</p>
  573. <p>Et puis, la grande majorité des gens ne savent pas utiliser interactivement <code>man</code> autrement que pour défiler à travers le contenu (et encore). Le recours aux hyperliens, notamment, est extrêmement rare.</p>
  574. <p><strong>Git publie toutes ses documentations</strong> non seulement au format <code>man</code> mais aussi <strong>au format HTML</strong>, un type de document utilisable par tous, liens compris. Pour y recourir, il suffit d’utiliser l’option <code>-w</code> de <code>git help</code> :</p>
  575. <p>Sur certains environnements (l’installeur officiel pour Windows, je crois), c’est d’ailleurs le mode <strong>automatique</strong>. Si vous souhaitez vous aussi fonctionner comme ça, vous pouvez configurer ça dans votre config globale :</p>
  576. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  577. </pre></td><td class="code"><pre><code class=""><span class="line">git config --global help.format html</span></code></pre></td></tr></table></div></figure>
  578. <p>Si le navigateur employé par défaut ne vous convient pas (déterminé par <code>git-web--browse</code>, qui en connaît une tera-chiée), vous pouvez configurer <code>help.browser</code> avec la commande ou le chemin d’invocation de celui que vous désirez.</p>
  579. <p>Notez que ces contenus restent locaux (fichiers HTML installés par Git), vous n’avez donc même <strong>pas besoin d’une connexion à Internet</strong>.</p>
  580. <h2>Stasher plus efficacement avec <code>save</code> et <code>-u</code></h2>
  581. <p><strong>Trop de gens ne connaissent pas <a href="http://git-scm.com/docs/git-stash"><code>git stash</code></a></strong>, et ceux qui le connaissent ont rarement fait l’effort d’en lire la doc pour apprendre à <strong>bien l’utiliser</strong>. Je vois la plupart des gens juste faire un <code>git stash</code> en amont, et un <code>git stash apply</code> en aval.</p>
  582. <p>Le comportement par défaut de <code>stash</code> est assez pénible :</p>
  583. <ol>
  584. <li>Il laisse de côté les <em>untracked</em>, ce qui est rarement désirable</li>
  585. <li>Il fournit un message pourri par défaut (genre <code>WIP on master: &lt;message du dernier commit connu&gt;</code>)</li>
  586. </ol>
  587. <p>Un tel message n’a aucun intérêt, vu qu’il <strong>ne précise pas la nature du <em>work in progress</em></strong> (WIP), ce qui rend difficile l’identification du stash par la suite.</p>
  588. <p>Pour pouvoir régler ces deux points, il suffit de recourir explicitement à la <strong>sous-commande <code>save</code>, avec l’option <code>-u</code></strong> d’une part (pour intégrer les <em>untracked</em>), et le message voulu d’autre part. Par exemple :</p>
  589. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  590. </pre></td><td class="code"><pre><code class=""><span class="line">git stash save -u 'Début du refactoring Bootstrap 3'</span></code></pre></td></tr></table></div></figure>
  591. <p>On se retrouve alors sur un <em>clean tree</em> : le contenu du HEAD, plus les éventuels fichiers ignorés.</p>
  592. <p>Il est tout aussi important de <strong>bien récupérer le stash obtenu</strong> lorsqu’on aura fini de traiter l’urgence qui nous l’a fait créer à la base.</p>
  593. <p>Beaucoup de gens font <code>git stash apply</code>, ce qui est un peu dommage parce qu’en cas de succès (pas de conflit avec le nouvel état de base), le stash continue d’exister, ce qui pourrait nous tenter plus tard de le réappliquer, à tort évidemment.</p>
  594. <p>Ce qu’il faudrait en fait, c’est que si l’<code>apply</code> marche, on exécute automatiquement le <code>drop</code>. Et c’est précisément le sens de <code>git stash pop</code>.</p>
  595. <p>Contrairement à ce que son nom peut laisser penser, celui-ci ne se limite pas au dernier stash existant : on peut en préciser un autre (par exemple <code>git stash pop stash@{2}</code>). Ceci dit, je trouve que <strong>les stashes ne devraient exister que très peu de temps</strong>, pour contourner une situation complexe ou un obstacle, et qu’on ne devrait donc pratiquement jamais en avoir plusieurs.</p>
  596. <p>Par ailleurs, il est dommage que <code>apply</code> et <code>pop</code> ne restaurent pas, par défaut, l’état du <em>stage</em>. Celui-ci est pourtant bien sauvegardé par <code>save</code>, toujours par défaut. Alors pourquoi ne pas le restaurer d’office, dans la mesure où le <em>stage</em> est une info importante ?</p>
  597. <p>C’est parce que la commande est peureuse : si vous avez modifié un fichier figurant dans le <em>stage</em> du stash, elle serait contrainte de fusionner les deux et de re-<em>stager</em> le résultat. Git fait ce genre de choses d’habitude (suite à un <code>merge</code>, <code>rebase</code> ou <code>cherry-pick</code> par exemple, avec ou sans assistance de <code>rerere</code>), mais sur ce coup, il le refusera.</p>
  598. <p>Du coup, pour ne pas avoir à hurler <em>en cas de fusion portant sur le stage</em>, par défaut, il ne restaure pas ce dernier : les snapshots concernés redeviennent de simples modifs locales sur le disque.</p>
  599. <p>Ça m’exaspère au plus haut point, de sorte que <strong>je lui demande toujours explicitement de tenter la restauration du <em>stage</em></strong>. Au pire, si j’ai en effet un souci car j’aurais modifié un fichier par ailleurs <em>staged</em>, je retirerai mon option et reconstruirai mon <em>stage</em> manuellement. J’utilise donc systématiquement l’option <code>--index</code> :</p>
  600. <p>Il n’existe hélas pas de variable de configuration pour systématiser ce comportement…</p>
  601. <p>Ah oui, et puis il est <strong>très facile d’oublier qu’on a stashé</strong> : pensez à avoir <a href="http://www.git-attitude.fr/2013/05/22/prompt-git-qui-dechire/">un bon prompt</a> avec la variable d’environnement <code>GIT_PS1_SHOWSTASHSTATE</code> activée.</p>
  602. <h2>La précédente branche active : <code>-</code></h2>
  603. <p>Vous savez peut-être que dans les shells usuels, <code>cd -</code> vous permet de revenir au répertoire précédent (du coup, l’utilisation répétée de la commande bascule entre deux répertoires).</p>
  604. <p>Au fil des versions, Git a successivement appris cette astuce pour les commandes <code>checkout</code>, <code>merge</code>, <code>cherry-pick</code> et <code>rebase</code>. Voici un schéma classique :</p>
  605. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  606. <span class="line-number">2</span>
  607. </pre></td><td class="code"><pre><code class=""><span class="line">(topic) $ git checkout master
  608. </span><span class="line">(master) $ git merge -</span></code></pre></td></tr></table></div></figure>
  609. <p>Encore un autre :</p>
  610. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  611. <span class="line-number">2</span>
  612. <span class="line-number">3</span>
  613. </pre></td><td class="code"><pre><code class=""><span class="line">(2-3-stable +) $ git ci -m "fix: no more conflict with Underscore. Fixes #217."
  614. </span><span class="line">(2-3-stable) $ git checkout master
  615. </span><span class="line">(master) $ git cherry-pick -</span></code></pre></td></tr></table></div></figure>
  616. <p>Si vous êtes sur une version qui n’a pas cette astuce pour votre commande (<a href="http://www.git-attitude.fr/2014/08/28/git-evolutions-recentes/">vérifiez</a>), ou que vous en avez besoin pour une commande qui ne gère pas encore la syntaxe, n’oubliez pas que c’est juste du sucre syntaxique pour la syntaxe bien plus ancienne et universelle <code>@{-1}</code>.</p>
  617. <h2>Annuler le merge en cours sans effacer les modifs locales</h2>
  618. <p><strong>Git n’a pas besoin d’un <em>clean tree</em></strong> pour autoriser une fusion : il lui suffit que le <em>working directory</em> (WD) soit <a href="http://git-scm.com/docs/git-merge#_pre_merge_checks"><em>in good order</em></a>, c’est-à-dire, principalement, que les fichiers qui vont être touchés par la fusion ne fassent pas actuellement l’objet de modifs locales, et d’autre part qu’il n’y ait pas déjà un <em>stage</em> en cours (pour éviter à terme un commit fourre-tout).</p>
  619. <p>Du coup, lorsque vous vous retrouvez à arbitrer des conflits de fusion, vous pouvez avoir sous la main des modifs issues de conflits de fusion, mais aussi d’autres qui étaient là avant de lancer <code>git merge</code>.</p>
  620. <p>Si vous abandonnez la fusion faute d’informations suffisantes, il peut être tentant de faire un bon vieux <code>git reset --hard</code>. Ce serait en fait <strong>dangereux</strong>, parce que ça purgerait aussi les modifs locales qui étaient là <strong>avant le <code>merge</code></strong>.</p>
  621. <p>C’est pourquoi il est possible de faire un <code>git reset --merge</code> (ou la syntaxe plus récente <code>git merge --abort</code>, qui ressemble à son équivalent pour le <code>rebase</code>) : vous ne réinitialiserez alors <strong>que ce qui vient de la fusion</strong>.</p>
  622. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  623. <span class="line-number">2</span>
  624. <span class="line-number">3</span>
  625. <span class="line-number">4</span>
  626. <span class="line-number">5</span>
  627. <span class="line-number">6</span>
  628. <span class="line-number">7</span>
  629. </pre></td><td class="code"><pre><code class=""><span class="line">(master *) $ git merge cool-feature
  630. </span><span class="line">Auto-merging index.html
  631. </span><span class="line">CONFLICT (content): Merge conflict in index.html
  632. </span><span class="line">Automatic merge failed; fix conflicts and then commit the result.
  633. </span><span class="line">
  634. </span><span class="line">(master *+) $ git merge --abort
  635. </span><span class="line">(master *) $</span></code></pre></td></tr></table></div></figure>
  636. <p>C’est en fait même faisable <strong>après une fusion réussie</strong> !</p>
  637. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  638. <span class="line-number">2</span>
  639. <span class="line-number">3</span>
  640. <span class="line-number">4</span>
  641. <span class="line-number">5</span>
  642. <span class="line-number">6</span>
  643. <span class="line-number">7</span>
  644. <span class="line-number">8</span>
  645. </pre></td><td class="code"><pre><code class=""><span class="line">(master *) $ git merge cool-feature
  646. </span><span class="line">Auto-merging index.html
  647. </span><span class="line">Merge made by the `recursive` strategy.
  648. </span><span class="line">[afbd564] Merged `cool-feature` branch
  649. </span><span class="line">
  650. </span><span class="line">(master *) $ git reset --merge ORIG_HEAD
  651. </span><span class="line">[ac3489b] Original master tip
  652. </span><span class="line">(master *) $</span></code></pre></td></tr></table></div></figure>
  653. <p>La classe, non ? Ça nous évite du stash, tout ça…</p>
  654. <h2>Éviter de flinguer un <em>merge</em> en le <em>rebasant</em></h2>
  655. <p>Le <em>rebasing</em> est décidément <a href="http://www.git-attitude.fr/2014/05/04/bien-utiliser-git-merge-et-rebase/">un super couteau suisse</a>, avec juste un petit risque potentiel : par défaut, <strong>lorsqu’il <em>rebase</em> un commit de fusion, il <em>inline</em> la fusion</strong>. En gros, on court le risque suivant (ici illustré au travers d’un <code>pull</code> qui <em>rebase</em> au lieu de fusionner) :</p>
  656. <p><img src="http://www.git-attitude.fr/images/git-merge-rebase-inlining.png" title="&quot;Un exemple d’inlining fortuit d’un commit de fusion lors du rebase&quot;" alt="&quot;Un exemple d’inlining fortuit d’un commit de fusion lors du rebase&quot;"/></p>
  657. <p>Pour éviter ça, il suffit de demander au <em>rebase</em> de <strong>préserver les fusions</strong>, à l’aide de l’option <code>-p</code>, ou sa version longue <code>--preserve-merges</code>.</p>
  658. <p><img src="http://www.git-attitude.fr/images/git-merge-rebase-preserved.png" title="&quot;Un rebase qui préserve les fusions&quot;" alt="&quot;Un rebase qui préserve les fusions&quot;"/></p>
  659. <p>Évitez toutefois de combiner avec des réordonnancements en mode interactif, <a href="http://git-scm.com/docs/git-rebase#_bugs">ça donnerait des résultats inattendus</a>.</p>
  660. <h2>Être un ninja avec le rebase <code>-i</code></h2>
  661. <p>C’est en mode interactif que le <em>rebase</em> donne <strong>toute sa mesure</strong>, le cas d’utilisation pluri-quotidien étant le réflexe de bonne pratique qui consiste à toujours évaluer son historique local avant un <em>push</em>, et à le <a href="http://www.git-attitude.fr/2014/05/04/bien-utiliser-git-merge-et-rebase/#nettoyer-son%20historique%20local%20avant%20envoi">mettre au propre</a> si besoin, à l’aide d’un <code>git rebase -i @{u}</code>.</p>
  662. <p>Purge de commits à somme nulle (genre l’original et son <em>revert</em>), réordonnancement de commits, fusion de tentatives successives pour un même correctif, réécriture des messages, découpe de commits fourre-tout… <a href="http://www.git-attitude.fr/2014/05/04/bien-utiliser-git-merge-et-rebase/#nettoyer-son%20historique%20local%20avant%20envoi">Tout est possible</a>.</p>
  663. <h2>Nettoyer sereinement avec <code>-i</code> et <code>-n</code></h2>
  664. <p>La commande <code>git clean</code> est bien utile, mais <strong>potentiellement destructrice</strong> : en effet, elle impacte le <em>working directory</em> (WD) et, du coup, peut <strong>dégager des modifs locales qui n’ont jamais été committées</strong>, de sorte que si on a fait une bévue, Git ne pourra pas nous aider à récupérer ce travail !</p>
  665. <p>Par défaut d’ailleurs, <code>git clean</code> est une <em>no-op</em>, car <code>clean.requireForce</code> est à <code>true</code>, il faut donc au moins un <code>git clean -f</code> pour envoyer la purée ; et même ça, ça n’entrera pas dans les sous-répertoires (sauf <code>-d</code>), et laissera de côté les fichiers ignorés (sauf <code>-x</code>).</p>
  666. <p>Vous avez toutefois la possibilité de <strong>voir à quoi ressemblera votre <code>clean</code> sans aucun risque</strong>, avec la traditionnelle option <code>-n</code> (ou <code>--dry-run</code>), présente sur de nombreuses commandes Git, qui listera les fichiers et dossiers qui seraient supprimés si elle n’était pas là, mais se bornera, justement, à les lister.</p>
  667. <p>Et pour passer à l’action, vous pouvez vous <strong>tranquilliser encore</strong> en utilisant <code>-i</code> (le traditionnel <code>--interactive</code>), qui vous proposera une sorte de shell listant les candidats à la suppression, et permet de les filtrer, de confirmer pour chaque, etc. Plus d’angoisse !</p>
  668. <h2>Caler l’upstream à la volée avec <code>-u</code></h2>
  669. <p>Vous <em>pushez</em> une branche pour la première fois ? Il vous faudra toujours expliciter le <em>remote</em> voulu (même si vous n’en avez qu’un de défini) et la branche souhaitée (même si elle est active), par exemple <code>git push origin topic</code>.</p>
  670. <p>Toutefois, ce simple <code>push</code> <strong>ne met pas en place le <em>tracking</em></strong> : la configuration locale de votre dépôt ne retient pas l’association entre votre branche locale <code>topic</code> et son <em>upstream</em>, ici la branche <code>topic</code> du <em>remote</em> <code>origin</code>.</p>
  671. <p>Pour y remédier, vous pouvez à tout moment refaire ce <code>push</code> en précisant <code>-u</code> (ou <code>--set-upstream</code>), qui calera cette configuration avant le <em>push</em> proprement dit, afin que vous n’ayez plus besoin de préciser tout ça lors des futurs <code>push</code> et <code>pull</code>.</p>
  672. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  673. </pre></td><td class="code"><pre><code class=""><span class="line">git push -u origin topic</span></code></pre></td></tr></table></div></figure>
  674. <p>En interne, ça fait un <code>git branch --set-upstream-to=origin/topic topic</code>, ce que vous pouvez donc faire à la main si vous souhaitez le <strong>configurer sans <em>pusher</em> tout de suite</strong> pour autant.</p>
  675. <p>Pour rappel, vous n’êtes pas obligés de connecter votre branche locale à une branche distante homonyme : si les noms doivent différer, il vous suffira de connecter l’<em>upstream</em> en utilisant la <strong>syntaxe totale de <code>push</code></strong>, par exemple ici pour avoir une branche distante <code>christophe-topic</code> pour ma branche locale <code>topic</code> :</p>
  676. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  677. </pre></td><td class="code"><pre><code class=""><span class="line">git push -u origin topic:christophe-topic</span></code></pre></td></tr></table></div></figure>
  678. <p>C’est d’ailleurs pour cette raison que la syntaxe de <strong>suppression de branche distante</strong> est :</p>
  679. <figure class="code"><div class="highlight"><table><tr><td class="gutter"><pre class="line-numbers"><span class="line-number">1</span>
  680. </pre></td><td class="code"><pre><code class=""><span class="line">git push origin :old-remote-branch</span></code></pre></td></tr></table></div></figure>
  681. <p>Elle revient à dire « remplacer la branche distante <code>old-remote-branch</code> par rien du tout »… donc la supprimer.</p>
  682. <h2>Ah tiens, 36 en fait</h2>
  683. <p>Eh oui, 36 options finalement (sans compter les variations courte/longue). Et 4 variables de configuration (plus une d’environnement). Que voulez-vous, on est comme ça chez Git Attitude, on donne, on donne…</p>
  684. <h2>Envie d’en savoir plus ?</h2>
  685. <p>Déjà, vous pouvez voir par le menu <a href="http://www.git-attitude.fr/2014/08/28/git-evolutions-recentes/">tout ce qui est apparu d’intéressant dans Git depuis la 1.7</a>, grâce à notre article dédié. Vous pouvez aussi aller fouiller dans le détail <a href="http://www.git-attitude.fr/2014/05/04/bien-utiliser-git-merge-et-rebase/">la bonne utilisation de merge vs. rebase</a>, si ce n’est déjà fait.</p>
  686. <p>Mais surtout, notre formation <a href="http://www.git-attitude.fr/git-total/">Git Total</a> explore ces thématiques et bien d’autres pour vous donner une compréhension en profondeur de Git, vous transformant en experts en seulement 3 jours, pour un tarif très raisonnable ! Disponible en inter-entreprises tous les 2 mois (hors été) et en intra-entreprises <a href="http://www.git-attitude.fr/demander-une-intra-ou-custom">sur demande</a>.</p>
  687. </article>
  688. </section>
  689. <nav id="jumpto">
  690. <p>
  691. <a href="/david/blog/">Accueil du blog</a> |
  692. <a href="http://www.git-attitude.fr/2014/09/15/30-options-git-qui-gagnent-a-etre-connues/">Source originale</a> |
  693. <a href="/david/stream/2019/">Accueil du flux</a>
  694. </p>
  695. </nav>
  696. <footer>
  697. <div>
  698. <img src="/static/david/david-larlet-avatar.jpg" loading="lazy" class="avatar" width="200" height="200">
  699. <p>
  700. Bonjour/Hi!
  701. Je suis <a href="/david/" title="Profil public">David&nbsp;Larlet</a>, je vis actuellement à Montréal et j’alimente cet espace depuis 15 ans. <br>
  702. Si tu as apprécié cette lecture, n’hésite pas à poursuivre ton exploration. Par exemple via les <a href="/david/blog/" title="Expériences bienveillantes">réflexions bimestrielles</a>, la <a href="/david/stream/2019/" title="Pensées (dés)articulées">veille hebdomadaire</a> ou en t’abonnant au <a href="/david/log/" title="S’abonner aux publications via RSS">flux RSS</a> (<a href="/david/blog/2019/flux-rss/" title="Tiens c’est quoi un flux RSS ?">so 2005</a>).
  703. </p>
  704. <p>
  705. Je m’intéresse à la place que je peux avoir dans ce monde. En tant qu’humain, en tant que membre d’une famille et en tant qu’associé d’une coopérative. De temps en temps, je fais aussi des <a href="https://github.com/davidbgk" title="Principalement sur Github mais aussi ailleurs">trucs techniques</a>. Et encore plus rarement, <a href="/david/talks/" title="En ce moment je laisse plutôt la place aux autres">j’en parle</a>.
  706. </p>
  707. <p>
  708. Voici quelques articles choisis :
  709. <a href="/david/blog/2019/faire-equipe/" title="Accéder à l’article complet">Faire équipe</a>,
  710. <a href="/david/blog/2018/bivouac-automnal/" title="Accéder à l’article complet">Bivouac automnal</a>,
  711. <a href="/david/blog/2018/commodite-effondrement/" title="Accéder à l’article complet">Commodité et effondrement</a>,
  712. <a href="/david/blog/2017/donnees-communs/" title="Accéder à l’article complet">Des données aux communs</a>,
  713. <a href="/david/blog/2016/accompagner-enfant/" title="Accéder à l’article complet">Accompagner un enfant</a>,
  714. <a href="/david/blog/2016/senior-developer/" title="Accéder à l’article complet">Senior developer</a>,
  715. <a href="/david/blog/2016/illusion-sociale/" title="Accéder à l’article complet">L’illusion sociale</a>,
  716. <a href="/david/blog/2016/instantane-scopyleft/" title="Accéder à l’article complet">Instantané Scopyleft</a>,
  717. <a href="/david/blog/2016/enseigner-web/" title="Accéder à l’article complet">Enseigner le Web</a>,
  718. <a href="/david/blog/2016/simplicite-defaut/" title="Accéder à l’article complet">Simplicité par défaut</a>,
  719. <a href="/david/blog/2016/minimalisme-esthetique/" title="Accéder à l’article complet">Minimalisme et esthétique</a>,
  720. <a href="/david/blog/2014/un-web-omni-present/" title="Accéder à l’article complet">Un web omni-présent</a>,
  721. <a href="/david/blog/2014/manifeste-developpeur/" title="Accéder à l’article complet">Manifeste de développeur</a>,
  722. <a href="/david/blog/2013/confort-convivialite/" title="Accéder à l’article complet">Confort et convivialité</a>,
  723. <a href="/david/blog/2013/testament-numerique/" title="Accéder à l’article complet">Testament numérique</a>,
  724. et <a href="/david/blog/" title="Accéder aux archives">bien d’autres…</a>
  725. </p>
  726. <p>
  727. On peut <a href="mailto:david%40larlet.fr" title="Envoyer un courriel">échanger par courriel</a>. Si éventuellement tu souhaites que l’on travaille ensemble, tu devrais commencer par consulter le <a href="http://larlet.com">profil dédié à mon activité professionnelle</a> et/ou contacter directement <a href="http://scopyleft.fr/">scopyleft</a>, la <abbr title="Société coopérative et participative">SCOP</abbr> dont je fais partie depuis six ans. Je recommande au préalable de lire <a href="/david/blog/2018/cout-site/" title="Attention ce qui va suivre peut vous choquer">combien coûte un site</a> et pourquoi je suis plutôt favorable à une <a href="/david/pro/devis/" title="Discutons-en !">non-demande de devis</a>.
  728. </p>
  729. <p>
  730. Je ne traque pas ta navigation mais mon
  731. <abbr title="Alwaysdata, 62 rue Tiquetonne 75002 Paris, +33.184162340">hébergeur</abbr>
  732. conserve des logs d’accès.
  733. </p>
  734. </div>
  735. </footer>
  736. <script type="text/javascript">
  737. ;(_ => {
  738. const jumper = document.getElementById('jumper')
  739. jumper.addEventListener('click', e => {
  740. e.preventDefault()
  741. const anchor = e.target.getAttribute('href')
  742. const targetEl = document.getElementById(anchor.substring(1))
  743. targetEl.scrollIntoView({behavior: 'smooth'})
  744. })
  745. })()
  746. </script>