source: lliurex-detect/trunk/fuentes/lliurex-detect.install/usr/bin/lliurex-detect @ 4448

Last change on this file since 4448 was 4448, checked in by mabarracus, 2 years ago

Allow mirrors without 'lists' directory

  • Property svn:executable set to *
File size: 17.3 KB
Line 
1#!/usr/bin/python3
2
3#
4# Examples:
5# Export all variables into environment:>  eval export $(lliurex-detect -e -a)
6# Test one condition:> lliurex-detect -x thin && echo si || echo no
7# Test one condition:> if lliurex-detect -x ltsp; then echo si; else echo no; fi
8# Get flavour:> FLA=$(lliurex-detect -f)
9#
10import sys
11import os 
12import pwd,grp
13import argparse
14import re
15# import glob  # Commented: python 3.5 is not available on lliurex 15
16import fnmatch # allows search recursively without glob
17
18from subprocess import check_output
19
20ltsConfFile="/etc/lts.conf"
21
22def detect_live():
23    test1=False
24    test2=False
25    test3=False
26
27    if os.path.isdir('/rofs'):
28        r1=re.compile('^tmpfs\s/cow\stmpfs')
29        r2=re.compile('^/cow\s/\soverlayfs')
30        r3=re.compile('^/dev/loop0\s/rofs\ssquashfs')
31        with open('/proc/mounts') as mounts_file:
32            for line in mounts_file.readlines():
33                if r1.match(line):
34                    test1=True
35                if r2.match(line):
36                    test2=True
37                if r3.match(line):
38                    test3=True
39    if test1 and test2 and test3:
40        pass
41    else:
42        raise Exception('Not live')
43
44#def detect_live():
45
46def detect_flavour():
47# Use with python > 3.5 with glob new features
48#        cdd_content=[]
49#        file_name= [ filename for filename in glob.iglob('/usr/share/lliurex-cdd/**/cddflavour', recursive=True) ]
50#        if len(file_name) > 0:
51#           for cddfile in file_name:
52#               try:
53#                    with open(cddfile,'r') as fileccd:
54#                       cdd= [ line.strip() for line in fileccd ]
55#                       cdd_content.extend(cdd)
56#               except Exception as e:
57#                   raise Exception('cddflavour error '+str(e))
58#        else:
59#            file_name= [ filename for filename in glob.iglob('/usr/share/lliurex-cdd/**/cddversion', recursive=True) ]
60#            if len(file_name) > 0:
61#                for cddfile in file_name:
62#                    try:
63#                       with open(cddfile,'r') as fileccd:
64#                          cdd= [ line.strip() for line in fileccd ]
65#                          cdd_content.extend(cdd)
66#                    except Exception as e:
67#                        raise Exception('cddversion file error '+str(e))
68#            else:
69#               raise Exception('cddflavour or cddversion file not found! ')
70#       cdd_content=list(set(cdd_content))
71#       cdd_content.sort()
72#       return cdd_content
73
74# Code for python < 3.5 without glob
75        file_name=[]
76        cdd_content=[]
77        for root,dirnames,filenames in os.walk('/usr/share/lliurex-cdd'):
78            for filename in fnmatch.filter(filenames,'cddflavour'):
79                file_name.append(os.path.join(root,filename))
80        if len(file_name) > 0:
81            for cddfile in file_name:
82                try:
83                    with open(cddfile,'r') as fileccd:
84                        cdd= [ line.strip() for line in fileccd ]
85                        cdd_content.extend(cdd)
86                except Exception as e:
87                    cdd_content.append('None')
88                    #raise Exception('cddflavour file error '+str(e))
89        else:
90            for root,dirnames,filenames in os.walk('/usr/share/lliurex-cdd'):
91                for filename in fnmatch.filter(filenames,'cddversion'):
92                    file_name.append(os.path.join(root,filename))
93            if len(file_name) > 0:
94                for cddfile in file_name:
95                    try:
96                        with open(cddfile,'r') as fileccd:
97                            cdd= [ line.strip() for line in fileccd ]
98                            cdd_content.extend(cdd)
99                    except Exception as e:
100                        cdd_content.append('None')
101                        #raise Exception('cddversion file error '+str(e))
102            else:
103                #raise Exception('cddflavour or cddversion file not found! ')
104                cdd_content.append('None')
105        cdd_content=list(set(cdd_content))
106        cdd_content.sort()
107        return cdd_content
108#def detect_flavour
109
110def detect_num_cdd():
111    try:
112        with open(os.devnull, 'w') as devnull:
113            num_cdd=check_output(['dpkg-query','--showformat=\'${Version}\'','--show','lliurex-version-timestamp'],stderr=devnull).decode('utf-8')
114        return num_cdd.strip('\'')
115    except Exception as e1:
116        try:
117            with open('/usr/share/lliurex-cdd/version','r') as cdd_num_file:
118                return cdd_num_file.readline().strip()
119        except Exception as e2:
120            return '0'
121            #raise Exception('Error1: '+e1+' and Error2: '+e2)
122#def detect_num_cdd():
123
124def get_history_version():
125    try:
126        with open('/etc/lliurex-cdd-version','r') as etc_cdd_file:
127            return etc_cdd_file.read()
128    except Exception as e:
129        raise Exception('Error: '+e)
130#def get_history_version()
131
132def detect_type():
133#
134#   Possible output values: live,thin,semi,fat,unknown
135#
136#
137
138    display=os.environ.get('DISPLAY')
139    #On thin clients display is ip+display so it's at least 7 chars
140    if display != None and len(display)>= 7:
141        client_type='thin'
142    else:
143        if os.path.exists(ltsConfFile):
144        #Attempt to open lts.conf as is more reliable than check environment
145            try:
146                 re_true=re.compile('LTSP_FATCLIENT(\s)*=(\s)*true',re.IGNORECASE)
147                 re_false=re.compile('LTSP_FATCLIENT(\s)*=(\s)*false',re.IGNORECASE)
148                 with open(ltsConfFile) as ltsfile:
149                      for line in ltsfile.readlines():
150                             if re_true.match(line):
151                                 client_type='semi'
152                                 break
153                             elif re_false.match(line):
154                                 client_type='thin'
155                                 break
156                             else:
157                                 client_type='unknown'
158            except Exception as e:
159                fatclient=os.environ.get('LTSP_FATCLIENT')
160                if fatclient=='true':
161                    client_type="semi"
162                else:
163                    if fatclient=='false':
164                        client_type="thin"
165                    else:
166                        client_type="unknown" 
167        else:
168            client_type='fat'
169    try:
170        detect_live()
171        client_type=client_type+',live'
172    except:
173        pass
174
175    return client_type.rstrip()
176#def detect_type
177
178def detect_user(user=''):
179    if user == '' or user == None:
180        user_id=os.getuid()
181        user_name=pwd.getpwuid(user_id)[0]
182    else:
183        try:
184            user_uid=pwd.getpwnam(user)[3]
185        except:
186            raise Exception('user not found!')
187        user_name=user
188    #user_info [0]=>username [1]=>pwd [2]=>uid [3]=>gid [4]=>gecos [5]=>homedir [6]=>shell
189    #grp_info [0]=>name [1]=>pwd [2]=>gid [3]=>member
190    grupos = [ group[0] for group in grp.getgrall() if user_name in group[3] ]
191    with open('/etc/passwd','r') as filepwd:
192        localusers=[ line.split(':')[0] for line in filepwd.readlines() ]
193
194    ret=user_name
195    if 'admins' in grupos :
196        ret='*'+str(user_name)
197    if user_name in localusers:
198        ret += '/local'
199    else:
200        ret += '/ldap'
201    return ret
202#def detect_user():
203
204def check_mirror():
205    #
206    # Run checks against mirror
207    #
208    mirror_location='/net/mirror/llx16'
209    #dict = check keys & recurse into values
210    #list = check items
211    #tuple = check items(files or regexp) ; syntax=('type','value') where type = 'regexp' or something else to exact match
212    #string = check items
213
214    #need_structure={'pool':[{'main':{'l/lliurex-version-timestamp':('regexp','lliurex-version-timestamp_.*\.deb$')}},'universe','multiverse','preschool','restricted'],
215    #                'dists':['xenial','xenial-security','xenial-updates'],
216    #                'lists':[('regexp','.*xenial.*_amd64_Packages$'),('regexp','.*xenial.*_i386_Packages$')]}
217
218    need_structure={'pool':[{'main':{'l/lliurex-version-timestamp':('regexp','lliurex-version-timestamp_.*\.deb$')}},'universe','multiverse','preschool','restricted'],
219                    'dists':['xenial','xenial-security','xenial-updates']}
220
221    return os.path.isdir(mirror_location) and check_structure(mirror_location,need_structure)
222
223def check_structure(path_from,child=''):
224    DBG=False
225    dbg=lambda x: DBG and sys.stdout.write(x+'\n')
226    typeparam=type(child)
227    if typeparam == type(dict()):
228        for item in child:
229            new_path=path_from+'/'+item
230            if not check_structure(path_from,item):
231                return False
232            if not check_structure(new_path,child[item]):
233                return False
234    elif typeparam == type(list()):
235        for item in child:
236            if not check_structure(path_from,item):
237                return False
238    elif typeparam == type(tuple()):
239        mode,value=child
240        if mode=='regexp':
241            reg=re.compile(value)
242            for f in os.listdir(path_from):
243                path_check=path_from+'/'+f
244                dbg('checking file ' + path_check)
245                if os.path.isfile(path_check):
246                    if reg.match(f):
247                        return True
248            return False
249        else:
250            path_check=path_from+'/'+value
251            dbg('checking file ' + path_check)
252            return os.path.isfile(path_check)
253    elif typeparam == type(str()):
254        path_check=path_from+'/'+child
255        dbg('checking dir ' + path_check)
256        if not os.path.isdir(path_check):
257            return False
258    return True
259
260def store_result(results='',namevar='',action='store'):
261    global eval_mode
262    global result
263    global exit_return_code_mode
264
265    if action == 'init':
266        if eval_mode:
267            result={}
268            return 0
269        else:
270            result=[]
271            return 0
272    elif action == 'store':
273        if results=='':
274            raise Exception('Missing param results to store')
275        if eval_mode and namevar=='':
276            raise Exception('Eval mode need namevar param')
277
278        if eval_mode:
279            if namevar=='USERTYPE':
280                res=results
281                if res[0] == '*':
282                    result['PROMOTED_USER']='yes'
283                    res=results[1:]
284                else:
285                    result['PROMOTED_USER']='no'
286                result['USERNAME']=res.split('/')[0]
287                result['LOGIN_TYPE']=res.split('/')[1]
288            elif namevar=='SESSION_TYPE':
289                res=results.split(',')
290                result['LIVE']='no'
291                result['LTSP']='no'
292                result['THIN']='no'
293                result['SEMI']='no'
294                result['FAT']='no'
295
296                if len(res) > 1:
297                    result['LIVE']='yes'
298                if res[0] != 'fat':
299                    result['LTSP']='yes'
300
301                result[res[0].upper()]='yes'
302            elif namevar=='FLAVOUR':
303                res=results[-1]
304                result['SERVER']='no'
305                result['DESKTOP']='no'
306                result['CLIENT']='no'
307                result['INFANTIL']='no'
308                result['MUSIC']='no'
309                result['PIME']='no'
310
311                #lliurex 15 specific options & catch all
312                if res.upper() == 'NETWORK-CLIENT-PROMO':
313                    result['CLIENT']='yes'
314                elif res.upper() == 'LLIUREX':
315                    if 'INFANTIL' in [ x.upper() for x in results ]:
316                        result['INFANTIL']='yes'
317                    else:
318                        result['DESKTOP']='yes'
319                #end lliurex15 specific options
320                elif 'CLIENT' in [ x.upper() for x in results ] and 'EDU' in [ x.upper() for x in results ]:
321                    result['CLIENT']='yes'
322                else:
323                    result[res.upper()]='yes'
324            else:
325                result[namevar]=results
326               
327        else:
328            if type(list()) == type(results):
329                res=results[-1]
330            elif type(str()) == type(results):
331                res=results
332            else:
333                raise Exception('Unknown result type to store')
334            if namevar=='FLAVOUR':
335                #lliurex 15 specific options & catch all
336                if res.upper() == 'NETWORK-CLIENT-PROMO':
337                    res='client'
338                elif res.upper() == 'LLIUREX':
339                    if 'INFANTIL' in [ x.upper() for x in results ]:
340                        res='infantil'
341                    else:
342                        res='desktop'
343                #end lliurex15 specific options
344                elif 'CLIENT' in [ x.upper() for x in results ] and 'EDU' in [ x.upper() for x in results ]:
345                    res='client'
346            result.append(res)
347    elif action == 'print':
348        if exit_return_code_mode != False:
349            if exit_return_code_mode.upper() == 'LOCAL':
350                if result['LOGIN_TYPE']=='local':
351                    sys.exit(0)
352                else:
353                    sys.exit(1)
354            elif exit_return_code_mode.upper() == 'LDAP':
355                if result['LOGIN_TYPE']=='ldap':
356                    sys.exit(0)
357                else:
358                    sys.exit(1)
359            elif exit_return_code_mode.upper() == 'PROMOTED':
360                if result['PROMOTED_USER']=='yes':
361                    sys.exit(0)
362                else:
363                    sys.exit(1)
364            elif exit_return_code_mode.upper() == 'MIRROR':
365                if result['MIRROR']=='True':
366                    sys.exit(0)
367                else:
368                    sys.exit(1)
369            else:
370                if result[exit_return_code_mode.upper()]=='yes':
371                    sys.exit(0)
372                else:
373                    sys.exit(1)
374        if len(result) > 0:
375            if eval_mode:
376                for k,v in result.items():
377                    print(k+'='+v)
378            else:
379                print (','.join(result))
380    else:
381        raise Exception('Unknow action')
382
383
384#def store_result():
385
386
387#
388# MAIN PROGRAM
389#
390
391parser = argparse.ArgumentParser(description='Get information about running environment')
392parser.add_argument('-a','--all',metavar='',action='store_const',help='Get all information',const=True)
393parser.add_argument('-m','--mirror',metavar='',action='store_const',help='Check mirror available',const=True)
394parser.add_argument('-e','--eval',metavar='',action='store_const',help='Show all information to evaluate in bash variables',const=True)
395parser.add_argument('-s','--session',metavar='',action='store_const',help='Get current session type',const=True)
396parser.add_argument('-f','--flavour',metavar='',action='store_const',help='Get the flavour of current system',const=True)
397parser.add_argument('-u','--usertype',metavar='username',nargs='?',const='',help='Get the usertype from current user or from passed username')
398code_types=['live','ltsp','fat','semi','thin','desktop','server','client','infantil','pime','music','local','ldap','promoted','mirror']
399parser.add_argument('-x','--with-return-code',metavar='[live|ltsp|fat|semi|thin|desktop|server|client|infantil|pime|music|local|ldap|promoted|mirror]',nargs=1,choices=code_types,help='Execute mode testing value passed')
400# lliurex-version options
401parser.add_argument('-n','--number',metavar='',action='store_const',const=True,help='Get the cdd number version')
402parser.add_argument('-v','--version',metavar='',action='store_const',const=True,help='Get the cdd version')
403parser.add_argument('-t','--test',metavar='cdd_name',help='Test if the cdd is installed')
404parser.add_argument('--history',metavar='',action='store_const',const=True,help='Get the installed meta\'s history')
405args=parser.parse_args()
406
407#print(args)
408
409args_all=args.all
410#Commented due to compatibility with lliurex-version without parameters
411#args_all=True
412#if args.all != True:
413#    for arg in vars(args):
414#        if getattr(args,arg) != None:
415#            args_all=False
416#            break
417arg_none_for_lliurex_version=True
418if args.all != True:
419    for arg in vars(args):
420        if getattr(args,arg) != None:
421            arg_none_for_lliurex_version=False
422            break
423else:
424    arg_none_for_lliurex_version=False
425
426eval_mode=args.eval
427
428exit_return_code_mode=False
429if args.with_return_code != None:
430    eval_mode=True
431    args_all=True
432    exit_return_code_mode=args.with_return_code[0]
433   
434
435store_result(action='init')
436
437try:
438    if args_all or args.session != None:
439        store_result(detect_type(),'SESSION_TYPE')
440    if args_all or args.flavour != None:
441        store_result(detect_flavour(),'FLAVOUR')
442    if args_all or args.usertype != None:
443        store_result(detect_user(args.usertype),'USERTYPE')
444    if args_all or args.mirror != None:
445        store_result(str(check_mirror()),'MIRROR')
446
447#lliurex version options
448    if args.number != None:
449        print(detect_num_cdd())
450    if args.version != None:
451        print(', '.join(detect_flavour()))
452    if arg_none_for_lliurex_version:
453        ret=detect_flavour()
454        ret.append(detect_num_cdd())
455        print(', '.join(ret))
456    if args.test != None:
457        try:
458            ret=detect_flavour()
459        except:
460            sys.exit(1)
461        if args.test in ret:
462            sys.exit(0)
463        else:
464            sys.exit(1)
465    if args.history != None:
466        print(get_history_version().rstrip())
467#end lliurex-version options
468    store_result(action='print')
469except Exception as e:
470    print('Error '+str(e))
471    sys.exit(1)
472
473sys.exit(0)
Note: See TracBrowser for help on using the repository browser.