diff --git a/src/include/66/enum.h b/src/include/66/enum.h
index 022cd2d76a49d57f5d6bfab9069b8272e323c572..a1bbbf71cb62c85d04fdc25e83749f0a32dd3898 100644
--- a/src/include/66/enum.h
+++ b/src/include/66/enum.h
@@ -33,6 +33,7 @@ enum key_enum_e
 	DESCRIPTION ,
 	CONTENTS ,
 	DEPENDS ,
+	OPTSDEPS ,
 	OPTIONS ,
 	NOTIFY ,
 	USER ,
@@ -120,6 +121,7 @@ static key_description_t const main_section_list[] =
 	{ .name = "@name", .expected = LINE, .mandatory = OPTS },
 	{ .name = "@description", .expected = QUOTE, .mandatory = OPTS },
 	{ .name = "@depends", .expected = BRACKET, .mandatory = OPTS },
+	{ .name = "@optsdepends", .expected = BRACKET, .mandatory = OPTS },
 	{ .name = "@contents", .expected = BRACKET, .mandatory = BUNDLE },
 	{ .name = "@options", .expected = BRACKET, .mandatory = OPTS },
 	{ .name = "@flags", .expected = BRACKET, .mandatory = OPTS },
@@ -166,7 +168,7 @@ static key_description_t const environment_section_list[] =
 	{ .name = 0 }
 } ;
 
-static int const total_list_el[6] = { 17, 5, 5, 12, 2, 0 } ;
+static int const total_list_el[6] = { 18, 5, 5, 12, 2, 0 } ;
 
 static key_all_t const total_list[] =
 {
diff --git a/src/include/66/parser.h b/src/include/66/parser.h
index d7c33ca2f389ad152de99a439c9d2fdb6c966851..494828a3c80a8b131abda65f3859cfb9d810b0c9 100644
--- a/src/include/66/parser.h
+++ b/src/include/66/parser.h
@@ -104,8 +104,10 @@ struct sv_name_s
 	int itype ;/**int type =30->classic,31->bundle,32->longrun,33->oneshot*/
 	int name ;//pos in keep
 	unsigned int description ;//pos in keep
-	unsigned int idga ; //pos in genalloc gadeps
-	unsigned int nga ; //len of idga in genalloc gadeps
+	unsigned int idga ; //pos in genalloc gadeps-> depends field
+	unsigned int nga ; //number or deps in genalloc gadeps->depends field
+	unsigned int idopts ;// pos in genalloc gadeps -> optional depends
+	unsigned int nopts ; // number of optinal depends in genalloc gadeps->optional depends
 	unsigned int logname ; //pos in keep
 	unsigned int dstlog ; //pos in keep
 } ;
@@ -253,8 +255,9 @@ extern void freed_parser(void) ;
 /** enable phase */
 extern int parser(sv_alltype *service,stralloc *src,char const *svname) ;
 extern int parse_service_check_enabled(ssexec_t *info, char const *svname,uint8_t force,uint8_t *exist) ;
-extern int parse_service_before(ssexec_t *info, stralloc *parsed_list, char const *sv,unsigned int *nbsv, stralloc *sasv,uint8_t force,uint8_t *exist) ;
-extern int parse_service_deps(ssexec_t *info,stralloc *parsed_list, sv_alltype *sv_before, char const *sv,unsigned int *nbsv,stralloc *sasv,uint8_t force) ;
+extern int parse_service_before(ssexec_t *info, stralloc *parsed_list, stralloc *opts_deps_list, char const *sv,unsigned int *nbsv, stralloc *sasv,uint8_t force,uint8_t *exist) ;
+extern int parse_service_deps(ssexec_t *info,stralloc *parsed_list, stralloc *opts_deps_list, sv_alltype *sv_before, char const *sv,unsigned int *nbsv,stralloc *sasv,uint8_t force) ;
+extern int parse_service_opts_deps(ssexec_t *info,stralloc *parsed_list,stralloc *opts_deps_list,sv_alltype *sv_before,char const *sv,unsigned int *nbsv,stralloc *sasv,uint8_t force) ;
 extern int parse_add_service(stralloc *parsed_list,sv_alltype *sv_before,char const *service,unsigned int *nbsv,uid_t owner) ;
 /** split */
 extern int section_get_range(section_t *sasection,stralloc *src) ;
diff --git a/src/lib66/get_enum.c b/src/lib66/get_enum.c
index b2bd1bd8f22c6f59031b6a1e52d352d036926414..a3ccf551a3ed39d0121216690c7918b55e00787a 100644
--- a/src/lib66/get_enum.c
+++ b/src/lib66/get_enum.c
@@ -31,6 +31,7 @@ char const *get_keybyid(key_enum_t key)
 			(key == DESCRIPTION ) ? "@description" :
 			(key == CONTENTS ) ? "@contents" :
 			(key == DEPENDS ) ? "@depends" :
