subfolder_list.py 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. #!/usr/bin/env python3
  2. # SPDX-License-Identifier: Apache-2.0
  3. """Write subfolder list to a file
  4. This script will walk the specified directory and write the file specified with
  5. the list of all sub-directories found. If the output file already exists, the
  6. file will only be updated in case sub-directories have been added or removed
  7. since the previous invocation.
  8. """
  9. import os
  10. import argparse
  11. def parse_args():
  12. """Parse command line arguments and options"""
  13. parser = argparse.ArgumentParser(
  14. description=__doc__,
  15. formatter_class=argparse.RawDescriptionHelpFormatter)
  16. parser.add_argument('-d', '--directory', required=True,
  17. help='Directory to walk for sub-directory discovery')
  18. parser.add_argument('-c', '--create-links', required=False,
  19. help='Create links for each directory found in \
  20. directory given')
  21. parser.add_argument('-o', '--out-file', required=True,
  22. help='File to write containing a list of all \
  23. directories found')
  24. parser.add_argument('-t', '--trigger-file', required=False,
  25. help='Trigger file to be be touched to re-run CMake')
  26. args = parser.parse_args()
  27. return args
  28. def get_subfolder_list(directory, create_links=None):
  29. """Return subfolder list of a directory"""
  30. dirlist = []
  31. if create_links is not None:
  32. if not os.path.exists(create_links):
  33. os.makedirs(create_links)
  34. symbase = os.path.basename(directory)
  35. symlink = create_links + os.path.sep + symbase
  36. if not os.path.exists(symlink):
  37. os.symlink(directory, symlink)
  38. dirlist.append(symlink)
  39. else:
  40. dirlist.append(directory)
  41. for root, dirs, _ in os.walk(directory, topdown=True):
  42. dirs.sort()
  43. for subdir in dirs:
  44. if create_links is not None:
  45. targetdirectory = os.path.join(root, subdir)
  46. reldir = os.path.relpath(targetdirectory, directory)
  47. linkname = symbase + '_' + reldir.replace(os.path.sep, '_')
  48. symlink = create_links + os.path.sep + linkname
  49. if not os.path.exists(symlink):
  50. os.symlink(targetdirectory, symlink)
  51. dirlist.append(symlink)
  52. else:
  53. dirlist.append(os.path.join(root, subdir))
  54. return dirlist
  55. def gen_out_file(out_file, dirs):
  56. """Generate file with the list of directories
  57. File won't be updated if it already exists and has the same content
  58. """
  59. dirs_nl = "\n".join(dirs) + "\n"
  60. if os.path.exists(out_file):
  61. with open(out_file, 'r', encoding="utf-8") as out_file_fo:
  62. out_file_dirs = out_file_fo.read()
  63. if out_file_dirs == dirs_nl:
  64. return
  65. with open(out_file, 'w', encoding="utf-8") as out_file_fo:
  66. out_file_fo.writelines(dirs_nl)
  67. def touch(trigger):
  68. """Touch the trigger file
  69. If no trigger file is provided then do a return.
  70. """
  71. if trigger is None:
  72. return
  73. if os.path.exists(trigger):
  74. os.utime(trigger, None)
  75. else:
  76. with open(trigger, 'w') as trigger_fo:
  77. trigger_fo.write("")
  78. def main():
  79. """Parse command line arguments and take respective actions"""
  80. args = parse_args()
  81. dirs = get_subfolder_list(args.directory, args.create_links)
  82. gen_out_file(args.out_file, dirs)
  83. # Always touch trigger file to ensure json files are updated
  84. touch(args.trigger_file)
  85. if __name__ == "__main__":
  86. main()