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

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

Fix bug when clean environment is used

  • Property svn:executable set to *
File size: 17.0 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    return os.path.isdir(mirror_location) and check_structure(mirror_location,need_structure)
219
220def check_structure(path_from,child=''):
221    DBG=False
222    dbg=lambda x: DBG and sys.stdout.write(x+'\n')
223    typeparam=type(child)
224    if typeparam == type(dict()):
225        for item in child:
226            new_path=path_from+'/'+item
227            if not check_structure(path_from,item):
228                return False
229            if not check_structure(new_path,child[item]):
230                return False
231    elif typeparam == type(list()):
232        for item in child:
233            if not check_structure(path_from,item):
234                return False
235    elif typeparam == type(tuple()):
236        mode,value=child
237        if mode=='regexp':
238            reg=re.compile(value)
239            for f in os.listdir(path_from):
240                path_check=path_from+'/'+f
241                dbg('checking file ' + path_check)
242                if os.path.isfile(path_check):
243                    if reg.match(f):
244                        return True
245            return False
246        else:
247            path_check=path_from+'/'+value
248            dbg('checking file ' + path_check)
249            return os.path.isfile(path_check)
250    elif typeparam == type(str()):
251        path_check=path_from+'/'+child
252        dbg('checking dir ' + path_check)
253        if not os.path.isdir(path_check):
254            return False
255    return True
256
257def store_result(results='',namevar='',action='store'):
258    global eval_mode
259    global result
260    global exit_return_code_mode
261
262    if action == 'init':
263        if eval_mode:
264            result={}
265            return 0
266        else:
267            result=[]
268            return 0
269    elif action == 'store':
270        if results=='':
271            raise Exception('Missing param results to store')
272        if eval_mode and namevar=='':
273            raise Exception('Eval mode need namevar param')
274
275        if eval_mode:
276            if namevar=='USERTYPE':
277                res=results
278                if res[0] == '*':
279                    result['PROMOTED_USER']='yes'
280                    res=results[1:]
281                else:
282                    result['PROMOTED_USER']='no'
283                result['USERNAME']=res.split('/')[0]
284                result['LOGIN_TYPE']=res.split('/')[1]
285            elif namevar=='SESSION_TYPE':
286                res=results.split(',')
287                result['LIVE']='no'
288                result['LTSP']='no'
289                result['THIN']='no'
290                result['SEMI']='no'
291                result['FAT']='no'
292
293                if len(res) > 1:
294                    result['LIVE']='yes'
295                if res[0] != 'fat':
296                    result['LTSP']='yes'
297
298                result[res[0].upper()]='yes'
299            elif namevar=='FLAVOUR':
300                res=results[-1]
301                result['SERVER']='no'
302                result['DESKTOP']='no'
303                result['CLIENT']='no'
304                result['INFANTIL']='no'
305                result['MUSIC']='no'
306                result['PIME']='no'
307
308                #lliurex 15 specific options & catch all
309                if res.upper() == 'NETWORK-CLIENT-PROMO':
310                    result['CLIENT']='yes'
311                elif res.upper() == 'LLIUREX':
312                    if 'INFANTIL' in [ x.upper() for x in results ]:
313                        result['INFANTIL']='yes'
314                    else:
315                        result['DESKTOP']='yes'
316                #end lliurex15 specific options
317                elif 'CLIENT' in [ x.upper() for x in results ] and 'EDU' in [ x.upper() for x in results ]:
318                    result['CLIENT']='yes'
319                else:
320                    result[res.upper()]='yes'
321            else:
322                result[namevar]=results
323               
324        else:
325            if type(list()) == type(results):
326                res=results[-1]
327            elif type(str()) == type(results):
328                res=results
329            else:
330                raise Exception('Unknown result type to store')
331            if namevar=='FLAVOUR':
332                #lliurex 15 specific options & catch all
333                if res.upper() == 'NETWORK-CLIENT-PROMO':
334                    res='client'
335                elif res.upper() == 'LLIUREX':
336                    if 'INFANTIL' in [ x.upper() for x in results ]:
337                        res='infantil'
338                    else:
339                        res='desktop'
340                #end lliurex15 specific options
341                elif 'CLIENT' in [ x.upper() for x in results ] and 'EDU' in [ x.upper() for x in results ]:
342                    res='client'
343            result.append(res)
344    elif action == 'print':
345        if exit_return_code_mode != False:
346            if exit_return_code_mode.upper() == 'LOCAL':
347                if result['LOGIN_TYPE']=='local':
348                    sys.exit(0)
349                else:
350                    sys.exit(1)
351            elif exit_return_code_mode.upper() == 'LDAP':
352                if result['LOGIN_TYPE']=='ldap':
353                    sys.exit(0)
354                else:
355                    sys.exit(1)
356            elif exit_return_code_mode.upper() == 'PROMOTED':
357                if result['PROMOTED_USER']=='yes':
358                    sys.exit(0)
359                else:
360                    sys.exit(1)
361            elif exit_return_code_mode.upper() == 'MIRROR':
362                if result['MIRROR']=='True':
363                    sys.exit(0)
364                else:
365                    sys.exit(1)
366            else:
367                if result[exit_return_code_mode.upper()]=='yes':
368                    sys.exit(0)
369                else:
370                    sys.exit(1)
371        if len(result) > 0:
372            if eval_mode:
373                for k,v in result.items():
374                    print(k+'='+v)
375            else:
376                print (','.join(result))
377    else:
378        raise Exception('Unknow action')
379
380
381#def store_result():
382
383
384#
385# MAIN PROGRAM
386#
387
388parser = argparse.ArgumentParser(description='Get information about running environment')
389parser.add_argument('-a','--all',metavar='',action='store_const',help='Get all information',const=True)
390parser.add_argument('-m','--mirror',metavar='',action='store_const',help='Check mirror available',const=True)
391parser.add_argument('-e','--eval',metavar='',action='store_const',help='Show all information to evaluate in bash variables',const=True)
392parser.add_argument('-s','--session',metavar='',action='store_const',help='Get current session type',const=True)
393parser.add_argument('-f','--flavour',metavar='',action='store_const',help='Get the flavour of current system',const=True)
394parser.add_argument('-u','--usertype',metavar='username',nargs='?',const='',help='Get the usertype from current user or from passed username')
395code_types=['live','ltsp','fat','semi','thin','desktop','server','client','infantil','pime','music','local','ldap','promoted','mirror']
396parser.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')
397# lliurex-version options
398parser.add_argument('-n','--number',metavar='',action='store_const',const=True,help='Get the cdd number version')
399parser.add_argument('-v','--version',metavar='',action='store_const',const=True,help='Get the cdd version')
400parser.add_argument('-t','--test',metavar='cdd_name',help='Test if the cdd is installed')
401parser.add_argument('--history',metavar='',action='store_const',const=True,help='Get the installed meta\'s history')
402args=parser.parse_args()
403
404#print(args)
405
406args_all=args.all
407#Commented due to compatibility with lliurex-version without parameters
408#args_all=True
409#if args.all != True:
410#    for arg in vars(args):
411#        if getattr(args,arg) != None:
412#            args_all=False
413#            break
414arg_none_for_lliurex_version=True
415if args.all != True:
416    for arg in vars(args):
417        if getattr(args,arg) != None:
418            arg_none_for_lliurex_version=False
419            break
420else:
421    arg_none_for_lliurex_version=False
422
423eval_mode=args.eval
424
425exit_return_code_mode=False
426if args.with_return_code != None:
427    eval_mode=True
428    args_all=True
429    exit_return_code_mode=args.with_return_code[0]
430   
431
432store_result(action='init')
433
434try:
435    if args_all or args.session != None:
436        store_result(detect_type(),'SESSION_TYPE')
437    if args_all or args.flavour != None:
438        store_result(detect_flavour(),'FLAVOUR')
439    if args_all or args.usertype != None:
440        store_result(detect_user(args.usertype),'USERTYPE')
441    if args_all or args.mirror != None:
442        store_result(str(check_mirror()),'MIRROR')
443
444#lliurex version options
445    if args.number != None:
446        print(detect_num_cdd())
447    if args.version != None:
448        print(', '.join(detect_flavour()))
449    if arg_none_for_lliurex_version:
450        ret=detect_flavour()
451        ret.append(detect_num_cdd())
452        print(', '.join(ret))
453    if args.test != None:
454        try:
455            ret=detect_flavour()
456        except:
457            sys.exit(1)
458        if args.test in ret:
459            sys.exit(0)
460        else:
461            sys.exit(1)
462    if args.history != None:
463        print(get_history_version().rstrip())
464#end lliurex-version options
465    store_result(action='print')
466except Exception as e:
467    print('Error '+str(e))
468    sys.exit(1)
469
470sys.exit(0)
Note: See TracBrowser for help on using the repository browser.