+			(key == OPTSDEPS ) ? "@optsdepends" :
 			(key == OPTIONS ) ? "@options" :
 			(key == NOTIFY ) ? "@notify" :
 			(key == USER ) ? "@user" :
diff --git a/src/lib66/parser_enabled.c b/src/lib66/parser_enabled.c
index 306c47bc8c2268cd1b43125f8a0a59679f672474..f808f4a006d011ce4f0325b46066fae5bbc721ec 100644
--- a/src/lib66/parser_enabled.c
+++ b/src/lib66/parser_enabled.c
@@ -92,7 +92,7 @@ int parse_add_service(stralloc *parsed_list,sv_alltype *sv_before,char const *se
 		return 0 ;
 }
 
-int parse_service_deps(ssexec_t *info,stralloc *parsed_list, sv_alltype *sv_before, char const *sv,unsigned int *nbsv,stralloc *sasv,uint8_t force)
+int parse_service_deps(ssexec_t *info,stralloc *parsed_list,stralloc *opts_deps_list, sv_alltype *sv_before, char const *sv,unsigned int *nbsv,stralloc *sasv,uint8_t force)
 {
 	uint8_t exist = 0 ;
 	char *dname = 0 ;
@@ -113,7 +113,7 @@ int parse_service_deps(ssexec_t *info,stralloc *parsed_list, sv_alltype *sv_befo
 				log_warnu("resolve source path of: ",dname) ;
 				goto err ;
 			}
-			if (!parse_service_before(info,parsed_list,newsv.s,nbsv,sasv,force,&exist)) goto err ;
+			if (!parse_service_before(info,parsed_list,opts_deps_list,newsv.s,nbsv,sasv,force,&exist)) goto err ;
 		}
 	}
 	else log_trace(sv,": haven't dependencies") ;
@@ -124,7 +124,70 @@ int parse_service_deps(ssexec_t *info,stralloc *parsed_list, sv_alltype *sv_befo
 		return 0 ;
 }
 
-int parse_service_before(ssexec_t *info,stralloc *parsed_list, char const *sv,unsigned int *nbsv, stralloc *sasv,uint8_t force,uint8_t *exist)
+int parse_service_opts_deps(ssexec_t *info,stralloc *parsed_list,stralloc *opts_deps_list,sv_alltype *sv_before,char const *sv,unsigned int *nbsv,stralloc *sasv,uint8_t force)
+{
+	stralloc newsv = STRALLOC_ZERO ;
+	size_t pos = 0 , baselen = strlen(info->base.s) + SS_SYSTEM_LEN ;
+	uint8_t exist = 0, found = 0 ;
+	char *optname = 0 ;
+	char btmp[baselen + 1] ;
+	auto_strings(btmp,info->base.s,SS_SYSTEM) ;
+
+	// only pass here for the first time
+	if (!opts_deps_list->len)
+	{
+		if (!sastr_dir_get(opts_deps_list, btmp,SS_BACKUP + 1, S_IFDIR)) log_warnusys_return(LOG_EXIT_ZERO,"get list of tree at: ",btmp) ;
+	}
+	if (sv_before->cname.nopts)
+	{
+		// may have no tree yet
+		if (opts_deps_list->len)
+		{
+			size_t id = sv_before->cname.idopts, nid = sv_before->cname.nopts ;
+			for (;nid; id += strlen(deps.s + id) + 1, nid--)
+			{
+
+				newsv.len = 0 ;
+				optname = deps.s + id ;
+
+				for(pos = 0 ; pos < opts_deps_list->len ; pos += strlen(opts_deps_list->s + pos) +1 )
+				{
+					found = 0 ;
+					char *tree = opts_deps_list->s + pos ;
+					size_t treelen = strlen(tree) ;
+					char tmp[baselen + 1 + treelen + SS_SVDIRS_LEN + 1] ;
+					auto_strings(tmp,btmp,"/",tree,SS_SVDIRS) ;
+
+					// already added on a tree
+					if (ss_resolve_check(tmp,optname))
+					{
+						found = 1 ;
+						log_trace("optional service dependency: ",optname," is already enabled at tree: ",btmp,"/",tree) ;
+						break ;
+					}
+				}
+				if (!found)
+				{
+					if (!ss_resolve_src_path(&newsv,optname,info))
+					{
+						log_warnu_return(LOG_EXIT_ZERO,"resolve source path of: ",optname) ;
+						goto err ;
+					}
+					if (!parse_service_before(info,parsed_list,opts_deps_list,newsv.s,nbsv,sasv,force,&exist))
+						goto err ;
+				}
+			}
+		}
+	}
+
+	stralloc_free(&newsv) ;
+	return 1 ;
+	err:
+		stralloc_free(&newsv) ;
+		return 0 ;
+}
+
+int parse_service_before(ssexec_t *info,stralloc *parsed_list,stralloc *opts_deps_list, char const *sv,unsigned int *nbsv, stralloc *sasv,uint8_t force,uint8_t *exist)
 {
 	
 	int r, insta ;
@@ -204,8 +267,10 @@ int parse_service_before(ssexec_t *info,stralloc *parsed_list, char const *sv,un
 	if (!parse_add_service(parsed_list,&sv_before,svpath,nbsv,info->owner)) return 0 ;
 	
 	if ((sv_before.cname.itype > CLASSIC && force > 1) || !(*exist))
-		if (!parse_service_deps(info,parsed_list,&sv_before,sv,nbsv,sasv,force)) return 0 ;
-	
+	{
+		if (!parse_service_deps(info,parsed_list,opts_deps_list,&sv_before,sv,nbsv,sasv,force)) return 0 ;
+		if (!parse_service_opts_deps(info,parsed_list,opts_deps_list,&sv_before,sv,nbsv,sasv,force)) return 0 ;
+	}
 	freed:
 	return 1 ;
 }
diff --git a/src/lib66/parser_utils.c b/src/lib66/parser_utils.c
index 82594cdd495069ba5eb8ce37a11b1979f62c2289..1000f542720e4d00b6964c6c687046cff8a43990 100644
--- a/src/lib66/parser_utils.c
+++ b/src/lib66/parser_utils.c
@@ -634,6 +634,17 @@ int keep_common(sv_alltype *service,keynocheck *nocheck,int svtype)
 				service->cname.nga++ ;
 			}
 			break ;
