Atlassian Exporter es un proyecto desarrollado con Python y PyCharm, con el objetivo de poder descargar datos de Atlassian Cloud, como por ejemplo los datos de usuarios, grupos, licencias, y proyectos.
Estos datos pueden posteriormente importarse en una base de datos para su análisis, pudiendo utilizarse para diferentes propósitos, como el inventariado o el reparto de costes.
Este repo se ha creado para complementar el Post Python - Exportando datos de Jira, Confluence y Bitbucket con atlassian-exporter del Blog El Willie - The Geeks invaders
Para la ejecución sobre MiniKube te puede interesar leer los Posts Introducción a MiniKube e instalación en Windows 11 y Administración fácil y rápida con K9s.
Puedes apoyar mi trabajo haciendo "☆ Star" en el repo o nominarme a "GitHub Star". Muchas gracias :-)
Se trata de un programa de línea de comandos en Python, que accede a las APIs de Atlassian Cloud y Bitbucket Cloud, para descargar los datos que necesita, generando ficheros CSV en la carpeta ./export/ para que puedan ser utilizados para un posterior análisis y tratamiento.
De Jira Software, Jira Service Desk y Confluence, se desea obtener la siguiente información:
- Usuarios, grupos, y pertenencia de usuarios a grupos (de aquí se puede obtener la asignación de Licencias)
- Proyectos de Jira
- Espacios de Confluence
De Bitbucket se desea obtener la siguiente información:
- Usuarios, grupos, y pertenencia de usuarios a grupos
- Proyectos, repos, y pertenencia de repos a proyectos
- Commits de los repos
- Ramas y Tags de los repos
- git clone de todos los repos
Dentro de Atlassian Cloud developer documentation, tenemos varios enlaces de interés, incluyendo a la documentación de varias APIs de Atlassian Cloud como las siguientes:
- Jira Cloud Platform - REST API y Confluence Cloud - REST API: Para acceder podemos utilizar un API token que generaremos desde nuestro perfil de Atlassian (Account settings), es decir, utilizaremos Autenticación Básica.
- Bitbucket Cloud - REST API: Para acceder podemos utilizar una App Password, que generaremos desde nuestro perfil de Bitbucket (Personal settigns), es decir, utilizaremos Autenticación Básica.
La antigua API v1 de Bitbucket, aunque está deprecada, también nos resulta de utilidad, por ejemplo para obtener los Grupos de Bitbucket de forma sencilla.
La mayoría de las API están paginadas, pero no todas, y de las que están paginadas, no todas paginan de la misma forma (es decir, utilizan diferentes parámetros para poder paginar, como start ó startAt, aunque en el fondo son muy parecidas).
Atlassian protege sus APIs mediante Rate Limiting, por lo que tenemos que controlar el status code de la respuesta HTTP, de tal modo que si recibimos un HTTP-429 (Too many requests) deberemos gestionar esperas y reintentos.
Es un programa de línea de comandos, que espera recibir dos parámetros:
- Acción que se desea realizar. Básicamente es indicar qué datos deseamos exportar, que se generarán en la carpeta ./export/. En función de qué datos necesitemos, y de qué licencias de Atlassian tengamos, seleccionaremos las acciones que necesitemos. Si queremos exportar varios datos (ej: usuarios de Atlassian y usuarios de Bitbucket) bastará con ejecutarlo dos veces, en cada una especificando una acción.
- Fichero de configuración. Proporciona los datos de conexión en un fichero json con una formato determinado ubicado en la carpeta ./config/, que dependerá de la acción (datos de conexión a Atlassian Cloud o a Bitbucket Cloud).
El fichero JSON de configuración para la conexión con Atlassian Cloud (ej: ./config/atlassian_conn_elwillie.json) será similar al siguiente, en el que especificaremos el nombre de nuestro Site de Atlassian, usuario (email) y un Token. Es recomendable utilizar las credenciales de un usuario con permisos Site Admin y Confluence Admin, para garantizar que tiene permisos tanto para los datos de usuarios y grupos, como para todos los Proyectos de Jira y Espacios de Confluence.
{
"atlassian-site": "willie",
"atlassian-user": "alias@gmail.com",
"atlassian-token": "wSCqpbo7OgDXDBMRSjZQAE59"
}
El fichero JSON de configuración para la conexión con Bitbucket Cloud (ej: ./config/bitbucket_conn_elwillie.json) será similar al siguiente, en el que especificaremos el nombre de nuestro Workspace, usuario, y Token (App Password). Es recomendable utilizar las credenciales de un usuario con permisos Admin, para garantizar que tiene permisos tanto para los datos de los usuarios y grupos, como de todos los Proyectos y Repos.
{
"bitbucket-workspace": "willie",
"bitbucket-user": "willie",
"bitbucket-token": "ATBBsxRs4jqsAdbsRkjNb7JDZrVv436FE68A"
}
Las carpetas ./config/ y ./export/ están añadidas al fichero .gitignore, para evitar que se puedan subir tanto credenciales como datos a los repos remotos de Git, tanto por privacidad, como por intentar mantener el repo limpi (evitar subir exportaciones de diferentes pruebas, que no aportan valor en el repo remoto). Sin embargo, si se incluiran al construir nuestra imagen Docker en local, lo que nos permitirá utilizar nuestras configuraciones tanto para ejecutar en local con Python como en local con Docker, Docker Compose, o Kubernetes.
Si tenemos varios Sites de Atlassian y/o varios Workspaces de Bitbucket, podemos crear múltiples ficheros de configuración (los que necesitemos) y ejecutar varias veces el programa.
A continuación se muestra un ejemplo de uso.
python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_users
python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_groups_and_members
python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_projects
python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_confluence_spaces
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_users
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_groups_and_members
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_projects
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_repos
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_repos_commits
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_repos_branches
python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=git_clone_all_bitbucket_repos
Si te interesa aprender Python, tienes disponibles los siguientes cursos gratuitos de Python en Edube - OpenEDG:
- Python Essentials 1
- Python Essentials 2
- Python Advanced 1 – OOP
- Python Advanced 2 – Best Practices and Standardization
- Python Advanced 3 – GUI Programming
- Python Advanced 4 – RESTful APIs
- Python Advanced 5 – File Processing
Otro recurso muy interesante es Real Python, donde podrás encontrar tutoriales, baterías de preguntas para ponerte a prueba (quizzes), etc.
En mi Blog personal (El Willie - The Geeks invaders) y en mi perfil de GitHub, encontrarás más información sobre mi, y sobre los contenidos de tecnología que comparto con la comunidad.
Este repo se puede clonar desde GitHub utilizando este enlace HTTP.
A continuación se muestra el comando git clone usando SSH en lugar de HTTP.
git clone git@github.com:ElWillieES/atlassian-exporter.git
- Ramas permanentes. Utilizaremos master como rama principal.
- Ramas temporales o efímeras. Utilizaremos feature/xxx (nueva característica) y/o hotfix/xxx (corregir un bug crítico urgente), que nacen de la rama principal y mezclan de nuevo sobre ella mediante Merge Request.
- Gestión de Releases. Para cada Release generaremos una etiqueta (tag) del tipo release/a.b.c.
En la descripción de las ramas de feature y hotfix, se especificará el ID de la tarea o issue asociada, por ejemplo:
feature/3813
hotfix/2262
Si necesitáramos varias ramas para una misma tarea, añadiremos un número secuencial para evitar la colisión:
feature/3813-1
feature/3813-2
Como recomendación y buena práctica, el título para los Commits y de las Merge Request, pueden empezar con un icono y un prefijo, seguido de dos puntos y de un mensaje corto que comience por un verbo imperativo (ej: add, change, fix, remove, etc.). Por ejemplo:
✨ feat(backend): add support for users having multiple suscriptions
Prefijos:
feat: Nueva característica
fix: Corrección a un error
doc: Documentación
style: Cambios de formato (guía de estilo)
refactor: Renombrar una variable, simplificar un método, etc…
test: Añadir o modificar tests
chore: Tareas rutinarias, como modificar el .gitignore, etc…
Iconos:
💄 Cosmetic
🎨 Improve format / structure
🛠/🐛 Fix
✨ Feature
🚑 Hotfix
📝 Doc
🚀 Release
♻ Refactor
🐳 Devops
☸ Kubernetes
🧪 Arquitectura de tests
✅ Añadir un Test
✔ Hacer que un test pase
💩 Ñapas
🏗 Architectural changes
🤡 Mocks
💚 Fixing Build
📈 Analiltycs
🌐 Localizations
😒 Chore
💫 Animations & Transitions
♿ Accesibility
🚧 Feature work in progress
🚀 Launch a new build
Se puede ejecutar la aplicación en local con Docker.
Los siguientes comandos ejecutados en la raíz del Proyecto, muestran:
- Cómo crear una imagen en local con docker build. Antes de construir la imagen, borramos el contenido de app/export/, por si tuviéramos ficheros de pruebas de ejecución, para no engordar y ensuciar la imagen.
- Cómo listar las imágenes que tenemos disponibles en local. Deberá aparecer la que acabamos de crear.
- Cómo ejecutar un contenedor con nuestra imagen, con el comando deseado. Como nuestra aplicación es de línea de comandos, se incluyen varios ejemplos, en cada uno de los cuales se indica como parámetro el fichero de configuración que necesita (debe existir en la carpeta /usr/src/app/config/ del contenedor) y la acción a realizar (hay varias posible, según los datos que queramos exportar, que dependerá de los productos que tengamos contratados). Como se van a generar los ficheros de exportación en la carpeta del contenedor /usr/src/app/export, y los contenedores son efímeros, utilizamos un volumen sobre /usr/src/app/export para que los datos persistan y además podamos acceder a los ficheros que hemos generado. Es necesario especificar la ruta absoluta del host (ajustarla con la de cada uno).
rm app/export/*
rm git-clone/*
docker build -t atlassian-exporter .
docker images
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_users
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_groups_and_members
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_projects
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_confluence_spaces
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_users
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_groups_and_members
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_projects
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_repos
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_repos_commits
docker run -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=export_all_bitbucket_repos_branches
docker run -v d:/code/elwillie/atlassian-exporter/app/git-clone:/usr/src/app/git-clone --rm atlassian-exporter python atlassian-exporter.py --configfile=bitbucket_conn_elwillie.json --action=git_clone_all_bitbucket_repos
Podemos arrancar una sesión interativa de Bash sobre un Contendor con nuestra imagen Docker, para de este modo, analizar mejor incidencias y problemas que nos puedan surgir, depurar, etc. Suele ser bastante útil.
En el siguiente ejemplo, arrancamos una sesión bash sobre un contenedor con nuestra imagen y un volumen mapeando la carpeta export de nuestro portátil con la del contenedor, ejecutamos atlassian-exporter para exportar los usuarios de jira, y salimos del contenedor.
docker run --rm -v d:/code/elwillie/atlassian-exporter/app/export:/usr/src/app/export -it atlassian-exporter /bin/bash
python atlassian-exporter.py -c atlassian_conn_elwillie.json -a export_all_jira_users
exit
El siguiente comando ejecutado en la raíz del Proyecto, muestra cómo compilar (es decir, construir la imagen Docker) y ejecutar atlassian-exporter con Docker Compose, así como la forma de poder comprobar los logs de su ejecución.
Si observamos el fichero docker-compose.yml, podemos ver que incluye varias ejecuciones de atlassian-exporter, para las diferentes exportaciones que queremos realizar. Además, utiliza un volumen mapeando la carpeta export de nuestro portátil con la del contenedor, para así poder acceder a los datos generados desde nuestro portátil, después de su ejecución.
docker-compose -f docker-compose.yml up --build -d
docker-compose -f docker-compose.yml logs
Se puede ejecutar la aplicación en local con Kubernetes, si tienes instalado MiniKube. Para ampliar información te puede interesar leer Introducción a MiniKube e instalación en Windows 11
Los manifiestos de Kubernetes, están en la carpeta kube, y son los siguientes:
- exporter-ns.yml. Para la creación del namespace exporter, donde desplegaremos nuestra aplicación.
- atlassian-exporter-conf.vol. Permite crear un Volumen, es decir, un PersistentVolume de tipo hostPath y un PersistentVolumeClaim, que mapearemos a nuestro Job para tener persistencia entre ejecuciones. Los datos persistirán dentro del almacenamiento de MiniKube (nos podemos conectar con minikube ssh para verlos).
- atlassian-exporter-conf.yml. Permite crear un ConfigMap, que contiene los ficheros JSON de configuración que necesitamos para conectarnos a nuestras instancias de Atlassian Cloud y Bitbucket Cloud. Los mapearemos a los contenedores sobre el directorio /usr/src/app/config sobrescribiendo los ficheros que pudiera haber en la imagen original.
- atlassian-exporter-job.yml. Consiste en un Job que incluye un contenedor para cada comando que queremos ejecutar (en nuestro caso de ejemplo son ocho, para exportar todos los datos de Jira, Confluence, y Bitbucket), y mapea tanto el ConfigMap anterior como el volumen persistente.
Los siguientes comandos ejecutados en la raíz del Proyecto, muestran cómo tagear la Imagen Docker para subirla al Registry local de MiniKube.
docker tag atlassian-exporter localhost:5000/atlassian-exporter
docker push localhost:5000/atlassian-exporter
Realizado esto, en la ventana Terminal de PyCharm, podemos ejecutar los siguientes comandos para aplicar los manifiestos en nuestro Cluster de MiniKube (namespace y Job), y consultar el Log de ejecución del Job que acabamos de crear y ejecutar (la salida del Log, será igual a cuando lo ejecutamos en Docker o directamente en PyCharm).
cd kube
kubectl apply -f exporter-ns.yml
kubectl apply -f atlassian-exporter-vol.yml
kubectl get PersistentVolume -n exporter -o wide
kubectl get PersistentVolumeClaim -n exporter -o wide
kubectl apply -f atlassian-exporter-conf.yml
kubectl apply -f atlassian-exporter-job.yml
kubectl get jobs -n exporter
kubectl describe jobs atlassian-exporter -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-jira-users -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-jira-groups-and-members -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-jira-projects -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-confluence-spaces -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-bitbucket-users -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-bitbucket-groups-and-members -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-bitbucket-projects -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-bitbucket-repos -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-bitbucket-repos-branches -n exporter
kubectl logs job/atlassian-exporter -c atlassian-exporter-bitbucket-repos-commits -n exporter
Si queremos ver o incluso editar el ConfigMap, podemos utilizar el siguiente comando.
kubectl edit configmap atlassian-exporter-conf -n exporter
También podemos crear un nuevo contenedor al vuelo, con nuestra imagen, con una sesión bash a la que conectarnos para poder depurar y hacer pruebas.
kubectl run -it --rm atlassian-exporter --image=localhost:5000/atlassian-exporter -n exporter -- /bin/bash
Si necesitamos volver a crear el Job, tendremos que eliminarlo antes, para lo cual podemos utilizar un comando como el siguiente.
kubectl delete job atlassian-exporter -n exporter
Al finalizar podemos eliminar el namespace de Kubernetes, para eliminar todos los recursos y dejar "la casa limpia".
kubectl delete ns exporter
Nombre | |
---|---|
El Willie | elwillieES@gmail.com |