You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							145 lines
						
					
					
						
							5.1 KiB
						
					
					
				
			
		
		
	
	
							145 lines
						
					
					
						
							5.1 KiB
						
					
					
				import os
 | 
						|
import re
 | 
						|
from urllib.parse import urlparse
 | 
						|
from collections import defaultdict
 | 
						|
from itertools import chain
 | 
						|
 | 
						|
from tools.lib.auth_config import get_token
 | 
						|
from tools.lib.api import CommaApi
 | 
						|
 | 
						|
SEGMENT_NAME_RE = r'[a-z0-9]{16}[|_][0-9]{4}-[0-9]{2}-[0-9]{2}--[0-9]{2}-[0-9]{2}-[0-9]{2}--[0-9]+'
 | 
						|
EXPLORER_FILE_RE = r'^({})--([a-z]+\.[a-z0-9]+)$'.format(SEGMENT_NAME_RE)
 | 
						|
OP_SEGMENT_DIR_RE = r'^({})$'.format(SEGMENT_NAME_RE)
 | 
						|
 | 
						|
QLOG_FILENAMES = ['qlog.bz2']
 | 
						|
LOG_FILENAMES = ['rlog.bz2', 'raw_log.bz2']
 | 
						|
CAMERA_FILENAMES = ['fcamera.hevc', 'video.hevc']
 | 
						|
 | 
						|