+		case OPTSDEPS:
+			if ((service->cname.itype == CLASSIC) || (service->cname.itype == BUNDLE))
+				log_warn_return(LOG_EXIT_ZERO,"key: ",get_keybyid(nocheck->idkey),": is not valid for type ",get_keybyid(service->cname.itype)) ;
+			if (!get_clean_val(nocheck)) return 0 ;
+			service->cname.idopts = deps.len ;
+			for (;pos < *chlen; pos += strlen(chval + pos)+1)
+			{
+				if (!stralloc_catb(&deps,chval + pos,strlen(chval + pos) + 1)) log_warnsys_return(LOG_EXIT_ZERO,"stralloc") ;
+				service->cname.nopts++ ;
+			}
+			break ;
 		case CONTENTS:
 			if (service->cname.itype != BUNDLE)
 				log_warn_return(LOG_EXIT_ZERO,"key: ",get_keybyid(nocheck->idkey),": is not valid for type ",get_keybyid(service->cname.itype)) ;
diff --git a/src/lib66/ssexec_enable.c b/src/lib66/ssexec_enable.c
index 5ac0862fd899b920532574781dd271d7257a6f59..4a1f6ad1f467bc3e1d08069697f1789dc1f63625 100644
--- a/src/lib66/ssexec_enable.c
+++ b/src/lib66/ssexec_enable.c
@@ -61,7 +61,8 @@ static void start_parser(stralloc *list,ssexec_t *info, unsigned int *nbsv)
 	size_t i = 0, len = list->len ;
 	
 	stralloc sasv = STRALLOC_ZERO ;
-	stralloc tmp = STRALLOC_ZERO ;
+	stralloc parsed_list = STRALLOC_ZERO ;
+	stralloc opts_deps_list = STRALLOC_ZERO ;
 	
 	for (;i < len; i += strlen(list->s + i) + 1)
 	{
@@ -73,11 +74,12 @@ static void start_parser(stralloc *list,ssexec_t *info, unsigned int *nbsv)
 		r = parse_service_check_enabled(info,svname,FORCE,&exist) ;
 		if (!r) log_dieu(LOG_EXIT_SYS,"check enabled service: ",svname) ;
 		if (r == 2) continue ;
-		if (!parse_service_before(info,&tmp,name,nbsv,&sasv,FORCE,&exist))
+		if (!parse_service_before(info,&parsed_list,&opts_deps_list,name,nbsv,&sasv,FORCE,&exist))
 			log_dieu(LOG_EXIT_SYS,"parse service file: ",svname,": or its dependencies") ;
 	}
 	stralloc_free(&sasv) ;
-	stralloc_free(&tmp) ;	
+	stralloc_free(&parsed_list) ;
+	stralloc_free(&opts_deps_list) ;
 }
 
 int ssexec_enable(int argc, char const *const *argv,char const *const *envp,ssexec_t *info)