mirror of https://github.com/subgraph/fw-daemon
				
				
				
			Updated fw-settings to reflect all recent changes Moved rule notifications to dbus signal Lots of various fixes Local changes to gtk bindings ... And so much more...shw-merge
							parent
							
								
									61710110d2
								
							
						
					
					
						commit
						cfaf7b84ff
					
				| @ -0,0 +1,573 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <!-- Generated with glade 3.20.0 --> | ||||||
|  | <interface> | ||||||
|  |   <requires lib="gtk+" version="3.16"/> | ||||||
|  |   <object class="GtkDialog" id="dialog"> | ||||||
|  |     <property name="can_focus">False</property> | ||||||
|  |     <property name="title" translatable="yes">Edit Rule</property> | ||||||
|  |     <property name="role">SubgraphFirewallNewRule</property> | ||||||
|  |     <property name="resizable">False</property> | ||||||
|  |     <property name="modal">True</property> | ||||||
|  |     <property name="window_position">center</property> | ||||||
|  |     <property name="destroy_with_parent">True</property> | ||||||
|  |     <property name="icon_name">alacarte</property> | ||||||
|  |     <property name="type_hint">dialog</property> | ||||||
|  |     <property name="skip_taskbar_hint">True</property> | ||||||
|  |     <property name="deletable">False</property> | ||||||
|  |     <property name="startup_id">SubgraphFirewallNewRule</property> | ||||||
|  |     <child internal-child="vbox"> | ||||||
|  |       <object class="GtkBox" id="dialog-vbox1"> | ||||||
|  |         <property name="can_focus">False</property> | ||||||
|  |         <property name="halign">center</property> | ||||||
|  |         <property name="valign">center</property> | ||||||
|  |         <property name="margin_left">10</property> | ||||||
|  |         <property name="margin_right">10</property> | ||||||
|  |         <property name="margin_top">10</property> | ||||||
|  |         <property name="margin_bottom">10</property> | ||||||
|  |         <property name="orientation">vertical</property> | ||||||
|  |         <property name="spacing">5</property> | ||||||
|  |         <child internal-child="action_area"> | ||||||
|  |           <object class="GtkButtonBox" id="action_btnbox"> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |             <property name="hexpand">True</property> | ||||||
|  |             <property name="homogeneous">True</property> | ||||||
|  |             <property name="layout_style">spread</property> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkButton" id="allow_button"> | ||||||
|  |                 <property name="label" translatable="yes">_Allow</property> | ||||||
|  |                 <property name="sensitive">False</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="receives_default">True</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="use_underline">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="expand">True</property> | ||||||
|  |                 <property name="fill">True</property> | ||||||
|  |                 <property name="position">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkButton" id="cancel_button"> | ||||||
|  |                 <property name="label" translatable="yes">_Cancel</property> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="can_default">True</property> | ||||||
|  |                 <property name="receives_default">True</property> | ||||||
|  |                 <property name="use_underline">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="expand">True</property> | ||||||
|  |                 <property name="fill">True</property> | ||||||
|  |                 <property name="position">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkButton" id="ok_button"> | ||||||
|  |                 <property name="label" translatable="yes">_Ok</property> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="can_default">True</property> | ||||||
|  |                 <property name="receives_default">True</property> | ||||||
|  |                 <property name="use_underline">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="expand">True</property> | ||||||
|  |                 <property name="fill">True</property> | ||||||
|  |                 <property name="position">2</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">False</property> | ||||||
|  |             <property name="position">3</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |         <child> | ||||||
|  |           <object class="GtkComboBoxText" id="verb_combo"> | ||||||
|  |             <property name="visible">True</property> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |             <property name="active">0</property> | ||||||
|  |             <items> | ||||||
|  |               <item id="allow" translatable="yes">Allow</item> | ||||||
|  |               <item id="deny" translatable="yes">Deny</item> | ||||||
|  |             </items> | ||||||
|  |             <signal name="changed" handler="on_verb_changed" swapped="no"/> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">True</property> | ||||||
|  |             <property name="position">0</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |         <child> | ||||||
|  |           <object class="GtkGrid" id="grid3"> | ||||||
|  |             <property name="visible">True</property> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |             <property name="row_spacing">5</property> | ||||||
|  |             <property name="column_spacing">10</property> | ||||||
|  |             <property name="row_homogeneous">True</property> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="path_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Path:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="sandbox_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Sandbox:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="uid_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">UID:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">7</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="gid_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">GID:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">8</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkFileChooserButton" id="path_chooser"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="create_folders">False</property> | ||||||
|  |                 <property name="preview_widget_active">False</property> | ||||||
|  |                 <property name="use_preview_label">False</property> | ||||||
|  |                 <property name="title" translatable="yes">Select Executable Path</property> | ||||||
|  |                 <signal name="file-set" handler="on_path_changed" swapped="no"/> | ||||||
|  |                 <signal name="file-set" handler="on_path_set" swapped="no"/> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="scope_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Scope:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">9</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkComboBoxText" id="sandbox_combo"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="proto_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Protocol:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">6</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="host_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Host:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">4</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="host_entry"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="width_chars">64</property> | ||||||
|  |                 <property name="max_width_chars">34</property> | ||||||
|  |                 <property name="primary_icon_tooltip_markup" translatable="yes" context="The character * can be use to match any value.">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="placeholder_text" translatable="yes">Hostname or IP address</property> | ||||||
|  |                 <signal name="changed" handler="on_host_changed" swapped="no"/> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">4</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="port_entry"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="max_length">5</property> | ||||||
|  |                 <property name="width_chars">4</property> | ||||||
|  |                 <property name="max_width_chars">5</property> | ||||||
|  |                 <property name="primary_icon_tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="placeholder_text" translatable="yes">Port</property> | ||||||
|  |                 <signal name="changed" handler="on_port_changed" swapped="no"/> | ||||||
|  |                 <signal name="insert-text" handler="on_port_insert_text" swapped="no"/> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">5</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="port_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Port:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">5</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="sandbox_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">SANDBOX</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="pid_title"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Pid:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">2</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="pid_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">PID_LABEL</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">2</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="scope_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">SCOPE_LABEL</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">9</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkBox"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="homogeneous">True</property> | ||||||
|  |                 <property name="baseline_position">top</property> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkComboBoxText" id="uid_combo"> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">False</property> | ||||||
|  |                     <property name="active">0</property> | ||||||
|  |                     <property name="active_id">-1</property> | ||||||
|  |                     <items> | ||||||
|  |                       <item id="-1">Any User</item> | ||||||
|  |                     </items> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">0</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkCheckButton" id="uid_checkbox"> | ||||||
|  |                     <property name="label" translatable="yes">Apply</property> | ||||||
|  |                     <property name="sensitive">False</property> | ||||||
|  |                     <property name="can_focus">True</property> | ||||||
|  |                     <property name="receives_default">False</property> | ||||||
|  |                     <property name="no_show_all">True</property> | ||||||
|  |                     <property name="xalign">0</property> | ||||||
|  |                     <property name="draw_indicator">True</property> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">1</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">7</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkBox"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="homogeneous">True</property> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkComboBoxText" id="gid_combo"> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">False</property> | ||||||
|  |                     <property name="active">0</property> | ||||||
|  |                     <property name="active_id">-1</property> | ||||||
|  |                     <items> | ||||||
|  |                       <item id="-1">Any Group</item> | ||||||
|  |                     </items> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">0</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkCheckButton" id="gid_checkbox"> | ||||||
|  |                     <property name="label" translatable="yes">Apply</property> | ||||||
|  |                     <property name="sensitive">False</property> | ||||||
|  |                     <property name="can_focus">True</property> | ||||||
|  |                     <property name="receives_default">False</property> | ||||||
|  |                     <property name="no_show_all">True</property> | ||||||
|  |                     <property name="xalign">0</property> | ||||||
|  |                     <property name="draw_indicator">True</property> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">1</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">8</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="path_entry"> | ||||||
|  |                 <property name="sensitive">False</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="origin_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Origin:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">3</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="origin_entry"> | ||||||
|  |                 <property name="sensitive">False</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">3</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkComboBoxText" id="scope_combo"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="active">0</property> | ||||||
|  |                 <property name="active_id">0</property> | ||||||
|  |                 <items> | ||||||
|  |                   <item id="2" translatable="yes">Permanent</item> | ||||||
|  |                   <item id="0" translatable="yes">Session</item> | ||||||
|  |                   <item id="1" translatable="yes">Process</item> | ||||||
|  |                   <item id="4" translatable="yes">Once</item> | ||||||
|  |                   <item id="3" translatable="yes">System</item> | ||||||
|  |                 </items> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">9</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkBox"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="homogeneous">True</property> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkComboBoxText" id="proto_combo"> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">False</property> | ||||||
|  |                     <property name="active">1</property> | ||||||
|  |                     <property name="active_id">1</property> | ||||||
|  |                     <items> | ||||||
|  |                       <item translatable="yes">Any</item> | ||||||
|  |                       <item id="tcp" translatable="yes">TCP</item> | ||||||
|  |                       <item id="udp" translatable="yes">UDP</item> | ||||||
|  |                       <item id="icmp" translatable="yes">ICMP</item> | ||||||
|  |                     </items> | ||||||
|  |                     <signal name="changed" handler="on_proto_changed" swapped="no"/> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">0</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkCheckButton" id="tls_check"> | ||||||
|  |                     <property name="label" translatable="yes">TLS Only</property> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">True</property> | ||||||
|  |                     <property name="receives_default">False</property> | ||||||
|  |                     <property name="draw_indicator">True</property> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">1</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">6</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">True</property> | ||||||
|  |             <property name="padding">5</property> | ||||||
|  |             <property name="position">1</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |         <child> | ||||||
|  |           <object class="GtkSeparator"> | ||||||
|  |             <property name="visible">True</property> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">True</property> | ||||||
|  |             <property name="position">2</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |       </object> | ||||||
|  |     </child> | ||||||
|  |     <action-widgets> | ||||||
|  |       <action-widget response="3">allow_button</action-widget> | ||||||
|  |       <action-widget response="1">cancel_button</action-widget> | ||||||
|  |       <action-widget response="2">ok_button</action-widget> | ||||||
|  |     </action-widgets> | ||||||
|  |   </object> | ||||||
|  | </interface> | ||||||
| @ -0,0 +1,586 @@ | |||||||
|  | package definitions | ||||||
|  | 
 | ||||||