class Route(object):
 | 
						|
  def __init__(self, route_name, data_dir=None):
 | 
						|
    self.route_name = route_name.replace('_', '|')
 | 
						|
    if data_dir is not None:
 | 
						|
      self._segments = self._get_segments_local(data_dir)
 | 
						|
    else:
 | 
						|
      self._segments = self._get_segments_remote()
 | 
						|
    self.max_seg_number = self._segments[-1].canonical_name.segment_num
 | 
						|
 | 
						|
  @property
 | 
						|
  def segments(self):
 | 
						|
    return self._segments
 | 
						|
 | 
						|
  def log_paths(self):
 | 
						|
    log_path_by_seg_num = {s.canonical_name.segment_num: s.log_path for s in self._segments}
 | 
						|
    return [log_path_by_seg_num.get(i, None) for i in range(self.max_seg_number+1)]
 | 
						|
 | 
						|
  def qlog_paths(self):
 | 
						|
    qlog_path_by_seg_num = {s.canonical_name.segment_num: s.qlog_path for s in self._segments}
 | 
						|
    return [qlog_path_by_seg_num.get(i, None) for i in range(self.max_seg_number+1)]
 | 
						|
 | 
						|
  def camera_paths(self):
 | 
						|
    camera_path_by_seg_num = {s.canonical_name.segment_num: s.camera_path for s in self._segments}
 | 
						|
    return [camera_path_by_seg_num.get(i, None) for i in range(self.max_seg_number+1)]
 | 
						|
 | 
						|
  def _get_segments_remote(self):
 | 
						|
    api = CommaApi(get_token())
 | 
						|
    route_files = api.get('v1/route/' + self.route_name + '/files')
 | 
						|
 | 
						|
    segments = {}
 | 
						|
    for url in chain.from_iterable(route_files.values()):
 | 
						|
      _, dongle_id, time_str, segment_num, fn = urlparse(url).path.rsplit('/', maxsplit=4)
 | 
						|
      segment_name = f'{dongle_id}|{time_str}--{segment_num}'
 | 
						|
      if segments.get(segment_name):
 | 
						|
        segments[segment_name] = RouteSegment(
 | 
						|
          segment_name,
 | 
						|
          url if fn in LOG_FILENAMES else segments[segment_name].log_path,
 | 
						|
          url if fn in QLOG_FILENAMES else segments[segment_name].qlog_path,
 | 
						|
          url if fn in CAMERA_FILENAMES else segments[segment_name].camera_path
 | 
						|
        )
 | 
						|
      else:
 | 
						|
        segments[segment_name] = RouteSegment(
 | 
						|
          segment_name,
 | 
						|
          url if fn in LOG_FILENAMES else None,
 | 
						|
          url if fn in QLOG_FILENAMES else None,
 | 
						|
          url if fn in CAMERA_FILENAMES else None
 | 
						|
        )
 | 
						|
 | 
						|
    return sorted(segments.values(), key=lambda seg: seg.canonical_name.segment_num)
 | 
						|
 | 
						|
  def _get_segments_local(self, data_dir):
 | 
						|
    files = os.listdir(data_dir)
 | 
						|
    segment_files = defaultdict(list)
 | 
						|
 | 
						|
    for f in files:
 | 
						|
      fullpath = os.path.join(data_dir, f)
 | 
						|
      explorer_match = re.match(EXPLORER_FILE_RE, f)
 | 
						|
      op_match = re.match(OP_SEGMENT_DIR_RE, f)
 | 
						|
 | 
						|
      if explorer_match:
 | 
						|
        segment_name, fn = explorer_match.groups()
 | 
						|
        if segment_name.replace('_', '|').startswith(self.route_name):
 | 
						|
          segment_files[segment_name].append((fullpath, fn))
 | 
						|
      elif op_match and os.path.isdir(fullpath):
 | 
						|
        segment_name, = op_match.groups()
 | 
						|
        if segment_name.startswith(self.route_name):
 | 
						|
          for seg_f in os.listdir(fullpath):
 | 
						|
            segment_files[segment_name].append((os.path.join(fullpath, seg_f), seg_f))
 | 
						|
      elif f == self.route_name:
 | 
						|
        for seg_num in os.listdir(fullpath):
 | 
						|
          if not seg_num.isdigit():
 | 
						|
            continue
 | 
						|
 | 
						|
          segment_name = '{}--{}'.format(self.route_name, seg_num)
 | 
						|
          for seg_f in os.listdir(os.path.join(fullpath, seg_num)):
 | 
						|
            segment_files[segment_name].append((os.path.join(fullpath, seg_num, seg_f), seg_f))
 | 
						|
 | 
						|
    segments = []
 | 
						|
    for segment, files in segment_files.items():
 | 
						|
 | 
						|
      try:
 | 
						|
        log_path = next(path for path, filename in files if filename in LOG_FILENAMES)
 | 
						|
      except StopIteration:
 | 
						|
        log_path = None
 | 
						|
 | 
						|
      try:
 | 
						|
        qlog_path = next(path for path, filename in files if filename in QLOG_FILENAMES)
 | 
						|
      except StopIteration:
 | 
						|
        qlog_path = None
 | 
						|
 | 
						|
      try:
 | 
						|
        camera_path = next(path for path, filename in files if filename in CAMERA_FILENAMES)
 | 
						|
      except StopIteration:
 | 
						|
        camera_path = None
 | 
						|
 | 
						|
      segments.append(RouteSegment(segment, log_path, qlog_path, camera_path))
 | 
						|
 | 
						|
    if len(segments) == 0:
 | 
						|
      raise ValueError('Could not find segments for route {} in data directory {}'.format(self.route_name, data_dir))
 | 
						|
    return sorted(segments, key=lambda seg: seg.canonical_name.segment_num)
 | 
						|
 | 
						|
class RouteSegment(object):
 | 
						|
  def __init__(self, name, log_path, qlog_path, camera_path):
 | 
						|
    self._name = RouteSegmentName(name)
 | 
						|
    self.log_path = log_path
 | 
						|
    self.qlog_path = qlog_path
 | 
						|
    self.camera_path = camera_path
 | 
						|
 | 
						|
  @property
 | 
						|
  def name(self):
 | 
						|
    return str(self._name)
 | 
						|
 | 
						|
  @property
 | 
						|
  def canonical_name(self):
 | 
						|
    return self._name
 | 
						|
 | 
						|
class RouteSegmentName(object):
 | 
						|
  def __init__(self, name_str):
 | 
						|
    self._segment_name_str = name_str
 | 
						|
    self._route_name_str, num_str = self._segment_name_str.rsplit("--", 1)
 | 
						|
    self._num = int(num_str)
 | 
						|
 | 
						|
  @property
 | 
						|
  def segment_num(self):
 | 
						|
    return self._num
 | 
						|
 | 
						|
  def __str__(self):
 | 
						|
    return self._segment_name_str
 | 
						|
 |