StatisticsByCategories.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. """
  2. ***************************************************************************
  3. StatisticsByCategories.py
  4. ---------------------
  5. Date : September 2012
  6. Copyright : (C) 2012 by Victor Olaya
  7. Email : volayaf at gmail dot com
  8. ***************************************************************************
  9. * *
  10. * This program is free software; you can redistribute it and/or modify *
  11. * it under the terms of the GNU General Public License as published by *
  12. * the Free Software Foundation; either version 2 of the License, or *
  13. * (at your option) any later version. *
  14. * *
  15. ***************************************************************************
  16. """
  17. __author__ = 'Victor Olaya'
  18. __date__ = 'September 2012'
  19. __copyright__ = '(C) 2012, Victor Olaya'
  20. from qgis.core import (QgsProcessingParameterFeatureSource,
  21. QgsStatisticalSummary,
  22. QgsDateTimeStatisticalSummary,
  23. QgsStringStatisticalSummary,
  24. QgsFeatureRequest,
  25. QgsApplication,
  26. QgsProcessingException,
  27. QgsProcessingParameterField,
  28. QgsProcessingParameterFeatureSink,
  29. QgsFields,
  30. QgsField,
  31. QgsWkbTypes,
  32. QgsCoordinateReferenceSystem,
  33. QgsFeature,
  34. QgsFeatureSink,
  35. QgsProcessing,
  36. QgsProcessingFeatureSource,
  37. NULL)
  38. from qgis.PyQt.QtCore import QVariant
  39. from processing.algs.qgis.QgisAlgorithm import QgisAlgorithm
  40. from collections import defaultdict
  41. class StatisticsByCategories(QgisAlgorithm):
  42. INPUT = 'INPUT'
  43. VALUES_FIELD_NAME = 'VALUES_FIELD_NAME'
  44. CATEGORIES_FIELD_NAME = 'CATEGORIES_FIELD_NAME'
  45. OUTPUT = 'OUTPUT'
  46. def group(self):
  47. return self.tr('Vector analysis')
  48. def groupId(self):
  49. return 'vectoranalysis'
  50. def tags(self):
  51. return self.tr('groups,stats,statistics,table,layer,sum,maximum,minimum,mean,average,standard,deviation,'
  52. 'count,distinct,unique,variance,median,quartile,range,majority,minority,histogram,distinct,summary').split(',')
  53. def icon(self):
  54. return QgsApplication.getThemeIcon("/algorithms/mAlgorithmBasicStatistics.svg")
  55. def svgIconPath(self):
  56. return QgsApplication.iconPath("/algorithms/mAlgorithmBasicStatistics.svg")
  57. def __init__(self):
  58. super().__init__()
  59. def initAlgorithm(self, config=None):
  60. self.addParameter(QgsProcessingParameterFeatureSource(self.INPUT,
  61. self.tr('Input vector layer'),
  62. types=[QgsProcessing.TypeVector]))
  63. self.addParameter(QgsProcessingParameterField(self.VALUES_FIELD_NAME,
  64. self.tr(
  65. 'Field to calculate statistics on (if empty, only count is calculated)'),
  66. parentLayerParameterName=self.INPUT, optional=True))
  67. self.addParameter(QgsProcessingParameterField(self.CATEGORIES_FIELD_NAME,
  68. self.tr('Field(s) with categories'),
  69. parentLayerParameterName=self.INPUT,
  70. type=QgsProcessingParameterField.Any, allowMultiple=True))
  71. self.addParameter(QgsProcessingParameterFeatureSink(self.OUTPUT, self.tr('Statistics by category')))
  72. def name(self):
  73. return 'statisticsbycategories'
  74. def displayName(self):
  75. return self.tr('Statistics by categories')
  76. def processAlgorithm(self, parameters, context, feedback):
  77. source = self.parameterAsSource(parameters, self.INPUT, context)
  78. if source is None:
  79. raise QgsProcessingException(self.invalidSourceError(parameters, self.INPUT))
  80. value_field_name = self.parameterAsString(parameters, self.VALUES_FIELD_NAME, context)
  81. category_field_names = self.parameterAsFields(parameters, self.CATEGORIES_FIELD_NAME, context)
  82. value_field_index = source.fields().lookupField(value_field_name)
  83. if value_field_index >= 0:
  84. value_field = source.fields().at(value_field_index)
  85. else:
  86. value_field = None
  87. category_field_indexes = list()
  88. # generate output fields
  89. fields = QgsFields()
  90. for field_name in category_field_names:
  91. c = source.fields().lookupField(field_name)
  92. if c == -1:
  93. raise QgsProcessingException(self.tr('Field "{field_name}" does not exist.').format(field_name=field_name))
  94. category_field_indexes.append(c)
  95. fields.append(source.fields().at(c))
  96. def addField(name):
  97. """
  98. Adds a field to the output, keeping the same data type as the value_field
  99. """
  100. field = QgsField(value_field)
  101. field.setName(name)
  102. fields.append(field)
  103. if value_field is None:
  104. field_type = 'none'
  105. fields.append(QgsField('count', QVariant.Int))
  106. elif value_field.isNumeric():
  107. field_type = 'numeric'
  108. fields.append(QgsField('count', QVariant.Int))
  109. fields.append(QgsField('unique', QVariant.Int))
  110. fields.append(QgsField('min', QVariant.Double))
  111. fields.append(QgsField('max', QVariant.Double))
  112. fields.append(QgsField('range', QVariant.Double))
  113. fields.append(QgsField('sum', QVariant.Double))
  114. fields.append(QgsField('mean', QVariant.Double))
  115. fields.append(QgsField('median', QVariant.Double))
  116. fields.append(QgsField('stddev', QVariant.Double))
  117. fields.append(QgsField('minority', QVariant.Double))
  118. fields.append(QgsField('majority', QVariant.Double))
  119. fields.append(QgsField('q1', QVariant.Double))
  120. fields.append(QgsField('q3', QVariant.Double))
  121. fields.append(QgsField('iqr', QVariant.Double))
  122. elif value_field.type() in (QVariant.Date, QVariant.Time, QVariant.DateTime):
  123. field_type = 'datetime'
  124. fields.append(QgsField('count', QVariant.Int))
  125. fields.append(QgsField('unique', QVariant.Int))
  126. fields.append(QgsField('empty', QVariant.Int))
  127. fields.append(QgsField('filled', QVariant.Int))
  128. # keep same data type for these fields
  129. addField('min')
  130. addField('max')
  131. else:
  132. field_type = 'string'
  133. fields.append(QgsField('count', QVariant.Int))
  134. fields.append(QgsField('unique', QVariant.Int))
  135. fields.append(QgsField('empty', QVariant.Int))
  136. fields.append(QgsField('filled', QVariant.Int))
  137. # keep same data type for these fields
  138. addField('min')
  139. addField('max')
  140. fields.append(QgsField('min_length', QVariant.Int))
  141. fields.append(QgsField('max_length', QVariant.Int))
  142. fields.append(QgsField('mean_length', QVariant.Double))
  143. request = QgsFeatureRequest().setFlags(QgsFeatureRequest.NoGeometry)
  144. if value_field is not None:
  145. attrs = [value_field_index]
  146. else:
  147. attrs = []
  148. attrs.extend(category_field_indexes)
  149. request.setSubsetOfAttributes(attrs)
  150. features = source.getFeatures(request, QgsProcessingFeatureSource.FlagSkipGeometryValidityChecks)
  151. total = 50.0 / source.featureCount() if source.featureCount() else 0
  152. if field_type == 'none':
  153. values = defaultdict(lambda: 0)
  154. else:
  155. values = defaultdict(list)
  156. for current, feat in enumerate(features):
  157. if feedback.isCanceled():
  158. break
  159. feedback.setProgress(int(current * total))
  160. attrs = feat.attributes()
  161. cat = tuple([attrs[c] for c in category_field_indexes])
  162. if field_type == 'none':
  163. values[cat] += 1
  164. continue
  165. if field_type == 'numeric':
  166. if attrs[value_field_index] == NULL:
  167. continue
  168. else:
  169. value = float(attrs[value_field_index])
  170. elif field_type == 'string':
  171. if attrs[value_field_index] == NULL:
  172. value = ''
  173. else:
  174. value = str(attrs[value_field_index])
  175. elif attrs[value_field_index] == NULL:
  176. value = NULL
  177. else:
  178. value = attrs[value_field_index]
  179. values[cat].append(value)
  180. (sink, dest_id) = self.parameterAsSink(parameters, self.OUTPUT, context,
  181. fields, QgsWkbTypes.NoGeometry, QgsCoordinateReferenceSystem())
  182. if sink is None:
  183. raise QgsProcessingException(self.invalidSinkError(parameters, self.OUTPUT))
  184. if field_type == 'none':
  185. self.saveCounts(values, sink, feedback)
  186. elif field_type == 'numeric':
  187. self.calcNumericStats(values, sink, feedback)
  188. elif field_type == 'datetime':
  189. self.calcDateTimeStats(values, sink, feedback)
  190. else:
  191. self.calcStringStats(values, sink, feedback)
  192. return {self.OUTPUT: dest_id}
  193. def saveCounts(self, values, sink, feedback):
  194. total = 50.0 / len(values) if values else 0
  195. current = 0
  196. for cat, v in values.items():
  197. if feedback.isCanceled():
  198. break
  199. feedback.setProgress(int(current * total) + 50)
  200. f = QgsFeature()
  201. f.setAttributes(list(cat) + [v])
  202. sink.addFeature(f, QgsFeatureSink.FastInsert)
  203. current += 1
  204. def calcNumericStats(self, values, sink, feedback):
  205. stat = QgsStatisticalSummary()
  206. total = 50.0 / len(values) if values else 0
  207. current = 0
  208. for cat, v in values.items():
  209. if feedback.isCanceled():
  210. break
  211. feedback.setProgress(int(current * total) + 50)
  212. stat.calculate(v)
  213. f = QgsFeature()
  214. f.setAttributes(list(cat) + [stat.count(),
  215. stat.variety(),
  216. stat.min(),
  217. stat.max(),
  218. stat.range(),
  219. stat.sum(),
  220. stat.mean(),
  221. stat.median(),
  222. stat.stDev(),
  223. stat.minority(),
  224. stat.majority(),
  225. stat.firstQuartile(),
  226. stat.thirdQuartile(),
  227. stat.interQuartileRange()])
  228. sink.addFeature(f, QgsFeatureSink.FastInsert)
  229. current += 1
  230. def calcDateTimeStats(self, values, sink, feedback):
  231. stat = QgsDateTimeStatisticalSummary()
  232. total = 50.0 / len(values) if values else 0
  233. current = 0
  234. for cat, v in values.items():
  235. if feedback.isCanceled():
  236. break
  237. feedback.setProgress(int(current * total) + 50)
  238. stat.calculate(v)
  239. f = QgsFeature()
  240. f.setAttributes(list(cat) + [stat.count(),
  241. stat.countDistinct(),
  242. stat.countMissing(),
  243. stat.count() - stat.countMissing(),
  244. stat.statistic(QgsDateTimeStatisticalSummary.Min),
  245. stat.statistic(QgsDateTimeStatisticalSummary.Max)
  246. ])
  247. sink.addFeature(f, QgsFeatureSink.FastInsert)
  248. current += 1
  249. def calcStringStats(self, values, sink, feedback):
  250. stat = QgsStringStatisticalSummary()
  251. total = 50.0 / len(values) if values else 0
  252. current = 0
  253. for cat, v in values.items():
  254. if feedback.isCanceled():
  255. break
  256. feedback.setProgress(int(current * total) + 50)
  257. stat.calculate(v)
  258. f = QgsFeature()
  259. f.setAttributes(list(cat) + [stat.count(),
  260. stat.countDistinct(),
  261. stat.countMissing(),
  262. stat.count() - stat.countMissing(),
  263. stat.min(),
  264. stat.max(),
  265. stat.minLength(),
  266. stat.maxLength(),
  267. stat.meanLength()
  268. ])
  269. sink.addFeature(f, QgsFeatureSink.FastInsert)
  270. current += 1