Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc
Commits
c39b64d5
Commit
c39b64d5
authored
Nov 20, 2007
by
Yoann Peronneau
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
* demux/playlist/itml.c: iTunes Media Library importer (not tested yet)
parent
930bf06c
Changes
5
Show whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
591 additions
and
0 deletions
+591
-0
modules/demux/playlist/Modules.am
modules/demux/playlist/Modules.am
+2
-0
modules/demux/playlist/itml.c
modules/demux/playlist/itml.c
+500
-0
modules/demux/playlist/itml.h
modules/demux/playlist/itml.h
+81
-0
modules/demux/playlist/playlist.c
modules/demux/playlist/playlist.c
+5
-0
modules/demux/playlist/playlist.h
modules/demux/playlist/playlist.h
+3
-0
No files found.
modules/demux/playlist/Modules.am
View file @
c39b64d5
...
...
@@ -14,4 +14,6 @@ SOURCES_playlist = \
qtl.c \
gvp.c \
ifo.c \
itml.c \
itml.h \
$(NULL)
modules/demux/playlist/itml.c
0 → 100644
View file @
c39b64d5
/*******************************************************************************
* itml.c : iTunes Music Library import functions
*******************************************************************************
* Copyright (C) 2007 the VideoLAN team
* $Id: $
*
* Authors: Yoann Peronneau <yoann@videolan.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*******************************************************************************/
/**
* \file modules/demux/playlist/itml.c
* \brief iTunes Music Library import functions
*/
#include <vlc/vlc.h>
#include <vlc_demux.h>
#include "playlist.h"
#include "vlc_xml.h"
#include "vlc_strings.h"
#include "vlc_url.h"
#include "itml.h"
struct
demux_sys_t
{
int
i_ntracks
;
};
static
int
Control
(
demux_t
*
,
int
,
va_list
);
static
int
Demux
(
demux_t
*
);
/**
* \brief iTML submodule initialization function
*/
int
E_
(
Import_iTML
)(
vlc_object_t
*
p_this
)
{
DEMUX_BY_EXTENSION_OR_FORCED_MSG
(
".xml"
,
"itml"
,
"using iTunes Media Library reader"
);
return
VLC_SUCCESS
;
}
void
E_
(
Close_iTML
)(
vlc_object_t
*
p_this
)
{
demux_t
*
p_demux
=
(
demux_t
*
)
p_this
;
free
(
p_demux
->
p_sys
);
}
/**
* \brief demuxer function for iTML parsing
*/
int
Demux
(
demux_t
*
p_demux
)
{
int
i_ret
=
VLC_SUCCESS
;
xml_t
*
p_xml
=
NULL
;
xml_reader_t
*
p_xml_reader
=
NULL
;
char
*
psz_name
=
NULL
;
INIT_PLAYLIST_STUFF
;
p_demux
->
p_sys
->
i_ntracks
=
0
;
/* create new xml parser from stream */
p_xml
=
xml_Create
(
p_demux
);
if
(
!
p_xml
)
i_ret
=
VLC_ENOMOD
;
else
{
p_xml_reader
=
xml_ReaderCreate
(
p_xml
,
p_demux
->
s
);
if
(
!
p_xml_reader
)
i_ret
=
VLC_EGENERIC
;
}
/* locating the root node */
if
(
i_ret
==
VLC_SUCCESS
)
{
do
{
if
(
xml_ReaderRead
(
p_xml_reader
)
!=
1
)
{
msg_Err
(
p_demux
,
"can't read xml stream"
);
i_ret
=
VLC_EGENERIC
;
}
}
while
(
i_ret
==
VLC_SUCCESS
&&
xml_ReaderNodeType
(
p_xml_reader
)
!=
XML_READER_STARTELEM
);
}
/* checking root node name */
if
(
i_ret
==
VLC_SUCCESS
)
{
psz_name
=
xml_ReaderName
(
p_xml_reader
);
if
(
!
psz_name
||
strcmp
(
psz_name
,
"plist"
)
)
{
msg_Err
(
p_demux
,
"invalid root node name: %s"
,
psz_name
);
i_ret
=
VLC_EGENERIC
;
}
FREE_NAME
();
}
if
(
i_ret
==
VLC_SUCCESS
)
{
xml_elem_hnd_t
pl_elements
[]
=
{
{
"dict"
,
COMPLEX_CONTENT
,
{.
cmplx
=
parse_plist_dict
}
}
};
i_ret
=
parse_plist_node
(
p_demux
,
p_playlist
,
p_current_input
,
NULL
,
p_xml_reader
,
"plist"
,
pl_elements
);
HANDLE_PLAY_AND_RELEASE
;
}
if
(
p_xml_reader
)
xml_ReaderDelete
(
p_xml
,
p_xml_reader
);
if
(
p_xml
)
xml_Delete
(
p_xml
);
return
-
1
;
/* Needed for correct operation of go back */
}
/** \brief dummy function for demux callback interface */
static
int
Control
(
demux_t
*
p_demux
,
int
i_query
,
va_list
args
)
{
return
VLC_EGENERIC
;
}
/**
* \brief parse the root node of the playlist
*/
static
vlc_bool_t
parse_plist_node
COMPLEX_INTERFACE
{
char
*
psz_name
=
NULL
;
char
*
psz_value
=
NULL
;
vlc_bool_t
b_version_found
=
VLC_FALSE
;
/* read all playlist attributes */
while
(
xml_ReaderNextAttr
(
p_xml_reader
)
==
VLC_SUCCESS
)
{
psz_name
=
xml_ReaderName
(
p_xml_reader
);
psz_value
=
xml_ReaderValue
(
p_xml_reader
);
if
(
!
psz_name
||
!
psz_value
)
{
msg_Err
(
p_demux
,
"invalid xml stream @ <plist>"
);
FREE_ATT
();
return
VLC_FALSE
;
}
/* attribute: version */
if
(
!
strcmp
(
psz_name
,
"version"
)
)
{
b_version_found
=
VLC_TRUE
;
if
(
strcmp
(
psz_value
,
"1.0"
)
)
msg_Warn
(
p_demux
,
"unsupported iTunes Media Library version"
);
}
/* unknown attribute */
else
msg_Warn
(
p_demux
,
"invalid <plist> attribute:
\"
%s
\"
"
,
psz_name
);
FREE_ATT
();
}
/* attribute version is mandatory !!! */
if
(
!
b_version_found
)
msg_Warn
(
p_demux
,
"<plist> requires
\"
version
\"
attribute"
);
return
parse_dict
(
p_demux
,
p_playlist
,
p_input_item
,
NULL
,
p_xml_reader
,
"plist"
,
p_handlers
);
}
/**
* \brief parse a <dict>
* \param COMPLEX_INTERFACE
*/
static
vlc_bool_t
parse_dict
COMPLEX_INTERFACE
{
int
i_node
;
char
*
psz_name
=
NULL
;
char
*
psz_value
=
NULL
;
char
*
psz_key
=
NULL
;
xml_elem_hnd_t
*
p_handler
=
NULL
;
while
(
xml_ReaderRead
(
p_xml_reader
)
==
1
)
{
i_node
=
xml_ReaderNodeType
(
p_xml_reader
);
switch
(
i_node
)
{
case
XML_READER_NONE
:
break
;
case
XML_READER_STARTELEM
:
/* element start tag */
psz_name
=
xml_ReaderName
(
p_xml_reader
);
if
(
!
psz_name
||
!*
psz_name
)
{
msg_Err
(
p_demux
,
"invalid xml stream"
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
/* choose handler */
for
(
p_handler
=
p_handlers
;
p_handler
->
name
&&
strcmp
(
psz_name
,
p_handler
->
name
);
p_handler
++
);
if
(
!
p_handler
->
name
)
{
msg_Err
(
p_demux
,
"unexpected element <%s>"
,
psz_name
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
FREE_NAME
();
/* complex content is parsed in a separate function */
if
(
p_handler
->
type
==
COMPLEX_CONTENT
)
{
if
(
p_handler
->
pf_handler
.
cmplx
(
p_demux
,
p_playlist
,
p_input_item
,
NULL
,
p_xml_reader
,
p_handler
->
name
,
NULL
)
)
{
p_handler
=
NULL
;
FREE_ATT_KEY
();
}
else
{
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
}
break
;
case
XML_READER_TEXT
:
/* simple element content */
FREE_ATT
();
psz_value
=
xml_ReaderValue
(
p_xml_reader
);
if
(
!
psz_value
)
{
msg_Err
(
p_demux
,
"invalid xml stream"
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
break
;
case
XML_READER_ENDELEM
:
/* element end tag */
psz_name
=
xml_ReaderName
(
p_xml_reader
);
if
(
!
psz_name
)
{
msg_Err
(
p_demux
,
"invalid xml stream"
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
/* leave if the current parent node <track> is terminated */
if
(
!
strcmp
(
psz_name
,
psz_element
)
)
{
FREE_ATT_KEY
();
return
VLC_TRUE
;
}
/* there MUST have been a start tag for that element name */
if
(
!
p_handler
||
!
p_handler
->
name
||
strcmp
(
p_handler
->
name
,
psz_name
))
{
msg_Err
(
p_demux
,
"there's no open element left for <%s>"
,
psz_name
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
/* special case: key */
if
(
!
strcmp
(
p_handler
->
name
,
"key"
)
)
{
psz_key
=
strdup
(
psz_value
);
}
/* call the simple handler */
else
if
(
p_handler
->
pf_handler
.
smpl
)
{
p_handler
->
pf_handler
.
smpl
(
p_track
,
psz_key
,
psz_value
);
}
FREE_ATT
();
p_handler
=
NULL
;
break
;
default:
/* unknown/unexpected xml node */
msg_Err
(
p_demux
,
"unexpected xml node %i"
,
i_node
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
FREE_NAME
();
}
msg_Err
(
p_demux
,
"unexpected end of xml data"
);
FREE_ATT_KEY
();
return
VLC_FALSE
;
}
static
vlc_bool_t
parse_plist_dict
COMPLEX_INTERFACE
{
xml_elem_hnd_t
pl_elements
[]
=
{
{
"dict"
,
COMPLEX_CONTENT
,
{.
cmplx
=
parse_tracks_dict
}
},
{
"array"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"key"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"integer"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"string"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"date"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"true"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"false"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
NULL
,
UNKNOWN_CONTENT
,
{
NULL
}
}
};
return
parse_dict
(
p_demux
,
p_playlist
,
p_input_item
,
NULL
,
p_xml_reader
,
"dict"
,
pl_elements
);
}
static
vlc_bool_t
parse_tracks_dict
COMPLEX_INTERFACE
{
xml_elem_hnd_t
tracks_elements
[]
=
{
{
"dict"
,
COMPLEX_CONTENT
,
{.
cmplx
=
parse_track_dict
}
},
{
"key"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
NULL
,
UNKNOWN_CONTENT
,
{
NULL
}
}
};
parse_dict
(
p_demux
,
p_playlist
,
p_input_item
,
NULL
,
p_xml_reader
,
"dict"
,
tracks_elements
);
msg_Info
(
p_demux
,
"added %i tracks successfully"
,
p_demux
->
p_sys
->
i_ntracks
);
return
VLC_TRUE
;
}
static
vlc_bool_t
parse_track_dict
COMPLEX_INTERFACE
{
input_item_t
*
p_new_input
=
NULL
;
int
i_ret
=
-
1
;
char
*
psz_uri
=
NULL
;
p_track
=
new_track
();
xml_elem_hnd_t
track_elements
[]
=
{
{
"array"
,
COMPLEX_CONTENT
,
{.
cmplx
=
skip_element
}
},
{
"key"
,
SIMPLE_CONTENT
,
{.
smpl
=
save_data
}
},
{
"integer"
,
SIMPLE_CONTENT
,
{.
smpl
=
save_data
}
},
{
"string"
,
SIMPLE_CONTENT
,
{.
smpl
=
save_data
}
},
{
"date"
,
SIMPLE_CONTENT
,
{.
smpl
=
save_data
}
},
{
"true"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
"false"
,
SIMPLE_CONTENT
,
{
NULL
}
},
{
NULL
,
UNKNOWN_CONTENT
,
{
NULL
}
}
};
i_ret
=
parse_dict
(
p_demux
,
p_playlist
,
p_input_item
,
p_track
,
p_xml_reader
,
"dict"
,
track_elements
);
msg_Dbg
(
p_demux
,
"name: %s, artist: %s, album: %s, genre: %s, trackNum: %s, location: %s"
,
p_track
->
name
,
p_track
->
artist
,
p_track
->
album
,
p_track
->
genre
,
p_track
->
trackNum
,
p_track
->
location
);
if
(
!
p_track
->
location
)
{
msg_Err
(
p_demux
,
"Track needs Location"
);
free_track
(
p_track
);
return
VLC_FALSE
;
}
psz_uri
=
decode_URI_duplicate
(
p_track
->
location
);
if
(
psz_uri
)
{
if
(
strlen
(
psz_uri
)
>
17
&&
!
strncmp
(
psz_uri
,
"file://localhost/"
,
17
)
)
{
/* remove 'localhost/' */
strcpy
(
psz_uri
+
7
,
psz_uri
+
17
);
msg_Info
(
p_demux
,
"Adding '%s'"
,
psz_uri
);
p_new_input
=
input_ItemNewExt
(
p_playlist
,
psz_uri
,
NULL
,
0
,
NULL
,
-
1
);
input_ItemAddSubItem
(
p_input_item
,
p_new_input
);
/* add meta info */
add_meta
(
p_new_input
,
p_track
);
p_demux
->
p_sys
->
i_ntracks
++
;
}
else
{
msg_Err
(
p_demux
,
"Don't know how to handle %s"
,
psz_uri
);
}
free
(
psz_uri
);
}
free_track
(
p_track
);
return
i_ret
;
}
static
track_elem_t
*
new_track
()
{
track_elem_t
*
p_track
=
NULL
;
p_track
=
(
track_elem_t
*
)
malloc
(
sizeof
(
track_elem_t
)
);
if
(
p_track
)
{
p_track
->
name
=
NULL
;
p_track
->
artist
=
NULL
;
p_track
->
album
=
NULL
;
p_track
->
genre
=
NULL
;
p_track
->
trackNum
=
NULL
;
p_track
->
location
=
NULL
;
p_track
->
duration
=
0
;
}
return
p_track
;
}
static
void
free_track
(
track_elem_t
*
p_track
)
{
fprintf
(
stderr
,
"free track
\n
"
);
if
(
!
p_track
)
return
;
FREE
(
p_track
->
name
)
FREE
(
p_track
->
artist
)
FREE
(
p_track
->
album
)
FREE
(
p_track
->
genre
)
FREE
(
p_track
->
trackNum
)
FREE
(
p_track
->
location
)
p_track
->
duration
=
0
;
free
(
p_track
);
}
static
vlc_bool_t
save_data
SIMPLE_INTERFACE
{
/* exit if setting is impossible */
if
(
!
psz_name
||
!
psz_value
||
!
p_track
)
return
VLC_FALSE
;
/* re-convert xml special characters inside psz_value */
resolve_xml_special_chars
(
psz_value
);
#define SAVE_INFO( name, value ) \
if( !strcmp( psz_name, name ) ) { p_track->value = strdup( psz_value ); }
SAVE_INFO
(
"Name"
,
name
)
else
SAVE_INFO
(
"Artist"
,
artist
)
else
SAVE_INFO
(
"Album"
,
album
)
else
SAVE_INFO
(
"Genre"
,
genre
)
else
SAVE_INFO
(
"Track Number"
,
trackNum
)
else
SAVE_INFO
(
"Location"
,
location
)
else
if
(
!
strcmp
(
psz_name
,
"Total Time"
)
)
{
long
i_num
=
atol
(
psz_value
);
p_track
->
duration
=
(
mtime_t
)
i_num
*
1000
;
}
return
VLC_TRUE
;
}
/**
* \brief handles the supported <track> sub-elements
*/
static
vlc_bool_t
add_meta
(
input_item_t
*
p_input_item
,
track_elem_t
*
p_track
)
{
/* exit if setting is impossible */
if
(
!
p_input_item
||
!
p_track
)
return
VLC_FALSE
;
#define SET_INFO( func, prop ) \
if( p_track->prop ) { func( p_input_item, p_track->prop ); }
SET_INFO
(
input_item_SetTitle
,
name
)
SET_INFO
(
input_item_SetArtist
,
artist
)
SET_INFO
(
input_item_SetAlbum
,
album
)
SET_INFO
(
input_item_SetGenre
,
genre
)
SET_INFO
(
input_item_SetTrackNum
,
trackNum
)
SET_INFO
(
input_item_SetDuration
,
duration
)
return
VLC_TRUE
;
}
/**
* \brief skips complex element content that we can't manage
*/
static
vlc_bool_t
skip_element
COMPLEX_INTERFACE
{
char
*
psz_endname
;
while
(
xml_ReaderRead
(
p_xml_reader
)
==
1
)
{
if
(
xml_ReaderNodeType
(
p_xml_reader
)
==
XML_READER_ENDELEM
)
{
psz_endname
=
xml_ReaderName
(
p_xml_reader
);
if
(
!
psz_endname
)
return
VLC_FALSE
;
if
(
!
strcmp
(
psz_element
,
psz_endname
)
)
{
free
(
psz_endname
);
return
VLC_TRUE
;
}
else
free
(
psz_endname
);
}
}
return
VLC_FALSE
;
}
modules/demux/playlist/itml.h
0 → 100644
View file @
c39b64d5
/*******************************************************************************
* itml.c : iTunes Music Library import functions
*******************************************************************************
* Copyright (C) 2007 the VideoLAN team
* $Id: $
*
* Authors: Yoann Peronneau <yoann@videolan.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*******************************************************************************/
/**
* \file modules/demux/playlist/itml.h
* \brief iTunes Music Library import: prototypes, datatypes, defines
*/
/* defines */
#define FREE(v) if (v) {free(v);v=NULL;}
#define FREE_NAME() if (psz_name) {free(psz_name);psz_name=NULL;}
#define FREE_VALUE() if (psz_value) {free(psz_value);psz_value=NULL;}
#define FREE_KEY() if (psz_key) {free(psz_key);psz_key=NULL;}
#define FREE_ATT() FREE_NAME();FREE_VALUE()
#define FREE_ATT_KEY() FREE_NAME();FREE_VALUE();FREE_KEY()
#define UNKNOWN_CONTENT 0
#define SIMPLE_CONTENT 1
#define COMPLEX_CONTENT 2
#define SIMPLE_INTERFACE (track_elem_t *p_track,\
const char *psz_name,\
char *psz_value)
#define COMPLEX_INTERFACE (demux_t *p_demux,\
playlist_t *p_playlist,\
input_item_t *p_input_item,\
track_elem_t *p_track,\
xml_reader_t *p_xml_reader,\
const char *psz_element,\
struct xml_elem_hnd *p_handlers)
/* datatypes */
typedef
struct
{
char
*
name
,
*
artist
,
*
album
,
*
genre
,
*
trackNum
,
*
location
;
mtime_t
duration
;
}
track_elem_t
;
struct
xml_elem_hnd
{
const
char
*
name
;
int
type
;
union
{
vlc_bool_t
(
*
smpl
)
SIMPLE_INTERFACE
;
vlc_bool_t
(
*
cmplx
)
COMPLEX_INTERFACE
;
}
pf_handler
;
};
typedef
struct
xml_elem_hnd
xml_elem_hnd_t
;
/* prototypes */
static
vlc_bool_t
parse_plist_node
COMPLEX_INTERFACE
;
static
vlc_bool_t
skip_element
COMPLEX_INTERFACE
;
static
vlc_bool_t
parse_dict
COMPLEX_INTERFACE
;
static
vlc_bool_t
parse_plist_dict
COMPLEX_INTERFACE
;
static
vlc_bool_t
parse_tracks_dict
COMPLEX_INTERFACE
;
static
vlc_bool_t
parse_track_dict
COMPLEX_INTERFACE
;
static
vlc_bool_t
save_data
SIMPLE_INTERFACE
;
static
vlc_bool_t
add_meta
(
input_item_t
*
,
track_elem_t
*
);
static
track_elem_t
*
new_track
(
void
);
static
void
free_track
(
track_elem_t
*
);
modules/demux/playlist/playlist.c
View file @
c39b64d5
...
...
@@ -123,6 +123,11 @@ vlc_module_begin();
set_description
(
_
(
"Dummy ifo demux"
)
);
set_capability
(
"demux2"
,
12
);
set_callbacks
(
E_
(
Import_IFO
),
E_
(
Close_IFO
)
);
add_submodule
();
set_description
(
_
(
"iTunes Music Library importer"
)
);
add_shortcut
(
"itml"
);
set_capability
(
"demux2"
,
10
);
set_callbacks
(
E_
(
Import_iTML
),
E_
(
Close_iTML
)
);
vlc_module_end
();
...
...
modules/demux/playlist/playlist.h
View file @
c39b64d5
...
...
@@ -73,6 +73,9 @@ void E_(Close_IFO) ( vlc_object_t * );
int
E_
(
Import_VideoPortal
)
(
vlc_object_t
*
);
void
E_
(
Close_VideoPortal
)
(
vlc_object_t
*
);
int
E_
(
Import_iTML
)
(
vlc_object_t
*
);
void
E_
(
Close_iTML
)
(
vlc_object_t
*
);
#define INIT_PLAYLIST_STUFF \
playlist_t *p_playlist = pl_Yield( p_demux ); \
input_thread_t *p_input_thread = (input_thread_t *)vlc_object_find( p_demux, VLC_OBJECT_INPUT, FIND_PARENT ); \
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment