Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc-2-2
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-2-2
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