Repository with sources and generator of https://larlet.fr/david/ https://larlet.fr/david/
Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

article.md 6.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. title: De Dotclear à Django : migration des données et redirections
  2. slug: de-dotclear-a-django-migration-des-donnees-et-redirections
  3. date: 2007-05-23 19:35:18
  4. type: post
  5. vignette: images/logos/biologeek.png
  6. contextual_title1: Biologeek (enfin) propulsé par Django
  7. contextual_url1: 20080423-biologeek-enfin-propulse-par-django
  8. contextual_title2: Ajout des flux RSS, du sitemap et des commentaires avec Django
  9. contextual_url2: 20070623-ajout-des-flux-rss-du-sitemap-et-des-commentaires-avec-django
  10. contextual_title3: Vues génériques, héritage et templatetags : développez rapidement avec Django
  11. contextual_url3: 20070424-vues-generiques-heritage-et-templatetags-developpez-rapidement-avec-django
  12. <p>À force de me <a href="http://www.prendreuncafe.com/blog/post/2007/05/22/Tags-faciles-avec-Propel-Symfony#c10885">faire</a> <a href="https://larlet.fr/david/biologeek/archives/20070516-torture/#c22043">chambrer</a> sur le <a href="https://larlet.fr/david/biologeek/archives/20070224-objectifs-et-motivations-de-la-refonte-de-ce-blog/">retard de ma refonte</a>, j'ai décidé de prendre le taureau par les cornes en attaquant la migration des données. Je me suis rendu compte que ce n'était finalement pas si difficile que ça avec Django...</p>
  13. <h2>Migration des données</h2>
  14. <p>La première étape est de récupérer les données depuis Dotclear. J'avais commencé à faire des scripts SQL mais <strong>j'ai finalement opté pour une solution en texte brut</strong> qui m'offrait plus de souplesse. J'ai donc installé le <a href="http://doc.dotclear.net/2.0/admin/1-to-2">plugin flatExport pour Dotclear</a> qui permet de récupérer le contenu du blog sous forme de fichier texte. Quelques lignes de python et l'on converti ce fichier en dictionnaires pour faciliter l'accès aux données (c'est un peu le relationnel du pauvre...)&nbsp;:</p>
  15. <pre>for table in open('blog-backup.txt').read().split('
  16. ',1)[1].strip().split('
  17. '):
  18. header, content = table.split('
  19. ',1)
  20. table_name, column_names = header[1:-1].split(' ')
  21. column_names = tuple(column_names.split(','))
  22. blog[table_name] = {}
  23. for item in content.split('
  24. '):
  25. column_contents = tuple((decode_html(column) for column in item[1:-1].split('","')))
  26. item_dict = dict(zip(column_names, column_contents))
  27. id = item_dict[column_names[0]]
  28. blog[table_name][id] = item_dict</pre>
  29. <p><strong>La grande force de Django c'est de pouvoir utiliser votre projet dans un script python</strong> afin de pouvoir appeler directement les objets/classes/tables en python (ce sont ces petits détails qui font apprécier un framework plutôt qu'un autre), c'est ce que j'ai fait pour remplir mes nouvelles tables avec les données issues de Dotclear&nbsp;:</p>
  30. <pre>for post_dict in blog['post'].values():
  31. django_post = Post()
  32. django_post.title = post_dict['post_titre']
  33. django_post.slug = post_dict['post_titre_url']
  34. django_post.tags = blog['categorie'][post_dict['cat_id']]['cat_libelle_url'].lower()
  35. django_post.summary_html = post_dict['post_chapo']
  36. django_post.summary = post_dict['post_chapo_wiki']
  37. django_post.content_html = post_dict['post_content']
  38. django_post.content = post_dict['post_content_wiki']
  39. django_post.creation_date = post_dict['post_creadt']
  40. django_post.modification_date = post_dict['post_upddt']
  41. django_post.publication_date = post_dict['post_dt']
  42. django_post.is_online = post_dict['post_pub']
  43. django_post.is_bestof = post_dict['post_selected']
  44. django_post.markup = 'wiki2xhtml'
  45. django_post.save()</pre>
  46. <p>Et voila, je peux maintenant vérifier le nombre de billets présents avec le shell de Django&nbsp;:</p>
  47. <pre>$ python manage.py shell
  48. &gt;&gt;&gt; from biologeek.journal.models import Post
  49. &gt;&gt;&gt; Post.objects.count()
  50. 190</pre>
  51. <p><strong>Parfait !</strong></p>
  52. <p>Le script complet est disponible <a href="http://code.google.com/p/biologeek/">sur le dépôt</a>, il a fallu que je modifie un peu le plugin de Dotclear pour qu'il prenne en compte mes tables de blogmarks. Il ne me reste plus qu'à migrer les commentaires.</p>
  53. <h2>Redirections</h2>
  54. <p>Un autre point essentiel lors d'une refonte est de <strong>rediriger chaque ancienne <abbr title="Uniform Resource Locator">URL</abbr> vers la nouvelle et ce de manière permanente</strong>. Il existe une <a href="http://www.djangoproject.com/documentation/redirects/">application de redirection dans les contributions de Django</a> mais ça ne convenait pas à mes modifications au niveau des URL des billets qui nécessitent de récupérer leurs tags pour faire la redirection.</p>
  55. <p>Heureusement, oh oui <strong>heureusement, que j'ai conservé le index.php dans toutes mes URL</strong> car sinon la complexité des expressions régulières à gérer m'aurait donné un mal de crâne interminable. Du coup, ça devient vraiment plus simple car je peux rediriger toutes les URL commençant par /journal/index.php vers mon script de redirection. Le script prend en compte toutes les URL que j'ai pu répertorié (j'espère ne pas en avoir oublié, au pire je les rajouterais lors de la migration)&nbsp;:</p>
  56. <pre>urlpatterns = patterns('',
  57. ('archives', redirect('/archives/')),
  58. ('rss.php$', redirect_rss('/rss/journal/')),
  59. ('atom.php$', redirect_rss('/rss/journal/')),
  60. ('rss_ailleurs.xml$', redirect('/rss/bistrot/')),
  61. ('(?P&lt;tag&gt;[A-Z][-\w]+)/$', redirect_tag('/%(tag)s/')),
  62. ('(?P&lt;slug&gt;[-\w]+)/$', redirect_post()),
  63. ('$', redirect('/journal/')),
  64. )</pre>
  65. <p><strong>[edit du lendemain]</strong>&nbsp;: finalement c'est pas si simple car certaines URL ne sont pas derrière index.php, du coup ça donne ça&nbsp;:</p>
  66. <pre>urlpatterns = patterns('',
  67. ('index.php/archives', redirect('/archives/')),
  68. ('index.php/(?P&lt;tag&gt;[A-Z][-\w]+)/$', redirect_tag('/%(tag)s/')),
  69. ('index.php/(?P&lt;slug&gt;[-\w]+)/$', redirect_post()),
  70. ('index.php/$', redirect('/journal/')),
  71. ('rss.php$', redirect_rss('/feeds/rss/')),
  72. ('atom.php$', redirect_rss('/feeds/rss/')),
  73. ('rss_ailleurs.xml$', redirect('/feeds/rss/bistrot/')),
  74. )</pre>
  75. <p>La fonction appelée fait ensuite une redirection permanente (301) vers la nouvelle URL, par exemple dans le cas des billets&nbsp;:</p>
  76. <pre>def redirect_post():
  77. def inner(request, slug):
  78. post = Post.published.get(slug=slug)
  79. return HttpResponsePermanentRedirect(post.get_absolute_url())
  80. return inner</pre>
  81. <p>Au final il ne m'aura fallu que <strong>quelques lignes de python</strong> pour franchir cette nouvelle étape, la prochaine fois on ajoute les commentaires et les fils RSS. On avance, on avance...</p>