]>
git.ipfire.org Git - ipfire.org.git/blob - www/webapp/backend/stasy.py
3 from __future__
import division
11 from misc
import Singleton
13 DATABASE_HOST
= ["wilhelmina.ipfire.org"]
14 DATABASE_NAME
= "stasy"
16 CPU_SPEED_CONSTRAINTS
= (0, 500, 1000, 1500, 2000, 2500, 3000, 3500)
17 MEMORY_CONSTRAINTS
= (0, 128, 256, 512, 1024, 2048, 4096, 8192, 16384)
20 "AMDisbetter!" : "AMD",
21 "AuthenticAMD" : "AMD",
22 "CentaurHauls" : "VIA",
23 "CyrixInstead" : "Cyrix",
24 "GenuineIntel" : "Intel",
25 "TransmetaCPU" : "Transmeta",
26 "GenuineTMx86" : "Transmeta",
27 "Geode by NSC" : "NSC",
28 "NexGenDriven" : "NexGen",
29 "RiseRiseRise" : "Rise",
30 "SiS SiS SiS " : "SiS",
31 "UMC UMC UMC " : "UMC",
32 "VIA VIA VIA " : "VIA",
37 (r
"(AMD Athlon).*(XP).*", r
"\1 \2"),
38 (r
"(AMD Phenom).* ([0-9]+) .*", r
"\1 \2"),
39 (r
"(AMD Phenom).*", r
"\1"),
40 (r
"(AMD Sempron).*", r
"\1"),
41 (r
"AMD Athlon.* II X2 ([a-z0-9]+).*", r
"AMD Athlon X2 \1"),
42 (r
"(Geode).*", r
"\1"),
45 (r
"Intel.*(Atom|Celeron).*CPU\s*([A-Z0-9]+) .*", r
"Intel \1 \2"),
46 (r
"(Intel).*(Celeron).*", r
"\1 \2"),
47 (r
"Intel.* Core.*2 Duo *CPU .* ([A-Z0-9]+) .*", r
"Intel C2D \1"),
48 (r
"Intel.* Core.*2 CPU .* ([A-Z0-9]+) .*", r
"Intel C2 \1"),
49 (r
"Intel.* Core.*2 Quad *CPU .* ([A-Z0-9]+) .*", r
"Intel C2Q \1"),
50 (r
"Intel.* Xeon.* CPU .* ([A-Z0-9]+) .*", r
"Intel Xeon \1"),
51 (r
"(Intel).*(Xeon).*", r
"\1 \2"),
52 (r
"Intel.* Pentium.* (D|4) .*", r
"Intel Pentium \1"),
53 (r
"Intel.* Pentium.* Dual .* ([A-Z0-9]+) .*", r
"Intel Pentium Dual \1"),
54 (r
"Pentium.* Dual-Core .* ([A-Z0-9]+) .*", r
"Intel Pentium Dual \1"),
55 (r
"(Pentium I{2,3}).*", r
"Intel \1"),
56 (r
"(Celeron \(Coppermine\))", r
"Intel Celeron"),
59 (r
"(VIA \w*).*", r
"\1"),
62 (r
"QEMU Virtual CPU version .*", r
"QEMU CPU"),
65 (r
"Feroceon .*", r
"ARM Feroceon"),
68 CPU_CORES
= range(1, 9)
70 class ProfileDict(object):
71 def __init__(self
, data
):
75 class ProfileCPU(ProfileDict
):
78 return self
._data
.get("arch")
82 vendor
= self
._data
.get("vendor")
84 return CPU_VENDORS
.get(vendor
, vendor
)
88 speed
= self
._data
.get("speed")
90 # ARM boxes do not report speed but
91 # bogomips equals speed
98 def friendly_speed(self
):
100 return "%dMHz" % self
.speed
102 return "%.1fGHz" % round(self
.speed
/ 1000, 1)
106 return self
._data
.get("bogomips")
110 return self
._data
.get("model")
113 def model_string(self
):
114 return self
._data
.get("model_string")
118 return self
._data
.get("flags")
122 return self
._data
.get("count")
125 def capable_64bit(self
):
126 return "lm" in self
.flags
129 def capable_pae(self
):
130 return "pae" in self
.flags
133 def capable_virt(self
):
134 return "vmx" in self
.flags
or "svm" in self
.flags
137 def friendly_vendor(self
):
138 s
= self
.model_string
139 for pattern
, repl
in CPU_STRINGS
:
140 if re
.match(pattern
, s
) is None:
142 return re
.sub(pattern
, repl
, s
)
147 def friendly_string(self
):
148 s
= self
.friendly_vendor
151 s
+= " @ %s" % self
.friendly_speed
156 class ProfileHypervisor(ProfileDict
):
158 return "<%s %s-%s>" % (self
.__class
__.__name
__, self
.vendor
, self
.type)
162 return self
._data
.get("vendor")
166 return self
._data
.get("type")
169 class ProfileNetwork(ProfileDict
):
170 def has_zone(self
, name
):
171 return self
._data
.get(name
)
175 return self
._data
.get("green")
179 return self
._data
.get("red")
183 return self
._data
.get("blue")
187 return self
._data
.get("orange")
190 class ProfileDevice(ProfileDict
):
198 "00" : "Unclassified",
199 "01" : "Mass storage",
203 "05" : "Memory controller",
205 "07" : "Communication",
206 "08" : "Generic system peripheral",
207 "09" : "Input device",
208 "0a" : "Docking station",
212 "0e" : "Intelligent controller",
213 "0f" : "Satellite communications controller",
215 "11" : "Signal processing controller",
216 "ff" : "Unassigned class",
220 "00" : "Unclassified",
222 "02" : "Communication",
223 "03" : "Input device",
224 "05" : "Generic system peripheral",
227 "08" : "Mass storage",
229 "0a" : "Communication",
233 "0f" : "Personal Healthcare",
234 "dc" : "Diagnostic Device",
236 "ef" : "Unclassified",
237 "fe" : "Unclassified",
238 "ff" : "Unclassified",
242 def __cmp__(self
, other
):
243 return cmp(self
.vendor
, other
.vendor
) or \
244 cmp(self
.model
, other
.model
) or \
245 cmp(self
.driver
, other
.driver
)
249 return self
._data
.get("model")
252 def model_string(self
):
253 cls
= self
.subsystem2class
[self
.subsystem
]
255 return cls().get_device(self
.vendor
, self
.model
)
259 return self
._data
.get("subsystem")
263 return self
._data
.get("vendor")
266 def vendor_string(self
):
267 cls
= self
.subsystem2class
[self
.subsystem
]
269 return cls().get_vendor(self
.vendor
)
273 return self
._data
.get("driver")
277 classid
= self
._data
.get("deviceclass")
279 if self
.subsystem
== "pci":
280 classid
= classid
[:-4]
281 if len(classid
) == 1:
282 classid
= "0%s" % classid
284 elif self
.subsystem
== "usb" and classid
:
285 classid
= classid
.split("/")[0]
286 classid
= "%02x" % int(classid
)
289 return self
.classid2name
[self
.subsystem
][classid
]
294 class Profile(ProfileDict
):
295 def __init__(self
, profile_blob
):
296 ProfileDict
.__init
__(self
, profile_blob
.get("profile"))
298 self
.public_id
= profile_blob
.get("public_id")
299 self
.updated
= profile_blob
.get("updated")
300 self
._geoip
= profile_blob
.get("geoip", None)
303 return "<%s %s>" % (self
.__class
__.__name
__, self
.public_id
)
307 return ProfileCPU(self
._data
.get("cpu"))
312 for d
in self
._data
.get("devices"):
315 if d
.driver
in ("pcieport", "usb", "hub"):
323 def hypervisor(self
):
325 return ProfileHypervisor(self
._data
.get("hypervisor"))
329 return self
.system
.get("virtual")
333 return self
._data
.get("system")
337 return self
.system
.get("release")
341 return self
.system
.get("kernel_release")
345 return self
.system
.get("memory")
348 def friendly_memory(self
):
349 units
= ("k", "M", "G", "T")
357 return "%d%s" % (round(mem
, 0), units
[i
])
361 return self
.system
.get("root_size") or 0
364 def friendly_root_size(self
):
365 units
= ("k", "M", "G", "T")
367 size
= self
.root_size
376 return "%d%s" % (round(size
, 0), units
[i
])
380 return self
.system
.get("vendor")
384 return self
.system
.get("model")
387 def country_code(self
):
389 return self
._geoip
["country_code"].lower()
395 network
= self
._data
.get("network", None)
397 return ProfileNetwork(network
)
401 __metaclass__
= Singleton
404 # Initialize database connection
405 self
._conn
= pymongo
.Connection(DATABASE_HOST
)
406 self
._db
= self
._conn
[DATABASE_NAME
]
408 def get_profile_count(self
):
409 # XXX need to implement something to get profiles updated since
412 # All distinct profiles (based on public_id)
413 # XXX possibly bad performance
414 return len(self
._db
.profiles
.distinct("public_id"))
416 def get_archives_count(self
):
417 return self
._db
.archives
.count()
419 #def _get_profile_cursor(self, public_id):
420 # c = self._db.profiles.find({ "public_id" : public_id })
421 # c.sort("updated", pymongo.ASCENDING)
425 def profile_exists(self
, public_id
):
426 return self
.query({ "public_id" : public_id
}).count() >= 1
428 def get_profile(self
, public_id
):
430 # XXX should only find one object in the end
431 for p
in self
.query({ "public_id" : public_id
}):
436 def get_profiles(self
):
437 # XXX needs nicer database query
439 for p
in self
._db
.profiles
.find():
440 if not p
.get("public_id") in profiles
:
441 profiles
.append(p
.get("public_id"))
445 def query(self
, query
, archives
=False, no_virt
=False, all
=False, fields
=None):
446 db
= self
._db
.profiles
449 db
= self
.db
.archives
452 # XXX cannot use the index?
453 query
.update({ "profile" : { "$exists" : True }})
456 query
.update({ "profile.system.virtual" : False })
458 logging
.debug("Executing query: %s" % query
)
460 return db
.find(query
, fields
=fields
)
463 def secret_ids(self
):
464 return self
._db
.profiles
.distinct("secret_id")
468 return self
.query({}, no_virt
=True).distinct("profile.cpu")
471 def cpu_vendors(self
):
472 return self
.query({}, no_virt
=True).distinct("profile.cpu.vendor")
475 def cpu_vendors_map(self
):
478 for vendor
in self
.cpu_vendors
:
479 friendly_vendor
= CPU_VENDORS
.get(vendor
, vendor
)
480 cpus
[friendly_vendor
] = \
482 "profile.cpu.vendor" : vendor
483 }, no_virt
=True).count()
488 def cpu_speed_average(self
):
491 all
= self
.query({}, no_virt
=True)
493 # XXX ugly. needs to be done by group()
495 if not m
.has_key("profile"):
497 speed
+= m
.get("profile").get("cpu").get("speed")
499 return (speed
/ all
.count())
501 def get_cpu_bogomips_accumulated(self
):
502 profiles
= self
.query({}, no_virt
=True, fields
=["profile.cpu.bogomips"])
504 return sum([int(o
["profile"]["cpu"]["bogomips"]) for o
in profiles
])
507 def cpu_speed_map(self
):
510 for i
in range(len(CPU_SPEED_CONSTRAINTS
) - 1):
511 min, max = CPU_SPEED_CONSTRAINTS
[i
:i
+2]
513 cpu_speeds
[min, max] = \
515 "profile.cpu.speed" : {
516 "$gte" : min, "$lt" : max
518 }, no_virt
=True).count()
522 def get_cpu_cores_map(self
):
528 "profile.cpu.count" : i
,
529 }, no_virt
=True).count()
533 def get_memory_map(self
):
536 for i
in range(len(MEMORY_CONSTRAINTS
) - 1):
537 min, max = MEMORY_CONSTRAINTS
[i
:i
+2]
541 { "profile.system.memory" : {
542 "$gte" : min * 1024, "$lt" : max * 1024
544 }, no_virt
=True).count()
549 def memory_average(self
):
552 all
= self
.query({}, no_virt
=True)
554 # XXX ugly. needs to be done by group()
556 if not m
.has_key("profile"):
558 memory
+= int(m
.get("profile").get("system").get("memory"))
560 return (memory
/ all
.count()) / 1024
563 def hypervisor_vendors(self
):
564 return self
.query({}).distinct("profile.hypervisor.vendor")
567 def hypervisor_map(self
):
570 for hypervisor
in self
.hypervisor_vendors
:
571 hypervisors
[hypervisor
] = \
573 "profile.hypervisor.vendor" : hypervisor
579 def hypervisor_models(self
):
580 return self
.query({}).distinct("profile.hypervisor.model")
583 def virtual_map(self
):
589 for k
in virtual
.keys():
591 self
.query({ "profile.system.virtual": k
}).count()
597 return self
.query({}).distinct("profile.system.language")
599 def get_language_map(self
):
602 for language
in self
.languages
:
603 languages
[language
] = \
605 "profile.system.language" : language
612 return self
.query({}).distinct("profile.system.vendor")
615 def vendor_map(self
):
618 for vendor
in self
.vendors
:
621 "profile.system.vendor" : vendor
628 return self
.query({}).distinct("profile.system.model")
634 for model
in self
.models
:
637 "profile.system.model" : model
644 return self
.query({}).distinct("profile.cpu.arch")
650 for arch
in self
.arches
:
653 "profile.cpu.arch" : arch
660 return self
.query({}).distinct("profile.system.kernel_release")
663 def kernel_map(self
):
666 for kernel
in self
.kernels
:
669 "profile.system.kernel_release" : kernel
676 return self
.query({}).distinct("profile.system.release")
679 def release_map(self
):
682 for release
in self
.releases
:
683 releases
[release
] = \
685 "profile.system.release" : release
690 def get_device_percentage(self
, bus
, vendor_id
, model_id
):
691 profiles_with_device
= self
.query({
692 "profile.devices.subsystem" : bus
,
693 "profile.devices.vendor" : vendor_id
,
694 "profile.devices.model" : model_id
,
697 profiles_all
= self
.query({})
699 if not profiles_all
.count():
702 return profiles_with_device
.count() / profiles_all
.count()
704 def get_cpu_flag_map(self
, flags
):
705 # XXX needs a cleanup
707 _flags
= { True : 0 }
709 if type(flags
) == type("a"):
715 "profile.cpu.flags" : flag
,
716 }, no_virt
=True).count()
718 _flags
[False] = self
.query({}, no_virt
=True).count() - _flags
[True]
723 def geo_locations(self
):
724 return [code
.lower() for code
in self
.query({}, all
=True).distinct("geoip.country_code")]
726 def get_geo_location_map(self
):
730 for geo_location
in self
.geo_locations
:
731 geo_locations
[geo_location
] = \
733 "geoip.country_code" : geo_location
.upper()
736 count
+= geo_locations
[geo_location
]
738 for geo_location
in geo_locations
.keys():
739 geo_locations
[geo_location
] /= count
743 def get_models_by_vendor(self
, subsystem
, vendor_id
):
746 profiles_all
= self
.query({})
748 for profile
in profiles_all
:
749 if not profile
.has_key("profile"):
752 profile
= Profile(profile
)
754 for device
in profile
.devices
:
755 if not device
.vendor
== vendor_id
:
758 if not device
in devices
:
759 devices
.append(device
)
763 def get_network_zones_map(self
):
764 zones
= { "green" : 0, "blue" : 0, "orange" : 0, "red" : 0 }
766 all
= self
.query({ "profile.network" : { "$exists" : True }})
768 for zone
in zones
.keys():
769 zones
[zone
] = self
.query({
770 "profile.network.%s" % zone
: True,
771 }).count() / all
.count()
775 def get_profile_ratio(self
):
776 return (self
.query({}).count(), self
.get_profile_count())
778 def get_updated_since(self
, since
, _query
={}):
779 since
= datetime
.datetime
.utcnow() - datetime
.timedelta(**since
)
781 query
= { "updated" : { "$gte" : since
}}
784 return self
.query(query
)
786 def get_updates_by_release_since(self
, since
):
789 for release
in self
.releases
:
790 updates
[release
] = self
.get_updated_since(since
,
791 { "profile.system.release" : release
}).count()
796 if __name__
== "__main__":