|  | func init() { | ||||||
|  | 	add(`RuleNew`, &defRuleNew{}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type defRuleNew struct{} | ||||||
|  | 
 | ||||||
|  | func (*defRuleNew) String() string { | ||||||
|  | 	return ` | ||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <!-- Generated with glade 3.20.0 --> | ||||||
|  | <interface> | ||||||
|  |   <requires lib="gtk+" version="3.16"/> | ||||||
|  |   <object class="GtkDialog" id="dialog"> | ||||||
|  |     <property name="can_focus">False</property> | ||||||
|  |     <property name="title" translatable="yes">Edit Rule</property> | ||||||
|  |     <property name="role">SubgraphFirewallNewRule</property> | ||||||
|  |     <property name="resizable">False</property> | ||||||
|  |     <property name="modal">True</property> | ||||||
|  |     <property name="window_position">center</property> | ||||||
|  |     <property name="destroy_with_parent">True</property> | ||||||
|  |     <property name="icon_name">alacarte</property> | ||||||
|  |     <property name="type_hint">dialog</property> | ||||||
|  |     <property name="skip_taskbar_hint">True</property> | ||||||
|  |     <property name="deletable">False</property> | ||||||
|  |     <property name="startup_id">SubgraphFirewallNewRule</property> | ||||||
|  |     <child internal-child="vbox"> | ||||||
|  |       <object class="GtkBox" id="dialog-vbox1"> | ||||||
|  |         <property name="can_focus">False</property> | ||||||
|  |         <property name="halign">center</property> | ||||||
|  |         <property name="valign">center</property> | ||||||
|  |         <property name="margin_left">10</property> | ||||||
|  |         <property name="margin_right">10</property> | ||||||
|  |         <property name="margin_top">10</property> | ||||||
|  |         <property name="margin_bottom">10</property> | ||||||
|  |         <property name="orientation">vertical</property> | ||||||
|  |         <property name="spacing">5</property> | ||||||
|  |         <child internal-child="action_area"> | ||||||
|  |           <object class="GtkButtonBox" id="action_btnbox"> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |             <property name="hexpand">True</property> | ||||||
|  |             <property name="homogeneous">True</property> | ||||||
|  |             <property name="layout_style">spread</property> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkButton" id="allow_button"> | ||||||
|  |                 <property name="label" translatable="yes">_Allow</property> | ||||||
|  |                 <property name="sensitive">False</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="receives_default">True</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="use_underline">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="expand">True</property> | ||||||
|  |                 <property name="fill">True</property> | ||||||
|  |                 <property name="position">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkButton" id="cancel_button"> | ||||||
|  |                 <property name="label" translatable="yes">_Cancel</property> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="can_default">True</property> | ||||||
|  |                 <property name="receives_default">True</property> | ||||||
|  |                 <property name="use_underline">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="expand">True</property> | ||||||
|  |                 <property name="fill">True</property> | ||||||
|  |                 <property name="position">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkButton" id="ok_button"> | ||||||
|  |                 <property name="label" translatable="yes">_Ok</property> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="can_default">True</property> | ||||||
|  |                 <property name="receives_default">True</property> | ||||||
|  |                 <property name="use_underline">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="expand">True</property> | ||||||
|  |                 <property name="fill">True</property> | ||||||
|  |                 <property name="position">2</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">False</property> | ||||||
|  |             <property name="position">3</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |         <child> | ||||||
|  |           <object class="GtkComboBoxText" id="verb_combo"> | ||||||
|  |             <property name="visible">True</property> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |             <property name="active">0</property> | ||||||
|  |             <items> | ||||||
|  |               <item id="allow" translatable="yes">Allow</item> | ||||||
|  |               <item id="deny" translatable="yes">Deny</item> | ||||||
|  |             </items> | ||||||
|  |             <signal name="changed" handler="on_verb_changed" swapped="no"/> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">True</property> | ||||||
|  |             <property name="position">0</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |         <child> | ||||||
|  |           <object class="GtkGrid" id="grid3"> | ||||||
|  |             <property name="visible">True</property> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |             <property name="row_spacing">5</property> | ||||||
|  |             <property name="column_spacing">10</property> | ||||||
|  |             <property name="row_homogeneous">True</property> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="path_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Path:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="sandbox_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Sandbox:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="uid_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">UID:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">7</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="gid_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">GID:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">8</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkFileChooserButton" id="path_chooser"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="create_folders">False</property> | ||||||
|  |                 <property name="preview_widget_active">False</property> | ||||||
|  |                 <property name="use_preview_label">False</property> | ||||||
|  |                 <property name="title" translatable="yes">Select Executable Path</property> | ||||||
|  |                 <signal name="file-set" handler="on_path_changed" swapped="no"/> | ||||||
|  |                 <signal name="file-set" handler="on_path_set" swapped="no"/> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="scope_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Scope:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">9</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkComboBoxText" id="sandbox_combo"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="proto_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Protocol:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">6</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="host_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Host:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">4</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="host_entry"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="width_chars">64</property> | ||||||
|  |                 <property name="max_width_chars">34</property> | ||||||
|  |                 <property name="primary_icon_tooltip_markup" translatable="yes" context="The character * can be use to match any value.">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="placeholder_text" translatable="yes">Hostname or IP address</property> | ||||||
|  |                 <signal name="changed" handler="on_host_changed" swapped="no"/> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">4</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="port_entry"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="max_length">5</property> | ||||||
|  |                 <property name="width_chars">4</property> | ||||||
|  |                 <property name="max_width_chars">5</property> | ||||||
|  |                 <property name="primary_icon_tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="placeholder_text" translatable="yes">Port</property> | ||||||
|  |                 <signal name="changed" handler="on_port_changed" swapped="no"/> | ||||||
|  |                 <signal name="insert-text" handler="on_port_insert_text" swapped="no"/> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">5</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="port_title"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Port:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">5</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="sandbox_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">SANDBOX</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">1</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="pid_title"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Pid:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">2</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="pid_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">PID_LABEL</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">2</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="scope_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">SCOPE_LABEL</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">9</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkBox"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="homogeneous">True</property> | ||||||
|  |                 <property name="baseline_position">top</property> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkComboBoxText" id="uid_combo"> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">False</property> | ||||||
|  |                     <property name="active">0</property> | ||||||
|  |                     <property name="active_id">-1</property> | ||||||
|  |                     <items> | ||||||
|  |                       <item id="-1">Any User</item> | ||||||
|  |                     </items> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">0</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkCheckButton" id="uid_checkbox"> | ||||||
|  |                     <property name="label" translatable="yes">Apply</property> | ||||||
|  |                     <property name="sensitive">False</property> | ||||||
|  |                     <property name="can_focus">True</property> | ||||||
|  |                     <property name="receives_default">False</property> | ||||||
|  |                     <property name="no_show_all">True</property> | ||||||
|  |                     <property name="xalign">0</property> | ||||||
|  |                     <property name="draw_indicator">True</property> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">1</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">7</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkBox"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="homogeneous">True</property> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkComboBoxText" id="gid_combo"> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">False</property> | ||||||
|  |                     <property name="active">0</property> | ||||||
|  |                     <property name="active_id">-1</property> | ||||||
|  |                     <items> | ||||||
|  |                       <item id="-1">Any Group</item> | ||||||
|  |                     </items> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">0</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkCheckButton" id="gid_checkbox"> | ||||||
|  |                     <property name="label" translatable="yes">Apply</property> | ||||||
|  |                     <property name="sensitive">False</property> | ||||||
|  |                     <property name="can_focus">True</property> | ||||||
|  |                     <property name="receives_default">False</property> | ||||||
|  |                     <property name="no_show_all">True</property> | ||||||
|  |                     <property name="xalign">0</property> | ||||||
|  |                     <property name="draw_indicator">True</property> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">1</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">8</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="path_entry"> | ||||||
|  |                 <property name="sensitive">False</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">0</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkLabel" id="origin_label"> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |                 <property name="tooltip_markup" translatable="yes">The character <b>*</b> can be use to match any value.</property> | ||||||
|  |                 <property name="halign">start</property> | ||||||
|  |                 <property name="hexpand">False</property> | ||||||
|  |                 <property name="label" translatable="yes">Origin:</property> | ||||||
|  |                 <attributes> | ||||||
|  |                   <attribute name="weight" value="bold"/> | ||||||
|  |                 </attributes> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">0</property> | ||||||
|  |                 <property name="top_attach">3</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkEntry" id="origin_entry"> | ||||||
|  |                 <property name="sensitive">False</property> | ||||||
|  |                 <property name="can_focus">True</property> | ||||||
|  |                 <property name="no_show_all">True</property> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">3</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkComboBoxText" id="scope_combo"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="active">0</property> | ||||||
|  |                 <property name="active_id">0</property> | ||||||
|  |                 <items> | ||||||
|  |                   <item id="2" translatable="yes">Permanent</item> | ||||||
|  |                   <item id="0" translatable="yes">Session</item> | ||||||
|  |                   <item id="1" translatable="yes">Process</item> | ||||||
|  |                   <item id="4" translatable="yes">Once</item> | ||||||
|  |                   <item id="3" translatable="yes">System</item> | ||||||
|  |                 </items> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">9</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |             <child> | ||||||
|  |               <object class="GtkBox"> | ||||||
|  |                 <property name="visible">True</property> | ||||||
|  |                 <property name="can_focus">False</property> | ||||||
|  |                 <property name="homogeneous">True</property> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkComboBoxText" id="proto_combo"> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">False</property> | ||||||
|  |                     <property name="active">1</property> | ||||||
|  |                     <property name="active_id">1</property> | ||||||
|  |                     <items> | ||||||
|  |                       <item translatable="yes">Any</item> | ||||||
|  |                       <item id="tcp" translatable="yes">TCP</item> | ||||||
|  |                       <item id="udp" translatable="yes">UDP</item> | ||||||
|  |                       <item id="icmp" translatable="yes">ICMP</item> | ||||||
|  |                     </items> | ||||||
|  |                     <signal name="changed" handler="on_proto_changed" swapped="no"/> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">0</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |                 <child> | ||||||
|  |                   <object class="GtkCheckButton" id="tls_check"> | ||||||
|  |                     <property name="label" translatable="yes">TLS Only</property> | ||||||
|  |                     <property name="visible">True</property> | ||||||
|  |                     <property name="can_focus">True</property> | ||||||
|  |                     <property name="receives_default">False</property> | ||||||
|  |                     <property name="draw_indicator">True</property> | ||||||
|  |                   </object> | ||||||
|  |                   <packing> | ||||||
|  |                     <property name="expand">False</property> | ||||||
|  |                     <property name="fill">True</property> | ||||||
|  |                     <property name="position">1</property> | ||||||
|  |                   </packing> | ||||||
|  |                 </child> | ||||||
|  |               </object> | ||||||
|  |               <packing> | ||||||
|  |                 <property name="left_attach">1</property> | ||||||
|  |                 <property name="top_attach">6</property> | ||||||
|  |               </packing> | ||||||
|  |             </child> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">True</property> | ||||||
|  |             <property name="padding">5</property> | ||||||
|  |             <property name="position">1</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |         <child> | ||||||
|  |           <object class="GtkSeparator"> | ||||||
|  |             <property name="visible">True</property> | ||||||
|  |             <property name="can_focus">False</property> | ||||||
|  |           </object> | ||||||
|  |           <packing> | ||||||
|  |             <property name="expand">False</property> | ||||||
|  |             <property name="fill">True</property> | ||||||
|  |             <property name="position">2</property> | ||||||
|  |           </packing> | ||||||
|  |         </child> | ||||||
|  |       </object> | ||||||
|  |     </child> | ||||||
|  |     <action-widgets> | ||||||
|  |       <action-widget response="3">allow_button</action-widget> | ||||||
|  |       <action-widget response="1">cancel_button</action-widget> | ||||||
|  |       <action-widget response="2">ok_button</action-widget> | ||||||
|  |     </action-widgets> | ||||||
|  |   </object> | ||||||
|  | </interface> | ||||||
|  | 
 | ||||||
|  | ` | ||||||
|  | } | ||||||
											
												
													File diff suppressed because it is too large
													Load Diff
												
											
										
									
								| @ -0,0 +1,38 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 
 | ||||||
|  | 	"github.com/subgraph/oz" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | var ozProfiles oz.Profiles | ||||||
|  | 
 | ||||||
|  | func init() { | ||||||
|  | 	c, err := oz.LoadConfig(oz.DefaultConfigPath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Fprintf(os.Stderr, "Unable to read oz config...") | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | 	p, err := oz.LoadProfiles(c.ProfileDir) | ||||||
|  | 	if err != nil { | ||||||
|  | 		fmt.Fprintf(os.Stderr, "Unable to read oz profiles...") | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | 	ozProfiles = p | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (fa *fwApp) initOZProfiles() { | ||||||
|  | 	for _, p := range ozProfiles { | ||||||
|  | 		// XXX: This actually should match against sgfw's opened sockets
 | ||||||
|  | 		switch { | ||||||
|  | 		case string(p.Networking.Nettype) == "host": | ||||||
|  | 			fallthrough | ||||||
|  | 		case len(p.Networking.Sockets) == 0: | ||||||
|  | 			continue | ||||||
|  | 		default: | ||||||
|  | 			fa.ozProfiles = append(fa.ozProfiles, p.Name) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -0,0 +1,901 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | 	spath "path" | ||||||
|  | 	"strings" | ||||||
|  | 	"strconv" | ||||||
|  | 	"sync" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	"github.com/subgraph/fw-daemon/sgfw" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gotk3/gotk3/gdk" | ||||||
|  | 	"github.com/gotk3/gotk3/glib" | ||||||
|  | 	"github.com/gotk3/gotk3/gtk" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type Prompt struct { | ||||||
|  | 	app *fwApp | ||||||
|  | 	tv *gtk.TreeView | ||||||
|  | 	ts *gtk.TreeStore | ||||||
|  | 	stack *gtk.Stack | ||||||
|  | 	pecols []*gtk.TreeViewColumn | ||||||
|  | 	pncol *gtk.TreeViewColumn | ||||||
|  | 	promptLock *sync.Mutex | ||||||
|  | 	recentLock *sync.Mutex | ||||||
|  | 	config *sgfw.FirewallConfigs | ||||||
|  | 	recentlyRemoved []string | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	COL_NO_NREFS = iota | ||||||
|  | 	COL_NO_ICON_PIXBUF | ||||||
|  | 	COL_NO_GUID | ||||||
|  | 	COL_NO_PATH | ||||||
|  | 	COL_NO_SANDBOX | ||||||
|  | 	COL_NO_ICON | ||||||
|  | 	COL_NO_PROTO | ||||||
|  | 	COL_NO_PID | ||||||
|  | 	COL_NO_DSTIP | ||||||
|  | 	COL_NO_HOSTNAME | ||||||
|  | 	COL_NO_PORT | ||||||
|  | 	COL_NO_UID | ||||||
|  | 	COL_NO_GID | ||||||
|  | 	COL_NO_ORIGIN | ||||||
|  | 	COL_NO_TIMESTAMP | ||||||
|  | 	COL_NO_IS_SOCKS | ||||||
|  | 	COL_NO_OPTSTRING | ||||||
|  | 	COL_NO_ACTION | ||||||
|  | 	COL_NO_FILLER | ||||||
|  | 	COL_NO_LAST | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | type ruleColumns struct { | ||||||
|  | 	nrefs     int | ||||||
|  | 	Path      string | ||||||
|  | 	Sandbox   string | ||||||
|  | 	GUID      string | ||||||
|  | 	Icon      string | ||||||
|  | 	Proto     string | ||||||
|  | 	Pid       int | ||||||
|  | 	Target    string | ||||||
|  | 	Hostname  string | ||||||
|  | 	Port      int | ||||||
|  | 	UID       int | ||||||
|  | 	GID       int | ||||||
|  | 	Uname     string | ||||||
|  | 	Gname     string | ||||||
|  | 	Origin    string | ||||||
|  | 	Timestamp string | ||||||
|  | 	IsSocks   bool | ||||||
|  | 	ForceTLS  bool | ||||||
|  | 	Scope     int | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func createPromptView(app *fwApp, sw *gtk.ScrolledWindow) (*Prompt, error) { | ||||||
|  | 	var err error | ||||||
|  | 	p := &Prompt{} | ||||||
|  | 	p.app = app | ||||||
|  | 	p.promptLock = &sync.Mutex{} | ||||||
|  | 	p.recentLock = &sync.Mutex{} | ||||||
|  | 	 | ||||||
|  | 	p.tv, err = gtk.TreeViewNew() | ||||||
|  | 
 | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err  | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.tv.SetSizeRequest(300, 300) | ||||||
|  | 	p.tv.SetHeadersClickable(true) | ||||||
|  | 	p.tv.SetEnableSearch(false) | ||||||
|  | 
 | ||||||
|  | 	p.tv.AppendColumn(createColumnText("#", COL_NO_NREFS)) | ||||||
|  | 	p.tv.AppendColumn(createColumnImg("", COL_NO_ICON_PIXBUF)) | ||||||
|  | 
 | ||||||
|  | 	guidcol := createColumnText("GUID", COL_NO_GUID) | ||||||
|  | 	guidcol.SetVisible(false) | ||||||
|  | 	p.tv.AppendColumn(guidcol) | ||||||
|  | 
 | ||||||
|  | 	p.tv.AppendColumn(createColumnText("Path", COL_NO_PATH)) | ||||||
|  | 
 | ||||||
|  | 	sbcol := createColumnText("Sandbox", COL_NO_SANDBOX) | ||||||
|  | 	sbcol.SetVisible(false) | ||||||
|  | 	p.tv.AppendColumn(sbcol) | ||||||
|  | 
 | ||||||
|  | 	icol := createColumnText("Icon", COL_NO_ICON) | ||||||
|  | 	icol.SetVisible(false) | ||||||
|  | 	p.tv.AppendColumn(icol) | ||||||
|  | 
 | ||||||
|  | 	var pecol *gtk.TreeViewColumn | ||||||
|  | 	p.tv.AppendColumn(createColumnText("Protocol", COL_NO_PROTO)) | ||||||
|  | 	pecol = createColumnText("PID", COL_NO_PID) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.pecols = append(p.pecols, pecol) | ||||||
|  | 	p.tv.AppendColumn(createColumnText("IP Address", COL_NO_DSTIP)) | ||||||
|  | 	pecol = createColumnText("Hostname", COL_NO_HOSTNAME) | ||||||
|  | 	pecol.SetMinWidth(64) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.tv.AppendColumn(createColumnText("Port", COL_NO_PORT)) | ||||||
|  | 	pecol = createColumnText("UID", COL_NO_UID) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.pecols = append(p.pecols, pecol) | ||||||
|  | 	pecol = createColumnText("GID", COL_NO_GID) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.pecols = append(p.pecols, pecol) | ||||||
|  | 	pecol = createColumnText("Origin", COL_NO_ORIGIN) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.pecols = append(p.pecols, pecol) | ||||||
|  | 	pecol = createColumnText("Timestamp", COL_NO_TIMESTAMP) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.pecols = append(p.pecols, pecol) | ||||||
|  | 
 | ||||||
|  | 	scol := createColumnText("Is SOCKS", COL_NO_IS_SOCKS) | ||||||
|  | 	scol.SetVisible(false) | ||||||
|  | 	p.tv.AppendColumn(scol) | ||||||
|  | 
 | ||||||
|  | 	pecol = createColumnText("Details", COL_NO_OPTSTRING) | ||||||
|  | 	p.tv.AppendColumn(pecol) | ||||||
|  | 	p.pecols = append(p.pecols, pecol) | ||||||
|  | 
 | ||||||
|  | 	acol := createColumnText("Scope", COL_NO_ACTION) | ||||||
|  | 	acol.SetVisible(false) | ||||||
|  | 	p.tv.AppendColumn(acol) | ||||||
|  | 
 | ||||||
|  | 	pncol := createColumnImg("", COL_NO_FILLER) | ||||||
|  | 	pncol.SetVisible(false) | ||||||
|  | 	pncol.SetSortIndicator(false) | ||||||
|  | 	p.tv.AppendColumn(pncol) | ||||||
|  | 	p.pncol = pncol | ||||||
|  | 
 | ||||||
|  | 	p.togglePECols() | ||||||
|  | 
 | ||||||
|  | 	p.ts = createTreeStore(true) | ||||||
|  | 	 | ||||||
|  | 	p.tv.SetModel(p.ts) | ||||||
|  | 	p.tv.Connect("row-activated", func() { | ||||||
|  | 		p.promptLock.Lock() | ||||||
|  | 		seldata, _, _, err := p.getSelectedRule() | ||||||
|  | 		p.promptLock.Unlock() | ||||||
|  | 		if err != nil { | ||||||
|  | 			warnDialog(&p.app.win.Window, "Unexpected error reading selected rule: " + err.Error() + "\n" + fmt.Sprintf("%+v", seldata)) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		rl := &ruleList{app: p.app} | ||||||
|  | 		target := seldata.Hostname | ||||||
|  | 		if target == "" { | ||||||
|  | 			target = seldata.Target | ||||||
|  | 		} | ||||||
|  | 		rr := &ruleRow{ rl: rl, rule: &sgfw.DbusRule{ | ||||||
|  | 			Path: seldata.Path, | ||||||
|  | 			Sandbox: seldata.Sandbox, | ||||||
|  | 			Pid: uint32(seldata.Pid), | ||||||
|  | 			UID: int32(seldata.UID), | ||||||
|  | 			GID: int32(seldata.GID), | ||||||
|  | 			Target: strings.Join([]string{target, strconv.Itoa(seldata.Port)}, ":"), | ||||||
|  | 			Proto: seldata.Proto, | ||||||
|  | 			Origin: seldata.Origin, | ||||||
|  | 			IsSocks: seldata.IsSocks, | ||||||
|  | 		}} | ||||||
|  | 		redit := newRuleAdd(rr, DIALOG_MODE_PROMPT) | ||||||
|  | 		redit.update() | ||||||
|  | 		redit.run(seldata.GUID, p.buttonAction) | ||||||
|  | 		return | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	p.app.appendConfigCallback(p.togglePECols) | ||||||
|  | 
 | ||||||
|  | 	sw.SetSizeRequest(600, 400) | ||||||
|  | 	p.createShortcuts() | ||||||
|  | 	sw.Add(p.tv) | ||||||
|  | 	return p, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) HasItems() bool { | ||||||
|  | 	return p.ts.IterNChildren(nil) > 0 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) togglePECols() { | ||||||
|  | 	v := p.app.Config.PromptExpanded | ||||||
|  | 	for _, pc := range p.pecols { | ||||||
|  | 		pc.SetVisible(v) | ||||||
|  | 	} | ||||||
|  | 	p.pncol.SetVisible(!v) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) createShortcuts() { | ||||||
|  | 	// We register here since the shortcuts are bound in an ephemeral window
 | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>a", "prompt", "Allow") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>d Escape", "prompt", "Deny") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>c", "prompt", "Cancel") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>h", "prompt", "Select the hostname/IP entry") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>p", "prompt", "Select the port entry") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>o", "prompt", "Select protocol") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>t", "prompt", "Toggle allow TLS only") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>s", "prompt", "Select scope") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>u", "prompt", "Toggle apply UID") | ||||||
|  | 	p.app.RegisterShortcutHelp("<Alt>g", "prompt", "Toggle apply GID") | ||||||
|  | 
 | ||||||
|  | 	p.app.ConnectShortcut("<Primary><Alt>space", "", "", p.app.win.Window, func (win gtk.Window) { | ||||||
|  | 		vis := p.app.tlStack.GetVisibleChildName() | ||||||
|  | 		iter, found := p.ts.GetIterFirst() | ||||||
|  | 		if iter == nil || found == false && vis != "prompt" { | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if vis != "prompt" { | ||||||
|  | 			p.app.tlStack.SetVisibleChildFull("prompt", gtk.STACK_TRANSITION_TYPE_SLIDE_LEFT_RIGHT) | ||||||
|  | 			p.app.onStackChanged() | ||||||
|  | 		} | ||||||
|  | 		pi, _ := p.ts.GetPath(iter) | ||||||
|  | 		if pi != nil { | ||||||
|  | 			p.tv.SetCursor(pi, nil, false) | ||||||
|  | 			p.tv.Emit("row-activated") | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func createColumnImg(title string, id int) *gtk.TreeViewColumn { | ||||||
|  | 	cellRenderer, err := gtk.CellRendererPixbufNew() | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal("Unable to create image cell renderer:", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	column, err := gtk.TreeViewColumnNewWithAttribute(title, cellRenderer, "pixbuf", id) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal("Unable to create cell column:", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return column | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func createColumnText(title string, id int) *gtk.TreeViewColumn { | ||||||
|  | 	cellRenderer, err := gtk.CellRendererTextNew() | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal("Unable to create text cell renderer:", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	column, err := gtk.TreeViewColumnNewWithAttribute(title, cellRenderer, "text", id) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal("Unable to create cell column:", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	column.SetSortColumnID(id) | ||||||
|  | 	column.SetResizable(true) | ||||||
|  | 	return column | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func createTreeStore(general bool) *gtk.TreeStore { | ||||||
|  | 	colData := []glib.Type{glib.TYPE_INT, glib.TYPE_OBJECT, glib.TYPE_STRING, glib.TYPE_STRING, glib.TYPE_STRING, glib.TYPE_STRING, glib.TYPE_STRING, glib.TYPE_INT, glib.TYPE_STRING, | ||||||
|  | 		glib.TYPE_STRING, glib.TYPE_INT, glib.TYPE_INT, glib.TYPE_INT, glib.TYPE_STRING, glib.TYPE_STRING, glib.TYPE_INT, glib.TYPE_STRING, glib.TYPE_INT, glib.TYPE_OBJECT} | ||||||
|  | 
 | ||||||
|  | 	treeStore, err := gtk.TreeStoreNew(colData...) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal("Unable to create list store:", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return treeStore | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) addRequestInc(guid, path, icon, proto string, pid int, ipaddr, hostname string, port, uid, gid int, | ||||||
|  | 	origin, timestamp string, is_socks bool, optstring string, sandbox string, action int) bool { | ||||||
|  | 	duplicated := false | ||||||
|  | 
 | ||||||
|  | 	p.promptLock.Lock() | ||||||
|  | 	defer p.promptLock.Unlock() | ||||||
|  | 	for ridx := 0; ridx < p.ts.IterNChildren(nil); ridx++ { | ||||||
|  | 		rule, iter, err := p.getRuleByIdx(ridx, -1) | ||||||
|  | 		if err != nil { | ||||||
|  | 			break | ||||||
|  | 			// XXX: not compared: optstring/sandbox
 | ||||||
|  | 		} else if (rule.Path == path) && (rule.Proto == proto) && (rule.Pid == pid) && (rule.Target == ipaddr) && (rule.Hostname == hostname) && | ||||||
|  | 			(rule.Port == port) && (rule.UID == uid) && (rule.GID == gid) && (rule.Origin == origin) && (rule.IsSocks == is_socks) { | ||||||
|  | 			rule.nrefs++ | ||||||
|  | 
 | ||||||
|  | 			err := p.ts.SetValue(iter, 0, rule.nrefs) | ||||||
|  | 			if err != nil { | ||||||
|  | 				fmt.Println("Error creating duplicate firewall prompt entry:", err) | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			duplicated = true | ||||||
|  | 			subiter := p.ts.Append(iter) | ||||||
|  | 			p.storeNewEntry(subiter, guid, path, sandbox, icon, proto, pid, ipaddr, hostname, port, uid, gid, origin, timestamp, is_socks, optstring, action) | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return duplicated | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) AddRequest(guid, path, icon, proto string, pid int, ipaddr, hostname string, port, uid, gid int, | ||||||
|  | 	origin, timestamp string, is_socks bool, optstring string, sandbox string, action int) bool { | ||||||
|  | 	if p.ts == nil { | ||||||
|  | 		waitTimes := []int{1, 2, 5, 10} | ||||||
|  | 
 | ||||||
|  | 		if p.ts == nil { | ||||||
|  | 			fmt.Println("SGFW prompter was not ready to receive firewall request... waiting") | ||||||
|  | 
 | ||||||
|  | 			for _, wtime := range waitTimes { | ||||||
|  | 				time.Sleep(time.Duration(wtime) * time.Second) | ||||||
|  | 
 | ||||||
|  | 				if p.ts != nil { | ||||||
|  | 					break | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				fmt.Println("SGFW prompter is still waiting...") | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if p.ts == nil { | ||||||
|  | 		log.Fatal("SGFW prompter GUI failed to load for unknown reasons") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if p.addRequestInc(guid, path, icon, proto, pid, ipaddr, hostname, port, uid, gid, origin, timestamp, is_socks, optstring, sandbox, action) { | ||||||
|  | 		fmt.Println("Request was duplicate: ", guid) | ||||||
|  | 		p.promptLock.Lock() | ||||||
|  | 		p.toggleHover() | ||||||
|  | 		p.promptLock.Unlock() | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.promptLock.Lock() | ||||||
|  | 	defer p.promptLock.Unlock() | ||||||
|  | 
 | ||||||
|  | 	iter := p.ts.Append(nil) | ||||||
|  | 	p.storeNewEntry(iter, guid, path, sandbox, icon, proto, pid, ipaddr, hostname, port, uid, gid, origin, timestamp, is_socks, optstring, action) | ||||||
|  | 
 | ||||||
|  | 	p.toggleHover() | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Needs to be locked by caller
 | ||||||
|  | func (p *Prompt)storeNewEntry(iter *gtk.TreeIter, guid, path, sandbox, icon, proto string, pid int, ipaddr, hostname string, port, uid, gid int, origin, | ||||||
|  | 	timestamp string, is_socks bool, optstring string, action int) { | ||||||
|  | 	var colVals = [COL_NO_LAST]interface{}{} | ||||||
|  | 
 | ||||||
|  | 	if is_socks { | ||||||
|  | 		if (optstring != "") && (strings.Index(optstring, "SOCKS") == -1) { | ||||||
|  | 			optstring = "SOCKS5 / " + optstring | ||||||
|  | 		} else if optstring == "" { | ||||||
|  | 			optstring = "SOCKS5" | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	colVals[COL_NO_NREFS] = 1 | ||||||
|  | 	colVals[COL_NO_ICON_PIXBUF] = nil | ||||||
|  | 	colVals[COL_NO_GUID] = guid | ||||||
|  | 	colVals[COL_NO_PATH] = path | ||||||
|  | 	colVals[COL_NO_SANDBOX] = sandbox | ||||||
|  | 	colVals[COL_NO_ICON] = icon | ||||||
|  | 	colVals[COL_NO_PROTO] = proto | ||||||
|  | 	colVals[COL_NO_PID] = pid | ||||||
|  | 
 | ||||||
|  | 	if ipaddr == "" { | ||||||
|  | 		colVals[COL_NO_DSTIP] = "---" | ||||||
|  | 	} else { | ||||||
|  | 		colVals[COL_NO_DSTIP] = ipaddr | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	colVals[COL_NO_HOSTNAME] = hostname | ||||||
|  | 	colVals[COL_NO_PORT] = port | ||||||
|  | 	colVals[COL_NO_UID] = uid | ||||||
|  | 	colVals[COL_NO_GID] = gid | ||||||
|  | 	colVals[COL_NO_ORIGIN] = origin | ||||||
|  | 	colVals[COL_NO_TIMESTAMP] = timestamp | ||||||
|  | 	colVals[COL_NO_IS_SOCKS] = 0 | ||||||
|  | 
 | ||||||
|  | 	if is_socks { | ||||||
|  | 		colVals[COL_NO_IS_SOCKS] = 1 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	colVals[COL_NO_OPTSTRING] = optstring | ||||||
|  | 	colVals[COL_NO_ACTION] = action | ||||||
|  | 	colVals[COL_NO_FILLER] = nil | ||||||
|  | 
 | ||||||
|  | 	itheme, err := gtk.IconThemeGetDefault() | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatal("Could not load default icon theme:", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	in := []string{spath.Base(path)} | ||||||
|  | 	if sandbox != "" { | ||||||
|  | 		in = append([]string{sandbox}, in...) | ||||||
|  | 	} | ||||||
|  | 	if icon != "" { | ||||||
|  | 		in = append(in, icon) | ||||||
|  | 	} | ||||||
|  | 	in = append(in, "terminal") | ||||||
|  | 	if path == "[unknown]" { | ||||||
|  | 		in = []string{"image-missing"} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for _, ia := range in { | ||||||
|  | 		pb, _ := itheme.LoadIcon(ia, int(gtk.ICON_SIZE_BUTTON), gtk.ICON_LOOKUP_GENERIC_FALLBACK) | ||||||
|  | 		if pb != nil { | ||||||
|  | 			colVals[COL_NO_ICON_PIXBUF] = pb | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pb, err := gdk.PixbufNew(gdk.COLORSPACE_RGB, true, 8, 24, 24) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Println("Error creating blank icon:", err) | ||||||
|  | 	} else { | ||||||
|  | 		colVals[COL_NO_FILLER] = pb | ||||||
|  | 
 | ||||||
|  | 		img, err := gtk.ImageNewFromPixbuf(pb) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Println("Error creating image from pixbuf:", err) | ||||||
|  | 		} else { | ||||||
|  | 			img.Clear() | ||||||
|  | 			pb = img.GetPixbuf() | ||||||
|  | 			colVals[COL_NO_FILLER] = pb | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for n := 0; n < len(colVals); n++ { | ||||||
|  | 		err := p.ts.SetValue(iter, n, colVals[n]) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Fatal("Unable to add row:", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) getRuleByIdx(idx, subidx int) (ruleColumns, *gtk.TreeIter, error) { | ||||||
|  | 	rule := ruleColumns{} | ||||||
|  | 	tpath := fmt.Sprintf("%d", idx) | ||||||
|  | 
 | ||||||
|  | 	if subidx != -1 { | ||||||
|  | 		tpath = fmt.Sprintf("%d:%d", idx, subidx) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	path, err := gtk.TreePathNewFromString(tpath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	iter, err := p.ts.GetIter(path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.nrefs, err = p.lsGetInt(iter, COL_NO_NREFS) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.GUID, err = p.lsGetStr(iter, COL_NO_GUID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Path, err = p.lsGetStr(iter, COL_NO_PATH) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Sandbox, err = p.lsGetStr(iter, COL_NO_SANDBOX) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Icon, err = p.lsGetStr(iter, COL_NO_ICON) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Proto, err = p.lsGetStr(iter, COL_NO_PROTO) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Pid, err = p.lsGetInt(iter, COL_NO_PID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Target, err = p.lsGetStr(iter, COL_NO_DSTIP) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Hostname, err = p.lsGetStr(iter, COL_NO_HOSTNAME) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Port, err = p.lsGetInt(iter, COL_NO_PORT) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.UID, err = p.lsGetInt(iter, COL_NO_UID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.GID, err = p.lsGetInt(iter, COL_NO_GID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Origin, err = p.lsGetStr(iter, COL_NO_ORIGIN) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Timestamp, err = p.lsGetStr(iter, COL_NO_TIMESTAMP) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.IsSocks = false | ||||||
|  | 	is_socks, err := p.lsGetInt(iter, COL_NO_IS_SOCKS) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if is_socks != 0 { | ||||||
|  | 		rule.IsSocks = true | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rule.Scope, err = p.lsGetInt(iter, COL_NO_ACTION) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return rule, iter, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) lsGetInt(iter *gtk.TreeIter, idx int) (int, error) { | ||||||
|  | 	val, err := p.ts.GetValue(iter, idx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return 0, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	ival, err := val.GoValue() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return 0, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return ival.(int), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) lsGetStr(iter *gtk.TreeIter, idx int) (string, error) { | ||||||
|  | 	val, err := p.ts.GetValue(iter, idx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	sval, err := val.GetString() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return "", err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return sval, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) toggleHover() { | ||||||
|  | 	nitems := p.ts.IterNChildren(nil) | ||||||
|  | 	stack := p.app.tlStack.GetChildByName("prompt") | ||||||
|  | 
 | ||||||
|  | 	if nitems > 0 { | ||||||
|  | 		if p.app.Settings.GetToplevelPrompt() { | ||||||
|  | 			//p.win.SetModal(true)
 | ||||||
|  | 			p.app.win.Deiconify() | ||||||
|  | 			p.app.win.SetKeepAbove(true) | ||||||
|  | 			p.app.win.Stick() | ||||||
|  | 		} | ||||||
|  | 		p.app.win.SetUrgencyHint(true) | ||||||
|  | 		p.app.win.Present() | ||||||
|  | 		if p.app.tlStack.GetVisibleChildName() != "prompt" { | ||||||
|  | 			err := p.app.tlStack.ChildSetProperty(stack, "needs-attention", true) | ||||||
|  | 			if err != nil { | ||||||
|  | 				fmt.Println("Error setting stack attention") | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		//p.win.SetModal(false)
 | ||||||
|  | 		p.app.win.SetUrgencyHint(false) | ||||||
|  | 		if p.app.Settings.GetToplevelPrompt() { | ||||||
|  | 			p.app.win.SetKeepAbove(false) | ||||||
|  | 			p.app.win.Unstick() | ||||||
|  | 		} | ||||||
|  | 		p.app.tlStack.ChildSetProperty(stack, "needs-attention", false) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Needs to be locked by the caller
 | ||||||
|  | func (p *Prompt) getSelectedRule() (ruleColumns, int, int, error) { | ||||||
|  | 	rule := ruleColumns{} | ||||||
|  | 
 | ||||||
|  | 	sel, err := p.tv.GetSelection() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, -1, -1, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rows := sel.GetSelectedRows(p.ts) | ||||||
|  | 
 | ||||||
|  | 	if rows.Length() <= 0 { | ||||||
|  | 		return rule, -1, -1, errors.New("no selection was made") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rdata := rows.NthData(0) | ||||||
|  | 	tpath := rdata.(*gtk.TreePath).String() | ||||||
|  | 
 | ||||||
|  | 	subidx := -1 | ||||||
|  | 	ptoks := strings.Split(tpath, ":") | ||||||
|  | 
 | ||||||
|  | 	if len(ptoks) > 2 { | ||||||
|  | 		return rule, -1, -1, errors.New("internal error parsing selected item tree path") | ||||||
|  | 	} else if len(ptoks) == 2 { | ||||||
|  | 		subidx, err = strconv.Atoi(ptoks[1]) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return rule, -1, -1, err | ||||||
|  | 		} | ||||||
|  | 		tpath = ptoks[0] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	lIndex, err := strconv.Atoi(tpath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, -1, -1, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	//	fmt.Printf("lindex = %d : %d\n", lIndex, subidx)
 | ||||||
|  | 	rule, _, err = p.getRuleByIdx(lIndex, subidx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return rule, -1, -1, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return rule, lIndex, subidx, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Needs to be locked by the caller
 | ||||||
|  | func (p *Prompt) numSelections() int { | ||||||
|  | 	sel, err := p.tv.GetSelection() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return -1 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rows := sel.GetSelectedRows(p.ts) | ||||||
|  | 	return int(rows.Length()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) removeSelectedRule(idx, subidx int) error { | ||||||
|  | 	fmt.Printf("XXX: attempting to remove idx = %v, %v\n", idx, subidx) | ||||||
|  | 	ppathstr := fmt.Sprintf("%d", idx) | ||||||
|  | 	pathstr := ppathstr | ||||||
|  | 
 | ||||||
|  | 	if subidx > -1 { | ||||||
|  | 		pathstr = fmt.Sprintf("%d:%d", idx, subidx) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	iter, err := p.ts.GetIterFromString(pathstr) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	nchildren := p.ts.IterNChildren(iter) | ||||||
|  | 
 | ||||||
|  | 	if nchildren >= 1 { | ||||||
|  | 		firstpath := fmt.Sprintf("%d:0", idx) | ||||||
|  | 		citer, err := p.ts.GetIterFromString(firstpath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		gnrefs, err := p.ts.GetValue(iter, COL_NO_NREFS) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		vnrefs, err := gnrefs.GoValue() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		nrefs := vnrefs.(int) - 1 | ||||||
|  | 
 | ||||||
|  | 		for n := 0; n < COL_NO_LAST; n++ { | ||||||
|  | 			val, err := p.ts.GetValue(citer, n) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if n == COL_NO_NREFS { | ||||||
|  | 				err = p.ts.SetValue(iter, n, nrefs) | ||||||
|  | 			} else { | ||||||
|  | 				err = p.ts.SetValue(iter, n, val) | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		p.ts.Remove(citer) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.ts.Remove(iter) | ||||||
|  | 
 | ||||||
|  | 	if subidx > -1 { | ||||||
|  | 		ppath, err := gtk.TreePathNewFromString(ppathstr) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		piter, err := p.ts.GetIter(ppath) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		nrefs, err := p.lsGetInt(piter, COL_NO_NREFS) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		err = p.ts.SetValue(piter, COL_NO_NREFS, nrefs-1) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.toggleHover() | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) addRecentlyRemoved(guid string) { | ||||||
|  | 	p.recentLock.Lock() | ||||||
|  | 	defer p.recentLock.Unlock() | ||||||
|  | 	fmt.Println("RECENTLY REMOVED: ", guid) | ||||||
|  | 	p.recentlyRemoved = append(p.recentlyRemoved, guid) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) wasRecentlyRemoved(guid string) bool { | ||||||
|  | 	p.recentLock.Lock() | ||||||
|  | 	defer p.recentLock.Unlock() | ||||||
|  | 
 | ||||||
|  | 	for gind, g := range p.recentlyRemoved { | ||||||
|  | 		if g == guid { | ||||||
|  | 			p.recentlyRemoved = append(p.recentlyRemoved[:gind], p.recentlyRemoved[gind+1:]...) | ||||||
|  | 			return true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) RemoveRequest(guid string) { | ||||||
|  | 	if p.wasRecentlyRemoved(guid) { | ||||||
|  | 		fmt.Printf("Entry for %s was recently removed; deleting from cache\n", guid) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	removed := false | ||||||
|  | 
 | ||||||
|  | 	if p.ts == nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	p.promptLock.Lock() | ||||||
|  | 	defer p.promptLock.Unlock() | ||||||
|  | 
 | ||||||
|  | remove_outer: | ||||||
|  | 	for ridx := 0; ridx < p.ts.IterNChildren(nil); ridx++ { | ||||||
|  | 		nchildren := 0 | ||||||
|  | 		this_iter, err := p.ts.GetIterFromString(fmt.Sprintf("%d", ridx)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Println("Strange condition; couldn't get iter of known tree index:", err) | ||||||
|  | 		} else { | ||||||
|  | 			nchildren = p.ts.IterNChildren(this_iter) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		for cidx := 0; cidx < nchildren-1; cidx++ { | ||||||
|  | 			sidx := cidx | ||||||
|  | 			if cidx == nchildren { | ||||||
|  | 				cidx = -1 | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			rule, _, err := p.getRuleByIdx(ridx, sidx) | ||||||
|  | 			if err != nil { | ||||||
|  | 				break remove_outer | ||||||
|  | 			} else if rule.GUID == guid { | ||||||
|  | 				p.removeSelectedRule(ridx, sidx) | ||||||
|  | 				removed = true | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if !removed { | ||||||
|  | 		fmt.Printf("Unexpected condition: SGFW requested prompt removal for non-existent GUID %v\n", guid) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) RemoveAll() { | ||||||
|  | 	p.promptLock.Lock() | ||||||
|  | 	defer p.promptLock.Unlock() | ||||||
|  | 	p.recentLock.Lock() | ||||||
|  | 	defer p.recentLock.Unlock() | ||||||
|  | 
 | ||||||
|  | 	p.recentlyRemoved = p.recentlyRemoved[:0] | ||||||
|  | 
 | ||||||
|  | 	for { | ||||||
|  | 		iter, found := p.ts.GetIterFirst() | ||||||
|  | 		if iter == nil || found == false { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		pi, _ := p.ts.GetPath(iter) | ||||||
|  | 		if pi == nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		p.tv.SetCursor(pi, nil, false) | ||||||
|  | 		_, idx, subidx, err := p.getSelectedRule() | ||||||
|  | 		if err != nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		p.removeSelectedRule(idx, subidx) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) makeDecision(rule string, scope int, guid string) error { | ||||||
|  | 	return p.app.Dbus.answerPrompt(uint32(scope), rule, guid) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (p *Prompt) buttonAction(guid string, rr *sgfw.DbusRule) { | ||||||
|  | 	p.promptLock.Lock() | ||||||
|  | 	rule, idx, subidx, err := p.getSelectedRule() | ||||||
|  | 	if err != nil { | ||||||
|  | 		p.promptLock.Unlock() | ||||||
|  | 		warnDialog(&p.app.win.Window, "Error occurred processing request: %s", err.Error()) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	tk := strings.Split(rr.Target, ":") | ||||||
|  | 	// Overlay the rules
 | ||||||
|  | 	rule.Scope = int(rr.Mode) | ||||||
|  | 	//rule.Path = urule.Path
 | ||||||
|  | 	rule.Port, _ = strconv.Atoi(tk[1]) | ||||||
|  | 	rule.Target = tk[0] | ||||||
|  | 	rule.Proto = rr.Proto | ||||||
|  | 	rule.UID = int(rr.UID) | ||||||
|  | 	rule.GID = int(rr.GID) | ||||||
|  | 	// rule.Uname = urule.Uname
 | ||||||
|  | 	// rule.Gname = urule.Gname
 | ||||||
|  | 
 | ||||||
|  | 	fmt.Println("rule = ", rule) | ||||||
|  | 
 | ||||||
|  | 	action := sgfw.RuleActionString[sgfw.RuleAction(rr.Verb)] | ||||||
|  | 	rulestr := action | ||||||
|  | 
 | ||||||
|  | 	proto := rule.Proto | ||||||
|  | 	if proto == "any" || proto == "" { | ||||||
|  | 		proto = "*" | ||||||
|  | 	} | ||||||
|  | 	rulestr += "|" + proto + ":" + rule.Target + ":" + strconv.Itoa(rule.Port) | ||||||
|  | 	rulestr += "|" + sgfw.RuleModeString[sgfw.RuleMode(rule.Scope)] | ||||||
|  | 	rulestr += "|" + strconv.Itoa(rule.UID) + ":" + strconv.Itoa(rule.GID) | ||||||
|  | 	if rule.Sandbox != "" { | ||||||
|  | 		rulestr += "|" + rule.Sandbox | ||||||
|  | 	} | ||||||
|  | 	fmt.Println("RULESTR = ", rulestr) | ||||||
|  | 	p.makeDecision(rulestr, int(rule.Scope), guid) | ||||||
|  | 	err = p.removeSelectedRule(idx, subidx) | ||||||
|  | 	p.addRecentlyRemoved(guid) | ||||||
|  | 	p.promptLock.Unlock() | ||||||
|  | 	if err != nil { | ||||||
|  | 		warnDialog(&p.app.win.Window, "Error setting new rule: %s", err.Error()) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -0,0 +1,551 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"strings" | ||||||
|  | 	"strconv" | ||||||
|  | 	"unicode" | ||||||
|  | 
 | ||||||
|  | 	"github.com/subgraph/fw-daemon/sgfw" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gotk3/gotk3/gtk" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	newDialogCancel = 1 | ||||||
|  | 	newDialogOk     = 2 | ||||||
|  | 	newDialogAllow  = 3 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	COLUMN_ID = iota | ||||||
|  | 	COLUMN_NAME | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type DialogMode uint | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	DIALOG_MODE_NEW DialogMode = iota | ||||||
|  | 	DIALOG_MODE_EDIT | ||||||
|  | 	DIALOG_MODE_SAVEAS | ||||||
|  | 	DIALOG_MODE_PROMPT | ||||||
|  | 	DIALOG_MODE_INFO | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	Setuid uint32 = 1 << (12 - 1 - iota) | ||||||
|  | 	Setgid | ||||||
|  | 	Sticky | ||||||
|  | 	UserRead | ||||||
|  | 	UserWrite | ||||||
|  | 	UserExecute | ||||||
|  | 	GroupRead | ||||||
|  | 	GroupWrite | ||||||
|  | 	GroupExecute | ||||||
|  | 	OtherRead | ||||||
|  | 	OtherWrite | ||||||
|  | 	OtherExecute | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type ruleNew struct { | ||||||
|  | 	dialog     *gtk.Dialog | ||||||
|  | 	row        *ruleRow | ||||||
|  | 	mode       DialogMode | ||||||
|  | 	nbSelected int | ||||||
|  | 	comboUID   *gtk.ComboBoxText | ||||||
|  | 	checkUID   *gtk.CheckButton | ||||||
|  | 	comboGID   *gtk.ComboBoxText | ||||||
|  | 	checkGID   *gtk.CheckButton | ||||||
|  | 	titleScope *gtk.Label | ||||||
|  | 	comboScope *gtk.ComboBoxText | ||||||
|  | 	labelScope *gtk.Label | ||||||
|  | 	comboVerb  *gtk.ComboBoxText | ||||||
|  | 	checkTLS   *gtk.CheckButton | ||||||
|  | 	titleSandbox *gtk.Label | ||||||
|  | 	labelSandbox *gtk.Label | ||||||
|  | 	comboSandbox *gtk.ComboBoxText | ||||||
|  | 	btnPathChooser *gtk.FileChooserButton | ||||||
|  | 	entryPath *gtk.Entry | ||||||
|  | 	hostEntry    *gtk.Entry | ||||||
|  | 	portEntry    *gtk.Entry | ||||||
|  | 	titlePort *gtk.Label | ||||||
|  | 	comboProto *gtk.ComboBoxText | ||||||
|  | 	ok *gtk.Button | ||||||
|  | 	allow *gtk.Button | ||||||
|  | 	cancel *gtk.Button | ||||||
|  | 	labelPID *gtk.Label | ||||||
|  | 	titlePID *gtk.Label | ||||||
|  | 	entryOrigin *gtk.Entry | ||||||
|  | 	labelOrigin *gtk.Label | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func newRuleAdd(rr *ruleRow, mode DialogMode) *ruleNew{ | ||||||
|  | 	rnew := &ruleNew{} | ||||||
|  | 	rnew.mode = mode | ||||||
|  | 	rnew.nbSelected = rr.rl.app.nbRules.GetCurrentPage() | ||||||
|  | 	b := newBuilder("RuleNew") | ||||||
|  | 	b.getItems( | ||||||
|  | 		"dialog", &rnew.dialog, | ||||||
|  | 		"uid_combo", &rnew.comboUID, | ||||||
|  | 		"uid_checkbox", &rnew.checkUID, | ||||||
|  | 		"gid_combo", &rnew.comboGID, | ||||||
|  | 		"gid_checkbox", &rnew.checkGID, | ||||||
|  | 		"scope_title", &rnew.titleScope, | ||||||
|  | 		"scope_combo", &rnew.comboScope, | ||||||
|  | 		"scope_label", &rnew.labelScope, | ||||||
|  | 		"verb_combo", &rnew.comboVerb, | ||||||
|  | 		"tls_check", &rnew.checkTLS, | ||||||
|  | 		"sandbox_title", &rnew.titleSandbox, | ||||||
|  | 		"sandbox_combo", &rnew.comboSandbox, | ||||||
|  | 		"sandbox_label", &rnew.labelSandbox, | ||||||
|  | 		"path_chooser", &rnew.btnPathChooser, | ||||||
|  | 		"path_entry", &rnew.entryPath, | ||||||
|  | 		"host_entry", &rnew.hostEntry, | ||||||
|  | 		"port_entry", &rnew.portEntry, | ||||||
|  | 		"port_title", &rnew.titlePort, | ||||||
|  | 		"proto_combo", &rnew.comboProto, | ||||||
|  | 		"ok_button", &rnew.ok, | ||||||
|  | 		"allow_button", &rnew.allow, | ||||||
|  | 		"cancel_button", &rnew.cancel, | ||||||
|  | 		"pid_label", &rnew.labelPID, | ||||||
|  | 		"pid_title", &rnew.titlePID, | ||||||
|  | 		"origin_entry", &rnew.entryOrigin, | ||||||
|  | 		"origin_label", &rnew.labelOrigin, | ||||||
|  | 	) | ||||||
|  | 
 | ||||||
|  | 	b.ConnectSignals(map[string]interface{}{ | ||||||
|  | 		"on_proto_changed":    rnew.onProtoChanged, | ||||||
|  | 		"on_verb_changed":     rnew.onVerbChanged, | ||||||
|  | 		"on_port_insert_text": rnew.onPortInsertText, | ||||||
|  | 		"on_port_changed":     rnew.onChanged, | ||||||
|  | 		"on_host_changed":     rnew.onChanged, | ||||||
|  | 		"on_path_changed":     rnew.onChanged, | ||||||
|  | 		"on_path_set":         rnew.onPathSet, | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	rnew.row = rr | ||||||
|  | 	switch rnew.mode { | ||||||
|  | 	case DIALOG_MODE_EDIT: | ||||||
|  | 		rnew.dialog.SetTitle("Edit Rule") | ||||||
|  | 	case DIALOG_MODE_NEW: | ||||||
|  | 		rnew.dialog.SetTitle("Add New Rule") | ||||||
|  | 	case DIALOG_MODE_SAVEAS: | ||||||
|  | 		rnew.ok.SetLabel("Save As New") | ||||||
|  | 		rnew.dialog.SetTitle("Save As New Rule") | ||||||
|  | 	case DIALOG_MODE_PROMPT: | ||||||
|  | 		rnew.connectShortcutsPromptWindow() | ||||||
|  | 		rnew.dialog.SetTitle("Firewall Prompt") | ||||||
|  | 	case DIALOG_MODE_INFO: | ||||||
|  | 		rnew.cancel.SetLabel("Close") | ||||||
|  | 		rnew.dialog.SetTitle("Rule Information") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return rnew | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) connectShortcutsPromptWindow() { | ||||||
|  | 	app := re.row.rl.app | ||||||
|  | 	// Shortcuts Help Registered in Prompt
 | ||||||
|  | 	app.ConnectShortcut("<Alt>h", "", "", re.dialog.Window, func(win gtk.Window) {re.hostEntry.Widget.GrabFocus()}) | ||||||
|  | 	app.ConnectShortcut("<Alt>p", "", "", re.dialog.Window, func(win gtk.Window) {re.portEntry.Widget.GrabFocus()}) | ||||||
|  | 	app.ConnectShortcut("<Alt>o", "", "", re.dialog.Window, func(win gtk.Window) {re.comboProto.ComboBox.Popup()}) | ||||||
|  | 	app.ConnectShortcut("<Alt>t", "", "", re.dialog.Window, func(win gtk.Window) { | ||||||
|  | 		if re.checkTLS.GetSensitive() { | ||||||
|  | 			re.checkTLS.SetActive(!re.checkTLS.GetActive()) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	app.ConnectShortcut("<Alt>s", "", "", re.dialog.Window, func(win gtk.Window) {re.comboScope.ComboBox.Popup()}) | ||||||
|  | 	app.ConnectShortcut("<Alt>u", "", "", re.dialog.Window, func(win gtk.Window) {re.checkUID.SetActive(!re.checkUID.GetActive())}) | ||||||
|  | 	app.ConnectShortcut("<Alt>g", "", "", re.dialog.Window, func(win gtk.Window) {re.checkGID.SetActive(!re.checkGID.GetActive())}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) updateRow(res int) { | ||||||
|  | 	if !re.validateFields() { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	r := re.row.rule | ||||||
|  | 	if re.mode == DIALOG_MODE_PROMPT { | ||||||
|  | 		if res == newDialogOk { | ||||||
|  | 			r.Verb = uint16(sgfw.RULE_ACTION_DENY) | ||||||
|  | 		} else if res == newDialogAllow { | ||||||
|  | 			r.Verb = uint16(sgfw.RULE_ACTION_ALLOW) | ||||||
|  | 		} | ||||||
|  | 		mid, _ := strconv.Atoi(re.comboScope.GetActiveID()) | ||||||
|  | 		r.Mode = uint16(mid) | ||||||
|  | 	} else { | ||||||
|  | 		switch re.comboVerb.GetActiveID() { | ||||||
|  | 		case "allow": | ||||||
|  | 			r.Verb = uint16(sgfw.RULE_ACTION_ALLOW) | ||||||
|  | 	//	case "allow_tls":
 | ||||||
|  | 	//		r.Verb = uint16(sgfw.RULE_ACTION_ALLOW_TLSONLY)
 | ||||||
|  | 		case "deny": | ||||||
|  | 			r.Verb = uint16(sgfw.RULE_ACTION_DENY) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	r.Proto = re.comboProto.GetActiveID() | ||||||
|  | 	if r.Proto == "any" { | ||||||
|  | 		r.Proto = "*" | ||||||
|  | 	} | ||||||
|  | 	if r.Proto == "tcp" && r.Verb == uint16(sgfw.RULE_ACTION_ALLOW) && re.checkTLS.GetActive() { | ||||||
|  | 		r.Verb = uint16(sgfw.RULE_ACTION_ALLOW_TLSONLY) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	host, _ := re.hostEntry.GetText() | ||||||
|  | 	port, _ := re.portEntry.GetText() | ||||||
|  | 	r.Target = fmt.Sprintf("%s:%s", host, port) | ||||||
|  | 	if re.mode != DIALOG_MODE_PROMPT || re.checkUID.GetActive() == true { | ||||||
|  | 		uid, _ := strconv.ParseInt(re.comboUID.GetActiveID(), 10, 32) | ||||||
|  | 		r.UID = int32(uid) | ||||||
|  | 	} else { | ||||||
|  | 		r.UID = -1 | ||||||
|  | 	} | ||||||
|  | 	if re.mode != DIALOG_MODE_PROMPT || re.checkGID.GetActive() == true { | ||||||
|  | 		gid, _ := strconv.ParseInt(re.comboGID.GetActiveID(), 10, 32) | ||||||
|  | 		r.GID = int32(gid) | ||||||
|  | 	} else { | ||||||
|  | 		r.GID = -1 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if re.mode == DIALOG_MODE_NEW { | ||||||
|  | 		r.Path = re.btnPathChooser.FileChooser.GetFilename() | ||||||
|  | 		mid, _ := strconv.Atoi(re.comboScope.GetActiveID()) | ||||||
|  | 		r.Mode = uint16(mid) | ||||||
|  | 		r.Sandbox = re.comboSandbox.GetActiveID() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if re.mode != DIALOG_MODE_NEW && re.mode != DIALOG_MODE_PROMPT { | ||||||
|  | 		re.row.update() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type cbPromptRequest func(guid string, rule *sgfw.DbusRule) | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) run(guid string, cb cbPromptRequest) { | ||||||
|  | 	re.dialog.SetTransientFor(re.row.rl.app.win) | ||||||
|  | 	re.dialog.ShowAll() | ||||||
|  | 	if re.mode == DIALOG_MODE_INFO { | ||||||
|  | 		re.dialog.Run() | ||||||
|  | 	} else if re.mode == DIALOG_MODE_PROMPT { | ||||||
|  | 		res := re.dialog.Run() | ||||||
|  | 		if res != newDialogCancel { | ||||||
|  | 			re.updateRow(res) | ||||||
|  | 			cb(guid, re.row.rule) | ||||||
|  | 		} | ||||||
|  | 	} else if re.mode == DIALOG_MODE_NEW { | ||||||
|  | 		if re.dialog.Run() == newDialogOk { | ||||||
|  | 			re.updateRow(newDialogOk) | ||||||
|  | 			r := *re.row.rule | ||||||
|  | 			res, err := re.row.rl.app.Dbus.addRule(&r) | ||||||
|  | 			if res == false || err != nil { | ||||||
|  | 				warnDialog(&re.row.rl.app.win.Window, "Error notifying SGFW of asynchronous rule addition:", err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else if re.mode == DIALOG_MODE_SAVEAS { | ||||||
|  | 		if re.dialog.Run() == newDialogOk { | ||||||
|  | 			re.updateRow(newDialogOk) | ||||||
|  | 			r := *re.row.rule | ||||||
|  | 			re.row.rl.app.Dbus.addRule(&r) | ||||||
|  | 			re.row.rl.remove(re.row) | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		if re.dialog.Run() == newDialogOk { | ||||||
|  | 			re.updateRow(newDialogOk) | ||||||
|  | 			re.row.rl.app.Dbus.updateRule(re.row.rule) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	re.dialog.Destroy() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (rr *ruleRow) runNewEditor(mode DialogMode) { | ||||||
|  | 	redit := newRuleAdd(rr, mode) | ||||||
|  | 	redit.update() | ||||||
|  | 	redit.run("", nil) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) update() { | ||||||
|  | 	re.populateUID() | ||||||
|  | 	re.populateGID() | ||||||
|  | 	r := re.row.rule | ||||||
|  | 
 | ||||||
|  | 	if re.mode != DIALOG_MODE_INFO { | ||||||
|  | 		re.comboScope.Remove(4) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if re.mode != DIALOG_MODE_PROMPT && re.mode != DIALOG_MODE_INFO { | ||||||
|  | 		re.comboScope.Remove(3) | ||||||
|  | 		re.comboScope.Remove(2) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	re.onVerbChanged() | ||||||
|  | 
 | ||||||
|  | 	if re.mode == DIALOG_MODE_NEW { | ||||||
|  | 		if re.nbSelected < 2 { | ||||||
|  | 			re.comboScope.SetActive(re.nbSelected) | ||||||
|  | 		} else { | ||||||
|  | 			re.comboScope.SetActive(0) | ||||||
|  | 		} | ||||||
|  | 		//re.titleSandbox.SetNoShowAll(true)
 | ||||||
|  | 		//re.titleSandbox.SetVisible(false)
 | ||||||
|  | 		//re.comboSandbox.SetNoShowAll(true)
 | ||||||
|  | 		//re.comboSandbox.SetVisible(false)
 | ||||||
|  | 		//re.comboSandbox.SetNoShowAll(true)
 | ||||||
|  | 		//re.comboSandbox.SetVisible(false)
 | ||||||
|  | 		re.comboSandbox.Append("", "") | ||||||
|  | 		for _, pn := range re.row.rl.app.ozProfiles { | ||||||
|  | 			re.comboSandbox.Append(pn, pn) | ||||||
|  | 		} | ||||||
|  | 		re.comboSandbox.SetActive(0) | ||||||
|  | 		re.btnPathChooser.SetCurrentFolder("/") | ||||||
|  | 		re.ok.SetSensitive(false) | ||||||
|  | 		re.onProtoChanged() | ||||||
|  | 
 | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if r.Proto == "" { | ||||||
|  | 		re.comboProto.SetActiveID("any") | ||||||
|  | 	} else { | ||||||
|  | 		re.comboProto.SetActiveID(strings.ToLower(r.Proto)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	re.comboSandbox.SetVisible(false) | ||||||
|  | 	re.comboSandbox.SetSensitive(false) | ||||||
|  | 	re.comboSandbox.SetNoShowAll(true) | ||||||
|  | 
 | ||||||
|  | 	if sgfw.RuleAction(r.Verb) == sgfw.RULE_ACTION_ALLOW || sgfw.RuleAction(r.Verb) == sgfw.RULE_ACTION_ALLOW_TLSONLY { | ||||||
|  | 		re.comboVerb.SetActiveID("allow") | ||||||
|  | 	} else { | ||||||
|  | 		re.comboVerb.SetActiveID("deny") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if sgfw.RuleAction(r.Verb) == sgfw.RULE_ACTION_ALLOW_TLSONLY { | ||||||
|  | 		re.checkTLS.SetActive(true) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if r.Sandbox == "" { | ||||||
|  | 		re.titleSandbox.SetNoShowAll(true) | ||||||
|  | 		re.titleSandbox.SetVisible(false) | ||||||
|  | 		re.labelSandbox.SetNoShowAll(true) | ||||||
|  | 		re.labelSandbox.SetVisible(false) | ||||||
|  | 	} else { | ||||||
|  | 		re.titleSandbox.SetVisible(true) | ||||||
|  | 		re.labelSandbox.SetNoShowAll(false) | ||||||
|  | 		re.labelSandbox.SetVisible(true) | ||||||
|  | 		re.labelSandbox.SetNoShowAll(false) | ||||||
|  | 		re.labelSandbox.SetText(r.Sandbox) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	re.btnPathChooser.SetNoShowAll(true) | ||||||
|  | 	re.btnPathChooser.SetVisible(false) | ||||||
|  | 	re.btnPathChooser.SetSensitive(false) | ||||||
|  | 	re.entryPath.SetNoShowAll(false) | ||||||
|  | 	re.entryPath.SetVisible(true) | ||||||
|  | 	re.entryPath.SetText(r.Path) | ||||||
|  | 
 | ||||||
|  | 	target := strings.Split(r.Target, ":") | ||||||
|  | 	if len(target) != 2 { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	re.hostEntry.SetText(target[0]) | ||||||
|  | 	re.portEntry.SetText(target[1]) | ||||||
|  | 
 | ||||||
|  | 	if r.UID > -1 { | ||||||
|  | 		re.comboUID.SetActiveID(strconv.FormatInt(int64(r.UID), 10)) | ||||||
|  | 	} | ||||||
|  | 	if r.GID > -1 { | ||||||
|  | 		re.comboGID.SetActiveID(strconv.FormatInt(int64(r.GID), 10)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if re.mode == DIALOG_MODE_EDIT { | ||||||
|  | 		re.comboScope.SetVisible(false) | ||||||
|  | 		re.comboScope.SetNoShowAll(true) | ||||||
|  | 		re.comboScope.SetSensitive(false) | ||||||
|  | 		re.labelScope.SetNoShowAll(false) | ||||||
|  | 		re.labelScope.SetVisible(true) | ||||||
|  | 		re.labelScope.SetText(strings.Title(strings.ToLower(sgfw.RuleModeString[sgfw.RuleMode(r.Mode)]))) | ||||||
|  | 	} | ||||||
|  | 	if re.mode == DIALOG_MODE_PROMPT || r.Mode == uint16(sgfw.RULE_MODE_PROCESS) { | ||||||
|  | 		re.titlePID.SetNoShowAll(false) | ||||||
|  | 		re.titlePID.SetVisible(true) | ||||||
|  | 		re.labelPID.SetNoShowAll(false) | ||||||
|  | 		re.labelPID.SetVisible(true) | ||||||
|  | 		pid := strconv.FormatUint(uint64(r.Pid), 10) | ||||||
|  | 		re.labelPID.SetText(pid) | ||||||
|  | 	} | ||||||
|  | 	if re.mode == DIALOG_MODE_SAVEAS { | ||||||
|  | 		re.comboScope.Remove(1) | ||||||
|  | 		re.comboScope.SetSensitive(false) | ||||||
|  | 	} | ||||||
|  | 	if re.mode == DIALOG_MODE_PROMPT { | ||||||
|  | 		re.entryOrigin.SetNoShowAll(false) | ||||||
|  | 		re.entryOrigin.SetVisible(true) | ||||||
|  | 		re.entryOrigin.SetSensitive(false) | ||||||
|  | 		re.entryOrigin.SetText(r.Origin) | ||||||
|  | 		re.labelOrigin.SetNoShowAll(false) | ||||||
|  | 		re.labelOrigin.SetVisible(true) | ||||||
|  | 		re.comboUID.SetSensitive(false) | ||||||
|  | 		re.comboGID.SetSensitive(false) | ||||||
|  | 		re.comboScope.SetActiveID(strconv.Itoa(int(sgfw.RuleModeValue[strings.ToUpper(re.row.rl.app.Config.DefaultAction)]))) | ||||||
|  | 		 | ||||||
|  | 		re.checkUID.SetNoShowAll(false) | ||||||
|  | 		re.checkUID.SetVisible(true) | ||||||
|  | 		re.checkUID.SetSensitive(true) | ||||||
|  | 		re.checkGID.SetNoShowAll(false) | ||||||
|  | 		re.checkGID.SetVisible(true) | ||||||
|  | 		re.checkGID.SetSensitive(true) | ||||||
|  | 
 | ||||||
|  | 		re.comboVerb.SetNoShowAll(true) | ||||||
|  | 		re.comboVerb.SetVisible(false) | ||||||
|  | 		re.comboVerb.SetSensitive(false) | ||||||
|  | 
 | ||||||
|  | 		re.setPromptButtons() | ||||||
|  | 		 | ||||||
|  | 		ctv := r.IsSocks | ||||||
|  | 		if !ctv { | ||||||
|  | 			re.checkTLS.SetSensitive(false) | ||||||
|  | 			re.checkTLS.SetActive(false) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if re.mode == DIALOG_MODE_INFO { | ||||||
|  | 		re.comboScope.SetActiveID(strconv.Itoa(int(r.Mode))) | ||||||
|  | 		re.comboScope.SetSensitive(false) | ||||||
|  | 		re.comboVerb.SetSensitive(false) | ||||||
|  | 		re.hostEntry.SetSensitive(false) | ||||||
|  | 		re.portEntry.SetSensitive(false) | ||||||
|  | 		re.comboUID.SetSensitive(false) | ||||||
|  | 		re.comboGID.SetSensitive(false) | ||||||
|  | 		re.checkUID.SetSensitive(false) | ||||||
|  | 		re.checkGID.SetSensitive(false) | ||||||
|  | 		re.comboProto.SetSensitive(false) | ||||||
|  | 		re.checkTLS.SetSensitive(false) | ||||||
|  | 		re.ok.SetNoShowAll(true) | ||||||
|  | 		re.ok.SetSensitive(false) | ||||||
|  | 		re.ok.SetVisible(false) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	re.onProtoChanged() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) setPromptButtons() { | ||||||
|  | 	re.allow.SetNoShowAll(false) | ||||||
|  | 	re.allow.SetVisible(true) | ||||||
|  | 	re.allow.SetSensitive(true) | ||||||
|  | 	re.ok.SetLabel("_Deny") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) toggleCheckTLS(val bool) { | ||||||
|  | 	if val && re.row.rule.IsSocks && re.mode != DIALOG_MODE_NEW  && re.mode != DIALOG_MODE_INFO { | ||||||
|  | 		re.checkTLS.SetSensitive(true) | ||||||
|  | 	} else { | ||||||
|  | 		re.checkTLS.SetSensitive(false) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) onProtoChanged() { | ||||||
|  | 	re.toggleCheckTLS( (re.comboProto.GetActiveID() == "tcp") ) | ||||||
|  | 	if re.comboProto.GetActiveID() == "icmp" { | ||||||
|  | 		re.titlePort.SetText("Code:") | ||||||
|  | 		re.portEntry.SetPlaceholderText("Code") | ||||||
|  | 	} else { | ||||||
|  | 		re.titlePort.SetText("Port:") | ||||||
|  | 		re.portEntry.SetPlaceholderText("Port") | ||||||
|  | 	} | ||||||
|  | 	re.onChanged() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) onVerbChanged() { | ||||||
|  | 	re.toggleCheckTLS( (re.comboVerb.GetActiveID() == "allow") ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) validateFields() bool { | ||||||
|  | 	id := re.comboVerb.GetActiveID() | ||||||
|  | 	if id != "allow" && id != "allow_tls" && id != "deny" { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	proto := re.comboProto.GetActiveID() | ||||||
|  | 	protos := []string{"", "tcp", "udp", "icmp"} | ||||||
|  | 	found := false | ||||||
|  | 	for _, p := range protos { | ||||||
|  | 		if proto == p { | ||||||
|  | 			found = true | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if !found { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	host, _ := re.hostEntry.GetText() | ||||||
|  | 	port, _ := re.portEntry.GetText() | ||||||
|  | 	if !isValidHost(host) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if !isValidPort(port, re.comboProto.GetActiveID()) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	if re.mode == DIALOG_MODE_NEW { | ||||||
|  | 		fp := re.btnPathChooser.FileChooser.GetFilename() | ||||||
|  | 		if fp == "" || !isExecutableFile(fp) { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func isExecutableFile(file string) bool { | ||||||
|  | 	fi, _ := os.Stat(file) | ||||||
|  | 	fm := fi.Mode() | ||||||
|  | 	perm := uint32(fm.Perm()) | ||||||
|  | 	return !( (perm&UserExecute == 0) && (perm&GroupExecute == 0) && (perm&OtherExecute == 0) ) | ||||||
|  | 	 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) onPortInsertText(entry *gtk.Entry, text string) { | ||||||
|  | 	current, _ := entry.GetText() | ||||||
|  | 	if current == "" && text == "*" { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	if current == "*" { | ||||||
|  | 		entry.StopEmission("insert-text") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	for _, c := range text { | ||||||
|  | 		if !unicode.IsDigit(c) { | ||||||
|  | 			entry.StopEmission("insert-text") | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) onChanged() { | ||||||
|  | 	valid := re.validateFields() | ||||||
|  | 	re.ok.SetSensitive(valid) | ||||||
|  | 	if re.mode == DIALOG_MODE_PROMPT { | ||||||
|  | 		re.allow.SetSensitive(valid) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) onPathSet(btnChooser *gtk.FileChooserButton) { | ||||||
|  | 	fp := btnChooser.FileChooser.GetFilename() | ||||||
|  | 	if !isExecutableFile(fp) { | ||||||
|  | 		warnDialog(&re.row.rl.app.win.Window, "%s", "File not an executable!") | ||||||
|  | 	} else { | ||||||
|  | 		btnChooser.SetTooltipText(fp) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) populateUID() { | ||||||
|  | 	for _, id := range re.row.rl.app.userIDs { | ||||||
|  | 		re.comboUID.Append(strconv.FormatInt(int64(id), 10), re.row.rl.app.userMap[id]) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (re *ruleNew) populateGID() { | ||||||
|  | 	for _, id := range re.row.rl.app.groupIDs { | ||||||
|  | 		re.comboGID.Append(strconv.FormatInt(int64(id), 10), re.row.rl.app.groupMap[id]) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -0,0 +1,17 @@ | |||||||
|  | .PHONY: generate | ||||||
|  | 
 | ||||||
|  | all: generate | ||||||
|  | 
 | ||||||
|  | gschemas.compiled: *.xml | ||||||
|  | 	glib-compile-schemas . | ||||||
|  | 
 | ||||||
|  | schemas.go: gschemas.compiled | ||||||
|  | 	ruby ./generate.rb | ||||||
|  | 
 | ||||||
|  | generate: touch schemas.go | ||||||
|  | 
 | ||||||
|  | touch: | ||||||
|  | 	ls *.xml | xargs -n1 touch | ||||||
|  | 
 | ||||||
|  | doctor: touch generate | ||||||
|  | 	git diff --exit-code . | ||||||
| @ -0,0 +1,31 @@ | |||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | 
 | ||||||
|  | <schemalist> | ||||||
|  |   <schema id="com.subgraph.Firewall.Settings"> | ||||||
|  |     <key name="window-height" type="u"> | ||||||
|  |       <default>0</default> | ||||||
|  |       <summary>Main window height</summary> | ||||||
|  |       <description>Preserved height of the main window</description> | ||||||
|  |     </key> | ||||||
|  |     <key name="window-width" type="u"> | ||||||
|  |       <default>0</default> | ||||||
|  |       <summary>Main window width</summary> | ||||||
|  |       <description>Preserved width of the main window</description> | ||||||
|  |     </key> | ||||||
|  |     <key name="window-top" type="u"> | ||||||
|  |       <default>0</default> | ||||||
|  |       <summary>Main window top</summary> | ||||||
|  |       <description>Preserved top coordinate of the main window</description> | ||||||
|  |     </key> | ||||||
|  |     <key name="window-left" type="u"> | ||||||
|  |       <default>0</default> | ||||||
|  |       <summary>Main window left</summary> | ||||||
|  |       <description>Preserved left coordinate of the main window</description> | ||||||
|  |     </key> | ||||||
|  |     <key name="prompt-toplevel" type="b"> | ||||||
|  |       <default>true</default> | ||||||
|  |       <summary>GTK Prompt As Top Level</summary> | ||||||
|  |       <description>Bring the prompt window as front most and sticky on new prompt requests</description> | ||||||
|  |     </key> | ||||||
|  |   </schema> | ||||||
|  | </schemalist> | ||||||
| @ -0,0 +1,23 @@ | |||||||
|  | package definitions | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"encoding/hex" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"path" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func fileContent() []byte { | ||||||
|  | 	decoded, _ := hex.DecodeString(schemaDefinition) | ||||||
|  | 	return decoded | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func writeSchemaToDir(dir string) { | ||||||
|  | 	ioutil.WriteFile(path.Join(dir, "gschemas.compiled"), fileContent(), 0664) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // SchemaInTempDir will create a new temporary directory and put the gsettings schema file in there. It is the callers responsibility to remove the directory
 | ||||||
|  | func SchemaInTempDir() string { | ||||||
|  | 	dir, _ := ioutil.TempDir("", "fw-settings-gschema") | ||||||
|  | 	writeSchemaToDir(dir) | ||||||
|  | 	return dir | ||||||
|  | } | ||||||
| @ -0,0 +1,19 @@ | |||||||
|  | #!/usr/bin/env ruby | ||||||
|  | 
 | ||||||
|  | def gen_go_file | ||||||
|  |   binary_definition = File.binread("gschemas.compiled") | ||||||
|  |   hex = binary_definition.each_byte.map { |b| "%02x" % b }.join | ||||||
|  | 
 | ||||||
|  |   File.open("schemas.go", "w") do |f| | ||||||
|  |     sliced = hex.chars.each_slice(80).map{ |s| s.join }.join "\"+\n\t\"" | ||||||
|  | 
 | ||||||
|  |     f.puts <<TEMPLATE | ||||||
|  | package definitions | ||||||
|  | 
 | ||||||
|  | const schemaDefinition = ""+ | ||||||
|  | \t"#{sliced}" | ||||||
|  | TEMPLATE | ||||||
|  |   end | ||||||
|  | end | ||||||
|  | 
 | ||||||
|  | gen_go_file | ||||||
											
												Binary file not shown.
											
										
									
								| @ -0,0 +1,14 @@ | |||||||
|  | package definitions | ||||||
|  | 
 | ||||||
|  | const schemaDefinition = ""+ | ||||||
|  | 	"4756617269616e740000000000000000180000005800000000000028020000000000000000000000"+ | ||||||
|  | 	"05150000ffffffff5800000000004c00580000005c00000091f48b72000000005c0000001e004800"+ | ||||||
|  | 	"7c0000002c01000001000000636f6d2e73756267726170682e4669726577616c6c2e53657474696e"+ | ||||||
|  | 	"67730000000000280600000000000000000000000200000003000000040000000400000055c38862"+ | ||||||
|  | 	"040000002c0100000b00760038010000400100001d8e1b7f04000000400100000a00760050010000"+ | ||||||
|  | 	"58010000ea646ab404000000580100000c0076006801000070010000c3c5731e0400000070010000"+ | ||||||
|  | 	"0d007600800100008801000005150000ffffffff8801000000004c00880100009c0100007ffe7dcf"+ | ||||||
|  | 	"040000009c0100000f007600b0010000b501000077696e646f772d6c656674000000000000287529"+ | ||||||
|  | 	"77696e646f772d746f70000000000000000000000028752977696e646f772d776964746800000000"+ | ||||||
|  | 	"000000000028752977696e646f772d68656967687400000000000000002875290500000003000000"+ | ||||||
|  | 	"00000000010000000200000070726f6d70742d746f706c6576656c00000000000100286229" | ||||||
| @ -0,0 +1,129 @@ | |||||||
|  | package settings | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"os" | ||||||
|  | 
 | ||||||
|  | 	"github.com/subgraph/fw-daemon/fw-settings/settings/definitions" | ||||||
|  | 	"github.com/gotk3/gotk3/glib" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | var cachedSchema *glib.SettingsSchemaSource | ||||||
|  | 
 | ||||||
|  | func getSchemaSource() *glib.SettingsSchemaSource { | ||||||
|  | 	if cachedSchema == nil { | ||||||
|  | 		dir := definitions.SchemaInTempDir() | ||||||
|  | 		defer os.Remove(dir) | ||||||
|  | 		cachedSchema = glib.SettingsSchemaSourceNewFromDirectory(dir, nil, true) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return cachedSchema | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func getSchema() *glib.SettingsSchema { | ||||||
|  | 	return getSchemaSource().Lookup("com.subgraph.Firewall.Settings", false) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func getDefaultSettings() *glib.Settings { | ||||||
|  | 	return glib.SettingsNewFull(getSchema(), nil, "/com/subgraph/firewall/settings/") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type Settings struct { | ||||||
|  | 	def, spec *glib.Settings | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func Init() *Settings { | ||||||
|  | 	s := &Settings{} | ||||||
|  | 	s.def = getDefaultSettings() | ||||||
|  | 	return s | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) settingsForGet(name string) *glib.Settings { | ||||||
|  | 	if s.spec != nil { | ||||||
|  | 		return s.spec | ||||||
|  | 	} | ||||||
|  | 	return s.def | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) settingsForSet() *glib.Settings { | ||||||
|  | 	if s.spec != nil { | ||||||
|  | 		return s.spec | ||||||
|  | 	} | ||||||
|  | 	return s.def | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) getBooleanSetting(name string) bool { | ||||||
|  | 	return s.settingsForGet(name).GetBoolean(name) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) setBooleanSetting(name string, val bool) { | ||||||
|  | 	sets := s.settingsForSet() | ||||||
|  | 	sets.SetBoolean(name, val) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) getIntegerSetting(name string) int { | ||||||
|  | 	return s.settingsForGet(name).GetInt(name) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) setIntegerSetting(name string, val int) { | ||||||
|  | 	sets := s.settingsForSet() | ||||||
|  | 	sets.SetInt(name, val) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | func (s *Settings) getUIntegerSetting(name string) uint { | ||||||
|  | 	return s.settingsForGet(name).GetUInt(name) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) setUIntegerSetting(name string, val uint) { | ||||||
|  | 	sets := s.settingsForSet() | ||||||
|  | 	sets.SetUInt(name, val) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) getStringSetting(name string) string { | ||||||
|  | 	return s.settingsForGet(name).GetString(name) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) setStringSetting(name string, val string) { | ||||||
|  | 	sets := s.settingsForSet() | ||||||
|  | 	sets.SetString(name, val) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) GetWindowHeight() uint { | ||||||
|  | 	return s.getUIntegerSetting("window-height") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) SetWindowHeight(v uint) { | ||||||
|  | 	s.setUIntegerSetting("window-height", v) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) GetWindowWidth() uint { | ||||||
|  | 	return s.getUIntegerSetting("window-width") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) SetWindowWidth(v uint) { | ||||||
|  | 	s.setUIntegerSetting("window-width", v) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) GetWindowTop() uint { | ||||||
|  | 	return s.getUIntegerSetting("window-top") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) SetWindowTop(v uint) { | ||||||
|  | 	s.setUIntegerSetting("window-top", v) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) GetWindowLeft() uint { | ||||||
|  | 	return s.getUIntegerSetting("window-left") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) SetWindowLeft(v uint) { | ||||||
|  | 	s.setUIntegerSetting("window-left", v) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) GetToplevelPrompt() bool { | ||||||
|  | 	return s.getBooleanSetting("prompt-toplevel") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (s *Settings) SetToplevelPrompt(v bool) { | ||||||
|  | 	s.setBooleanSetting("prompt-toplevel", v) | ||||||
|  | } | ||||||
| @ -0,0 +1,29 @@ | |||||||
|  | // +build go1.8
 | ||||||
|  | 
 | ||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"sort" | ||||||
|  | 
 | ||||||
|  | 	"github.com/subgraph/fw-daemon/sgfw" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func (rl *ruleList) sortRules(rules []sgfw.DbusRule) []sgfw.DbusRule { | ||||||
|  | 	sort.SliceStable(rules, func(i, j int) bool { | ||||||
|  | 		//sgfw.RuleActionString[sgfw.RuleAction(rules[i].Verb)] 
 | ||||||
|  | 		//sgfw.RuleActionString[sgfw.RuleAction(rules[j].Verb)] 
 | ||||||
|  | 		in := rules[i].Sandbox + rules[i].App + rules[i].Target | ||||||
|  | 		jn := rules[j].Sandbox + rules[j].App + rules[j].Target | ||||||
|  | 		order := []string{in,jn} | ||||||
|  | 		sort.Strings(order) | ||||||
|  | 		if rules[i].App == rules[j].App && rules[i].Sandbox == rules[j].Sandbox { | ||||||
|  | 			if sgfw.RuleAction(rules[i].Verb) == sgfw.RULE_ACTION_DENY || sgfw.RuleAction(rules[j].Verb) == sgfw.RULE_ACTION_DENY { | ||||||
|  | 				if rules[i].Verb != rules[j].Verb { | ||||||
|  | 					return (sgfw.RuleAction(rules[i].Verb) == sgfw.RULE_ACTION_DENY) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return (order[0] == in) | ||||||
|  | 	}) | ||||||
|  | 	return rules | ||||||
|  | } | ||||||
| @ -0,0 +1,160 @@ | |||||||
|  | // +build !go1.8
 | ||||||
|  | 
 | ||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"sort" | ||||||
|  | 	"reflect" | ||||||
|  | 
 | ||||||
|  | 	"github.com/subgraph/fw-daemon/sgfw" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // Copied from golang 1.8 sources
 | ||||||
|  | // START sort.SliceStable
 | ||||||
|  | type lessSwap struct { | ||||||
|  | 	Less func(i, j int) bool | ||||||
|  | 	Swap func(i, j int) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func SliceStable(slice interface{}, less func(i, j int) bool) { | ||||||
|  | 	rv := reflect.ValueOf(slice) | ||||||
|  | 	swap := reflect.Swapper(slice) | ||||||
|  | 	stable_func(lessSwap{less, swap}, rv.Len()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func symMerge_func(data lessSwap, a, m, b int) { | ||||||
|  | 	if m-a == 1 { | ||||||
|  | 		i := m | ||||||
|  | 		j := b | ||||||
|  | 		for i < j { | ||||||
|  | 			h := int(uint(i+j) >> 1) | ||||||
|  | 			if data.Less(h, a) { | ||||||
|  | 				i = h + 1 | ||||||
|  | 			} else { | ||||||
|  | 				j = h | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		for k := a; k < i-1; k++ { | ||||||
|  | 			data.Swap(k, k+1) | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	if b-m == 1 { | ||||||
|  | 		i := a | ||||||
|  | 		j := m | ||||||
|  | 		for i < j { | ||||||
|  | 			h := int(uint(i+j) >> 1) | ||||||
|  | 			if !data.Less(m, h) { | ||||||
|  | 				i = h + 1 | ||||||
|  | 			} else { | ||||||
|  | 				j = h | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		for k := m; k > i; k-- { | ||||||
|  | 			data.Swap(k, k-1) | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	mid := int(uint(a+b) >> 1) | ||||||
|  | 	n := mid + m | ||||||
|  | 	var start, r int | ||||||
|  | 	if m > mid { | ||||||
|  | 		start = n - b | ||||||
|  | 		r = mid | ||||||
|  | 	} else { | ||||||
|  | 		start = a | ||||||
|  | 		r = m | ||||||
|  | 	} | ||||||
|  | 	p := n - 1 | ||||||
|  | 	for start < r { | ||||||
|  | 		c := int(uint(start+r) >> 1) | ||||||
|  | 		if !data.Less(p-c, c) { | ||||||
|  | 			start = c + 1 | ||||||
|  | 		} else { | ||||||
|  | 			r = c | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	end := n - start | ||||||
|  | 	if start < m && m < end { | ||||||
|  | 		rotate_func(data, start, m, end) | ||||||
|  | 	} | ||||||
|  | 	if a < start && start < mid { | ||||||
|  | 		symMerge_func(data, a, start, mid) | ||||||
|  | 	} | ||||||
|  | 	if mid < end && end < b { | ||||||
|  | 		symMerge_func(data, mid, end, b) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func swapRange_func(data lessSwap, a, b, n int) { | ||||||
|  | 	for i := 0; i < n; i++ { | ||||||
|  | 		data.Swap(a+i, b+i) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func rotate_func(data lessSwap, a, m, b int) { | ||||||
|  | 	i := m - a | ||||||
|  | 	j := b - m | ||||||
|  | 	for i != j { | ||||||
|  | 		if i > j { | ||||||
|  | 			swapRange_func(data, m-i, m, j) | ||||||
|  | 			i -= j | ||||||
|  | 		} else { | ||||||
|  | 			swapRange_func(data, m-i, m+j-i, i) | ||||||
|  | 			j -= i | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	swapRange_func(data, m-i, m, i) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func insertionSort_func(data lessSwap, a, b int) { | ||||||
|  | 	for i := a + 1; i < b; i++ { | ||||||
|  | 		for j := i; j > a && data.Less(j, j-1); j-- { | ||||||
|  | 			data.Swap(j, j-1) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Auto-generated variant of sort.go:stable
 | ||||||
|  | func stable_func(data lessSwap, n int) { | ||||||
|  | 	blockSize := 20 | ||||||
|  | 	a, b := 0, blockSize | ||||||
|  | 	for b <= n { | ||||||
|  | 		insertionSort_func(data, a, b) | ||||||
|  | 		a = b | ||||||
|  | 		b += blockSize | ||||||
|  | 	} | ||||||
|  | 	insertionSort_func(data, a, n) | ||||||
|  | 	for blockSize < n { | ||||||
|  | 		a, b = 0, 2*blockSize | ||||||
|  | 		for b <= n { | ||||||
|  | 			symMerge_func(data, a, a+blockSize, b) | ||||||
|  | 			a = b | ||||||
|  | 			b += 2 * blockSize | ||||||
|  | 		} | ||||||
|  | 		if m := a + blockSize; m < n { | ||||||
|  | 			symMerge_func(data, a, m, n) | ||||||
|  | 		} | ||||||
|  | 		blockSize *= 2 | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | // END sort.SliceStable
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | func (rl *ruleList) sortRules(rules []sgfw.DbusRule) []sgfw.DbusRule { | ||||||
|  | 	SliceStable(rules, func(i, j int) bool { | ||||||
|  | 		in := rules[i].Sandbox + rules[i].App + rules[i].Target | ||||||
|  | 		jn := rules[j].Sandbox + rules[j].App + rules[j].Target | ||||||
|  | 		order := []string{in,jn} | ||||||
|  | 		sort.Strings(order) | ||||||
|  | 		if rules[i].App == rules[j].App && rules[i].Sandbox == rules[j].Sandbox { | ||||||
|  | 			if sgfw.RuleAction(rules[i].Verb) == sgfw.RULE_ACTION_DENY || sgfw.RuleAction(rules[j].Verb) == sgfw.RULE_ACTION_DENY { | ||||||
|  | 				if rules[i].Verb != rules[j].Verb { | ||||||
|  | 					return (sgfw.RuleAction(rules[i].Verb) == sgfw.RULE_ACTION_DENY) | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return (order[0] == in) | ||||||
|  | 	}) | ||||||
|  | 	return rules | ||||||
|  | } | ||||||
| @ -0,0 +1,12 @@ | |||||||
|  | [Desktop Entry] | ||||||
|  | Name=Subgraph Firewall Settings | ||||||
|  | Comment=Launch the Subgraph Firewall Settings | ||||||
|  | Keywords=Network;Security;Settings;Subgraph;Firewall;fwdaemon; | ||||||
|  | TryExec=fw-settings | ||||||
|  | Exec=fw-settings | ||||||
|  | Terminal=false | ||||||
|  | Type=Application | ||||||
|  | Icon=security-medium | ||||||
|  | Categories=Security;Settings; | ||||||
|  | X-Desktop-File-Install-Version=0.23 | ||||||
|  | DBusActivatable=true | ||||||
| @ -1,4 +1,5 @@ | |||||||
| [D-BUS Service] | [D-BUS Service] | ||||||
| Name=com.subgraph.FirewallPrompt | Name=com.subgraph.FirewallPrompt | ||||||
| Exec=/usr/bin/fw-prompt -display :0 | Exec=/usr/bin/fw-settings --prompt-only --display :0 | ||||||
| User=user | User=user | ||||||
|  | #SystemdService=fw-settings.service | ||||||
|  | |||||||
| @ -0,0 +1,31 @@ | |||||||
|  | language: go | ||||||
|  | 
 | ||||||
|  | go: | ||||||
|  |   - 1.4 | ||||||
|  |   - 1.5 | ||||||
|  |   - 1.6 | ||||||
|  |   - tip | ||||||
|  | 
 | ||||||
|  | env: | ||||||
|  |   - GOARCH=amd64 | ||||||
|  | 
 | ||||||
|  | sudo: required | ||||||
|  | dist: trusty | ||||||
|  | 
 | ||||||
|  | before_install: | ||||||
|  |   - sudo apt-get update -qq | ||||||
|  |   - sudo apt-get install -qq -y gtk+3.0 libgtk-3-dev | ||||||
|  |   - sudo apt-get install -qq -y xvfb | ||||||
|  |   - "export DISPLAY=:99.0" | ||||||
|  |   - sudo /usr/bin/Xvfb $DISPLAY 2>1 > /dev/null & | ||||||
|  |   - "export GTK_VERSION=$(pkg-config --modversion gtk+-3.0 | tr . _| cut -d '_' -f 1-2)" | ||||||
|  |   - "export Glib_VERSION=$(pkg-config --modversion glib-2.0)" | ||||||
|  |   - "export Cairo_VERSION=$(pkg-config --modversion cairo)" | ||||||
|  |   - "export Pango_VERSION=$(pkg-config --modversion pango)" | ||||||
|  |   - echo "GTK version ${GTK_VERSION} (Glib ${Glib_VERSION}, Cairo ${Cairo_VERSION}, Pango ${Pango_VERSION})" | ||||||
|  | 
 | ||||||
|  | install: | ||||||
|  |   - go get -t -tags "gtk_${GTK_VERSION}" github.com/gotk3/gotk3/... | ||||||
|  | 
 | ||||||
|  | script: | ||||||
|  |   - go test -tags "gtk_${GTK_VERSION}" ./... | ||||||
| @ -0,0 +1,210 @@ | |||||||
|  | package glib | ||||||
|  | 
 | ||||||
|  | // #cgo pkg-config: glib-2.0 gobject-2.0
 | ||||||
|  | // #include <gio/gio.h>
 | ||||||
|  | // #include <glib.h>
 | ||||||
|  | // #include <glib-object.h>
 | ||||||
|  | // #include "glib.go.h"
 | ||||||
|  | import "C" | ||||||
|  | import "unsafe" | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  | type ActionType int | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	ACTION_TYPE_ ActionType = C.GTK_ | ||||||
|  | ) | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | // ActionGroup is a representation of GActionGroup.
 | ||||||
|  | type ActionGroup struct { | ||||||
|  | 	*Object | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native() returns a pointer to the underlying GActionGroup.
 | ||||||
|  | func (v *ActionGroup) native() *C.GActionGroup { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return C.toGActionGroup(unsafe.Pointer(v.GObject)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *ActionGroup) Native() uintptr { | ||||||
|  | 	return uintptr(unsafe.Pointer(v.native())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalActionGroup(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	return wrapActionGroup(wrapObject(unsafe.Pointer(c))), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapActionGroup(obj *Object) *ActionGroup { | ||||||
|  | 	return &ActionGroup{obj} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ActionGroup is a representation of GActionGroup.
 | ||||||
|  | type Action struct { | ||||||
|  | 	*Object | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native() returns a pointer to the underlying GActionGroup.
 | ||||||
|  | func (v *Action) native() *C.GAction { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return C.toGAction(unsafe.Pointer(v.GObject)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *Action) Native() uintptr { | ||||||
|  | 	return uintptr(unsafe.Pointer(v.native())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalAction(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	return wrapAction(wrapObject(unsafe.Pointer(c))), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapAction(obj *Object) *Action { | ||||||
|  | 	return &Action{obj} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *Action) NameIsValid(name string) bool { | ||||||
|  | 	cstr := (*C.gchar)(C.CString(name)) | ||||||
|  | 	defer C.free(unsafe.Pointer(cstr)) | ||||||
|  | 	return gobool(C.g_action_name_is_valid(cstr)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *Action) GetName() string { | ||||||
|  | 	c := C.g_action_get_name(v.native()) | ||||||
|  | 
 | ||||||
|  | 	return C.GoString((*C.char)(c)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // XXX: g_action_get_parameter_type
 | ||||||
|  | // XXX: g_action_get_state_type
 | ||||||
|  | // XXX: g_action_get_state_hint
 | ||||||
|  | 
 | ||||||
|  | func (v *Action) GetEnabled() bool { | ||||||
|  | 	return gobool(C.g_action_get_enabled(v.native())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // XXX: g_action_get_state
 | ||||||
|  | // XXX: g_action_change_state
 | ||||||
|  | 
 | ||||||
|  | func (v *Action) Activate(p *Variant) { | ||||||
|  | 	C.g_action_activate(v.native(), p.native()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // XXX: g_action_parse_detailed_name
 | ||||||
|  | // XXX: g_action_print_detailed_name
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // ActionGroup is a representation of GActionGroup.
 | ||||||
|  | type SimpleAction struct { | ||||||
|  | 	Action | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native() returns a pointer to the underlying GActionGroup.
 | ||||||
|  | func (v *SimpleAction) native() *C.GSimpleAction { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return C.toGSimpleAction(unsafe.Pointer(v.GObject)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *SimpleAction) Native() uintptr { | ||||||
|  | 	return uintptr(unsafe.Pointer(v.native())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalSimpleAction(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	return wrapSimpleAction(wrapObject(unsafe.Pointer(c))), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapSimpleAction(obj *Object) *SimpleAction { | ||||||
|  | 	return &SimpleAction{*wrapAction(obj)} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func SimpleActionNew(name string, t *VariantType) *SimpleAction { | ||||||
|  | 	cstr := (*C.gchar)(C.CString(name)) | ||||||
|  | 	defer C.free(unsafe.Pointer(cstr)) | ||||||
|  | 	c := C.g_simple_action_new(cstr, t.native()) | ||||||
|  | 	if c == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return wrapSimpleAction(wrapObject(unsafe.Pointer(c))) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *SimpleAction) NewStateful(name string, t *VariantType, state *Variant) *SimpleAction { | ||||||
|  | 	cstr := (*C.gchar)(C.CString(name)) | ||||||
|  | 	defer C.free(unsafe.Pointer(cstr)) | ||||||
|  | 	c := C.g_simple_action_new_stateful(cstr, t.native(), state.native()) | ||||||
|  | 	if c == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return wrapSimpleAction(wrapObject(unsafe.Pointer(c))) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *SimpleAction) SetEnabled(state bool) { | ||||||
|  | 	C.g_simple_action_set_enabled(v.native(), gbool(state)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *SimpleAction) SetState(state Variant) { | ||||||
|  | 	C.g_simple_action_set_state(v.native(), state.ToGVariant()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // XXX: g_simple_action_set_state_hint
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | // ActionGroup is a representation of GActionGroup.
 | ||||||
|  | type ActionMap struct { | ||||||
|  | 	*Object | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native() returns a pointer to the underlying GActionGroup.
 | ||||||
|  | func (v *ActionMap) native() *C.GActionMap { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return C.toGActionMap(unsafe.Pointer(v.GObject)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *ActionMap) Native() uintptr { | ||||||
|  | 	return uintptr(unsafe.Pointer(v.native())) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalActionMap(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	return wrapActionMap(wrapObject(unsafe.Pointer(c))), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapActionMap(obj *Object) *ActionMap { | ||||||
|  | 	return &ActionMap{obj} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *ActionMap) LookupAction(name string) *Action { | ||||||
|  | 	cstr := (*C.gchar)(C.CString(name)) | ||||||
|  | 	defer C.free(unsafe.Pointer(cstr)) | ||||||
|  | 	c := C.g_action_map_lookup_action(v.native(), cstr) | ||||||
|  | 	if c == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return wrapAction(wrapObject(unsafe.Pointer(c))) | ||||||
|  | } | ||||||
|  | /* | ||||||
|  | // Requires GActionEntry
 | ||||||
|  | func (v *ActionMap) AddActionEntries() { | ||||||
|  | 	 | ||||||
|  | } | ||||||
|  | */ | ||||||
|  | func (v *ActionMap) AddAction(action *Action) { | ||||||
|  | 	C.g_action_map_add_action(v.native(), action.native()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (v *ActionMap) RemoveAction(name string) { | ||||||
|  | 	cstr := (*C.gchar)(C.CString(name)) | ||||||
|  | 	defer C.free(unsafe.Pointer(cstr)) | ||||||
|  | 	C.g_action_map_remove_action(v.native(), cstr) | ||||||
|  | } | ||||||
|  | 
 | ||||||
											
												
													File diff suppressed because it is too large
													Load Diff
												
											
										
									
								| @ -0,0 +1,26 @@ | |||||||
|  | // +build !gtk_3_6,!gtk_3_8
 | ||||||
|  | 
 | ||||||
|  | package gtk | ||||||
|  | 
 | ||||||
|  | // #cgo pkg-config: gtk+-3.0
 | ||||||
|  | // #include <gtk/gtk.h>
 | ||||||
|  | import "C" | ||||||
|  | import ( | ||||||
|  | 	"unsafe" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gotk3/gotk3/glib" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | //export goListBoxFilterFuncs
 | ||||||
|  | func goListBoxFilterFuncs(row *C.GtkListBoxRow, userData C.gpointer) C.gboolean { | ||||||
|  | 	id := int(uintptr(userData)) | ||||||
|  | 
 | ||||||
|  | 	listBoxFilterFuncRegistry.Lock() | ||||||
|  | 	r := listBoxFilterFuncRegistry.m[id] | ||||||
|  | 	// TODO: figure out a way to determine when we can clean up
 | ||||||
|  | 	//delete(printSettingsCallbackRegistry.m, id)
 | ||||||
|  | 	listBoxFilterFuncRegistry.Unlock() | ||||||
|  | 
 | ||||||
|  | 	return gbool(r.fn(wrapListBoxRow(glib.Take(unsafe.Pointer(row))), r.userData)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| @ -0,0 +1,10 @@ | |||||||
|  | // +build !gtk_3_6,!gtk_3_8,!gtk_3_10,!gtk_3_12,!gtk_3_14,!gtk_3_16,!gtk_3_18,gtk_3_20
 | ||||||
|  | 
 | ||||||
|  | // See: https://developer.gnome.org/gtk3/3.20/api-index-3-20.html
 | ||||||
|  | 
 | ||||||
|  | package gtk | ||||||
|  | 
 | ||||||
|  | // #cgo pkg-config: gtk+-3.0
 | ||||||
|  | // #include <gtk/gtk.h>
 | ||||||
|  | // #include "gtk_since_3_20.go.h"
 | ||||||
|  | import "C" | ||||||
| @ -0,0 +1,33 @@ | |||||||
|  | // Same copyright and license as the rest of the files in this project
 | ||||||
|  | // This file contains accelerator related functions and structures
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <string.h> | ||||||
|  | 
 | ||||||
|  | static GtkShortcutsWindow * | ||||||
|  | toGtkShortcutsWindow(void *p) | ||||||
|  | { | ||||||
|  | 	return (GTK_SHORTCUTS_WINDOW(p)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static GtkShortcutsSection * | ||||||
|  | toGtkShortcutsSection(void *p) | ||||||
|  | { | ||||||
|  | 	return (GTK_SHORTCUTS_SECTION(p)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static GtkShortcutsGroup * | ||||||
|  | toGtkShortcutsGroup(void *p) | ||||||
|  | { | ||||||
|  | 	return (GTK_SHORTCUTS_GROUP(p)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static GtkShortcutsShortcut * | ||||||
|  | toGtkShortcutsShortcut(void *p) | ||||||
|  | { | ||||||
|  | 	return (GTK_SHORTCUTS_SHORTCUT(p)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| @ -0,0 +1,147 @@ | |||||||
|  | // +build !gtk_3_6,!gtk_3_8,!gtk_3_10,!gtk_3_12,!gtk_3_14, !gtk_3_16, !gtk_3_18
 | ||||||
|  | 
 | ||||||
|  | // See: https://developer.gnome.org/gtk3/3.20/api-index-3-20.html
 | ||||||
|  | 
 | ||||||
|  | package gtk | ||||||
|  | 
 | ||||||
|  | // #include <gtk/gtk.h>
 | ||||||
|  | // #include "gtk.go.h"
 | ||||||
|  | // #include "gtk_since_3_20.go.h"
 | ||||||
|  | import "C" | ||||||
|  | import ( | ||||||
|  | 	"unsafe" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gotk3/gotk3/glib" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func init() { | ||||||
|  | 	tm := []glib.TypeMarshaler{ | ||||||
|  | 		{glib.Type(C.gtk_shortcuts_window_get_type()), marshalShortcutsWindow}, | ||||||
|  | 		{glib.Type(C.gtk_shortcuts_section_get_type()), marshalShortcutsSection}, | ||||||
|  | 		{glib.Type(C.gtk_shortcuts_group_get_type()), marshalShortcutsGroup}, | ||||||
|  | 		{glib.Type(C.gtk_shortcuts_shortcut_get_type()), marshalShortcutsShortcut}, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	glib.RegisterGValueMarshalers(tm) | ||||||
|  | 
 | ||||||
|  | 	WrapMap["GtkShortcutsWindow"] = wrapShortcutsWindow | ||||||
|  | 	WrapMap["GtkShortcutsSection"] = wrapShortcutsSection | ||||||
|  | 	WrapMap["GtkShortcutsGroup"] = wrapShortcutsGroup | ||||||
|  | 	WrapMap["GtkShortcutsShortcut"] = wrapShortcutsShortcut | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * GtkShortcutsWindow | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | // ShortcutsWindow is a representation of GTK's GtkShortcutsWindow.
 | ||||||
|  | type ShortcutsWindow struct { | ||||||
|  | 	Window | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native returns a pointer to the underlying GtkAboutDialog.
 | ||||||
|  | func (v *ShortcutsWindow) native() *C.GtkShortcutsWindow { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	p := unsafe.Pointer(v.GObject) | ||||||
|  | 	return C.toGtkShortcutsWindow(p) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalShortcutsWindow(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	obj := glib.Take(unsafe.Pointer(c)) | ||||||
|  | 	return wrapShortcutsWindow(obj), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapShortcutsWindow(obj *glib.Object) *ShortcutsWindow { | ||||||
|  | 	return &ShortcutsWindow{Window{Bin{Container{Widget{glib.InitiallyUnowned{obj}}}}}} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * GtkShortcutsSection | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | // ShortcutsWindow is a representation of GTK's GtkShortcutsWindow.
 | ||||||
|  | type ShortcutsSection struct { | ||||||
|  | 	Box | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native returns a pointer to the underlying GtkAboutDialog.
 | ||||||
|  | func (v *ShortcutsSection) native() *C.GtkShortcutsSection { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	p := unsafe.Pointer(v.GObject) | ||||||
|  | 	return C.toGtkShortcutsSection(p) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalShortcutsSection(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	obj := glib.Take(unsafe.Pointer(c)) | ||||||
|  | 	return wrapShortcutsSection(obj), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapShortcutsSection(obj *glib.Object) *ShortcutsSection { | ||||||
|  | 	return &ShortcutsSection{Box{Container{Widget{glib.InitiallyUnowned{obj}}}}} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * GtkShortcutsSection | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | // ShortcutsWindow is a representation of GTK's GtkShortcutsWindow.
 | ||||||
|  | type ShortcutsGroup struct { | ||||||
|  | 	Box | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native returns a pointer to the underlying GtkAboutDialog.
 | ||||||
|  | func (v *ShortcutsGroup) native() *C.GtkShortcutsGroup { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	p := unsafe.Pointer(v.GObject) | ||||||
|  | 	return C.toGtkShortcutsGroup(p) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalShortcutsGroup(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	obj := glib.Take(unsafe.Pointer(c)) | ||||||
|  | 	return wrapShortcutsGroup(obj), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapShortcutsGroup(obj *glib.Object) *ShortcutsGroup { | ||||||
|  | 	return &ShortcutsGroup{Box{Container{Widget{glib.InitiallyUnowned{obj}}}}} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * GtkShortcutsShortcut | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | // ShortcutsWindow is a representation of GTK's GtkShortcutsWindow.
 | ||||||
|  | type ShortcutsShortcut struct { | ||||||
|  | 	Box | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // native returns a pointer to the underlying GtkAboutDialog.
 | ||||||
|  | func (v *ShortcutsShortcut) native() *C.GtkShortcutsShortcut { | ||||||
|  | 	if v == nil || v.GObject == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	p := unsafe.Pointer(v.GObject) | ||||||
|  | 	return C.toGtkShortcutsShortcut(p) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func marshalShortcutsShortcut(p uintptr) (interface{}, error) { | ||||||
|  | 	c := C.g_value_get_object((*C.GValue)(unsafe.Pointer(p))) | ||||||
|  | 	obj := glib.Take(unsafe.Pointer(c)) | ||||||
|  | 	return wrapShortcutsShortcut(obj), nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func wrapShortcutsShortcut(obj *glib.Object) *ShortcutsShortcut { | ||||||
|  | 	return &ShortcutsShortcut{Box{Container{Widget{glib.InitiallyUnowned{obj}}}}} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| @ -0,0 +1,29 @@ | |||||||
|  | language: go | ||||||
|  | 
 | ||||||
|  | build_image: shippableimages/ubuntu1404_go | ||||||
|  | 
 | ||||||
|  | go: | ||||||
|  |   - tip | ||||||
|  | 
 | ||||||
|  | before_install: | ||||||
|  |   - sudo apt-get update -y | ||||||
|  |   - sudo apt-get install -y libglib2.0-dev libcairo2-dev libgtk-3-dev xvfb | ||||||
|  |   - source $HOME/.gvm/scripts/gvm; | ||||||
|  |   - if [[ $SHIPPABLE_GO_VERSION == "tip" ]]; then gvm install tip; gvm use tip; fi | ||||||
|  |   - if [[ $SHIPPABLE_GO_VERSION == *release* ]]; then gvm install release; gvm use release; fi | ||||||
|  |   - if [[ $SHIPPABLE_GO_VERSION =~ [0-9].[0-9] ]]; then gvm install go$SHIPPABLE_GO_VERSION; gvm use go$SHIPPABLE_GO_VERSION; fi | ||||||
|  |   - export GOPATH=$SHIPPABLE_GOPATH | ||||||
|  |   - go get github.com/t-yuki/gocover-cobertura | ||||||
|  |   - go get github.com/onsi/gomega | ||||||
|  |   - go get github.com/onsi/ginkgo | ||||||
|  |   - go get code.google.com/p/go.tools/cmd/cover | ||||||
|  |   - "export DISPLAY=:99.0" | ||||||
|  |   - /usr/bin/Xvfb :99 & | ||||||
|  |    | ||||||
|  | install: | ||||||
|  |   - go build -tags gtk_3_10 -v ./... | ||||||
|  | 
 | ||||||
|  | script: | ||||||
|  |   - go test -tags gtk_3_10 -coverprofile=coverage_gtk.txt -covermode count ./gtk | ||||||
|  |   - go test -tags gtk_3_10 ./glib | ||||||
|  |   - $GOPATH/bin/gocover-cobertura < coverage_gtk.txt > shippable/codecoverage/coverage_gtk.xml | ||||||
					Loading…
					
					
				
		Reference in new issue
	
	 xSmurf
						xSmurf